diff --git a/weixin4j-wxa/src/main/java/com/foxinmy/weixin4j/wxa/WeixinAppFacade.java b/weixin4j-wxa/src/main/java/com/foxinmy/weixin4j/wxa/WeixinAppFacade.java
index c36f6a88..70984cb2 100644
--- a/weixin4j-wxa/src/main/java/com/foxinmy/weixin4j/wxa/WeixinAppFacade.java
+++ b/weixin4j-wxa/src/main/java/com/foxinmy/weixin4j/wxa/WeixinAppFacade.java
@@ -12,6 +12,7 @@ import com.foxinmy.weixin4j.token.TokenManager;
import com.foxinmy.weixin4j.wxa.api.CustomMessageApi;
import com.foxinmy.weixin4j.wxa.api.LoginApi;
import com.foxinmy.weixin4j.wxa.api.QrCodeApi;
+import com.foxinmy.weixin4j.wxa.api.SecCheckApi;
import com.foxinmy.weixin4j.wxa.api.TemplateApi;
import com.foxinmy.weixin4j.wxa.api.TemplateMessageApi;
@@ -27,6 +28,7 @@ public class WeixinAppFacade {
private final TemplateApi templateApi;
private final TemplateMessageApi templateMessageApi;
private final CustomMessageApi customMessageApi;
+ private final SecCheckApi secCheckApi;
/**
* Constructs {@link WeixinAppFacade} using {@link FileCacheStorager}.
@@ -108,6 +110,7 @@ public class WeixinAppFacade {
this.templateApi = new TemplateApi(tokenManager, properties);
this.templateMessageApi = new TemplateMessageApi(tokenManager, properties);
this.customMessageApi = new CustomMessageApi(tokenManager, properties);
+ this.secCheckApi = new SecCheckApi(tokenManager, properties);
}
/**
@@ -155,4 +158,14 @@ public class WeixinAppFacade {
return customMessageApi;
}
+ /**
+ * 获取内容安全相关的 API。
+ *
+ * @return 内容安全相关的 API。
+ * @since 1.9
+ */
+ public SecCheckApi getSecCheckApi() {
+ return secCheckApi;
+ }
+
}
diff --git a/weixin4j-wxa/src/main/java/com/foxinmy/weixin4j/wxa/api/SecCheckApi.java b/weixin4j-wxa/src/main/java/com/foxinmy/weixin4j/wxa/api/SecCheckApi.java
new file mode 100644
index 00000000..aa94753d
--- /dev/null
+++ b/weixin4j-wxa/src/main/java/com/foxinmy/weixin4j/wxa/api/SecCheckApi.java
@@ -0,0 +1,98 @@
+package com.foxinmy.weixin4j.wxa.api;
+
+import java.awt.Image;
+import java.awt.image.BufferedImage;
+import java.io.ByteArrayInputStream;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.io.InputStream;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.Properties;
+
+import javax.imageio.ImageIO;
+
+import com.foxinmy.weixin4j.exception.WeixinException;
+import com.foxinmy.weixin4j.http.apache.content.InputStreamBody;
+import com.foxinmy.weixin4j.http.apache.mime.FormBodyPart;
+import com.foxinmy.weixin4j.http.weixin.WeixinResponse;
+import com.foxinmy.weixin4j.token.TokenManager;
+
+/**
+ * 违法违规内容检查。
+ *
+ * @since 1.9
+ */
+public class SecCheckApi extends TokenManagerApi {
+
+ public SecCheckApi(TokenManager tokenManager) {
+ super(tokenManager);
+ }
+
+ public SecCheckApi(TokenManager tokenManager, Properties properties) {
+ super(tokenManager, properties);
+ }
+
+ private InputStream scale(InputStream inputStream, int maxWidth, int maxHeight) throws IOException {
+ final BufferedImage srcBufferedImage = ImageIO.read(inputStream);
+
+ final int srcWidth = srcBufferedImage.getWidth();
+ final int srcHeight = srcBufferedImage.getHeight();
+ final float scale = Math.min((float) maxWidth / (float) srcWidth, (float) maxHeight / (float) srcHeight);
+
+ final BufferedImage scaledBufferedImage;
+ if (scale < 1F) {
+ final int width = (int) (srcWidth * scale);
+ final int height = (int) (srcHeight * scale);
+ final Image scaledImage = srcBufferedImage.getScaledInstance(width, height, Image.SCALE_DEFAULT);
+
+ scaledBufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
+ scaledBufferedImage.getGraphics().drawImage(scaledImage, 0, 0 , null);
+ } else {
+ scaledBufferedImage = srcBufferedImage;
+ }
+
+ final ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
+ ImageIO.write(scaledBufferedImage, "png", outputStream);
+ return new ByteArrayInputStream(outputStream.toByteArray());
+ }
+
+ /**
+ * 校验一张图片是否含有违法违规内容。
+ *
+ * @param inputStream the image input stream.
+ * @throws WeixinException indicates getting access token failed, or the content is risky.
+ * @see 校验一张图片是否含有违法违规内容
+ */
+ public void imgSecCheck(InputStream inputStream) throws WeixinException {
+ final String imgSecCheckUri = this.getAccessTokenRequestUri("wxa_img_sec_check");
+ final InputStreamBody inputStreamBody = new InputStreamBody(inputStream, "media");
+ final FormBodyPart formBodyPart = new FormBodyPart("media", inputStreamBody);
+ final WeixinResponse response = weixinExecutor.post(imgSecCheckUri, formBodyPart);
+ final WxaApiResult r = response.getAsObject(WxaApiResult.TYPE_REFERENCE);
+ r.checkErrCode();
+ }
+
+ public void imgSecCheck(InputStream inputStream, int maxWidth, int maxHeight) throws WeixinException {
+ try {
+ this.imgSecCheck(this.scale(inputStream, maxWidth, maxHeight));
+ } catch (IOException e) {
+ throw new WeixinException(e);
+ }
+ }
+
+ /**
+ * 检查一段文本是否含有违法违规内容。
+ *
+ * @param content 要检测的文本内容,长度不超过 500KB,编码格式为 UTF-8。
+ * @throws WeixinException indicates getting access token failed, or the content is risky.
+ * @see 检查一段文本是否含有违法违规内容
+ */
+ public void msgSecCheck(String content) throws WeixinException {
+ final Map params = new HashMap(1);
+ params.put("content", content);
+ final WxaApiResult r = this.post("wxa_msg_sec_check", params, WxaApiResult.TYPE_REFERENCE);
+ r.checkErrCode();
+ }
+
+}
diff --git a/weixin4j-wxa/src/main/resources/com/foxinmy/weixin4j/wxa/api/weixin.properties b/weixin4j-wxa/src/main/resources/com/foxinmy/weixin4j/wxa/api/weixin.properties
index 8f43d7e8..db3376f5 100644
--- a/weixin4j-wxa/src/main/resources/com/foxinmy/weixin4j/wxa/api/weixin.properties
+++ b/weixin4j-wxa/src/main/resources/com/foxinmy/weixin4j/wxa/api/weixin.properties
@@ -3,6 +3,7 @@
# ----------------------------------------------------------------------------
api_base_url=https://api.weixin.qq.com
+api_wxa_url={api_base_url}/wxa
api_cgi_url={api_base_url}/cgi-bin
# \u767b\u5f55\u51ed\u8bc1\u6821\u9a8c
@@ -10,10 +11,10 @@ sns_jscode2session={api_base_url}/sns/jscode2session?appid=%s&secret=%s&js_code=
# \u83b7\u53d6\u5c0f\u7a0b\u5e8f\u7801 \u63a5\u53e3A: \u9002\u7528\u4e8e\u9700\u8981\u7684\u7801\u6570\u91cf\u8f83\u5c11\u7684\u4e1a\u52a1\u573a\u666f
-wxa_getwxacode={api_base_url}/wxa/getwxacode?access_token=%s
+wxa_getwxacode={api_wxa_url}/getwxacode?access_token=%s
# \u83b7\u53d6\u5c0f\u7a0b\u5e8f\u7801 \u63a5\u53e3B\uff1a\u9002\u7528\u4e8e\u9700\u8981\u7684\u7801\u6570\u91cf\u6781\u591a\u7684\u4e1a\u52a1\u573a\u666f
-wxa_getwxacodeunlimit={api_base_url}/wxa/getwxacodeunlimit?access_token=%s
+wxa_getwxacodeunlimit={api_wxa_url}/getwxacodeunlimit?access_token=%s
# \u83b7\u53d6\u5c0f\u7a0b\u5e8f\u4e8c\u7ef4\u7801 \u63a5\u53e3C\uff1a\u9002\u7528\u4e8e\u9700\u8981\u7684\u7801\u6570\u91cf\u8f83\u5c11\u7684\u4e1a\u52a1\u573a\u666f
wxaapp_createwxaqrcode={api_base_url}/cgi-bin/wxaapp/createwxaqrcode?access_token=%s
@@ -43,3 +44,10 @@ message_custom_send={api_cgi_url}/message/custom/send?access_token=%
# \u5ba2\u670d\u8f93\u5165\u72b6\u6001
message_custom_typing={api_cgi_url}/message/custom/typing?access_token=%
+
+
+# imgSecCheck
+wxa_img_sec_check={api_wxa_url}/img_sec_check?access_token=%s
+
+# msgSecCheck
+wxa_msg_sec_check={api_wxa_url}/msg_sec_check?access_token=%s