[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传![(img-kMcD3TbA-1670856713399)(C:\Users\maido\AppData\Roaming\Typora\typora-user-images\image-20221212223824730.png)]](/uploadfile/202404/7bbbf0394357864.png)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-R8PLHOhD-1670856713401)(C:\Users\maido\AppData\Roaming\Typora\typora-user-images\image-20221212224028100.png)]](/uploadfile/202404/dc40580e5edef26.png)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-My0CyJOP-1670856713404)(C:\Users\maido\AppData\Roaming\Typora\typora-user-images\image-20221212224147798.png)]](/uploadfile/202404/1eb4fa316ed711b.png)
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传![(img-WxgwCe3k-1670856713405)(C:\Users\maido\AppData\Roaming\Typora\typora-user-images\image-20221212224209899.png)]](/uploadfile/202404/9c48d6ef5ff0.png)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6NWyeS8E-1670856713407)(C:\Users\maido\AppData\Roaming\Typora\typora-user-images\image-20221212224250346.png)]](/uploadfile/202404/d9273315df9a77d.png)
参考文档
package com.cjpension.ppmshttpbaffle.aspect;import cn.hutool.core.date.DateUtil;
import cn.hutool.core.net.NetUtil;
import cn.hutool.http.HttpUtil;
import com.alibaba.fastjson.JSONObject;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Profile;import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;/*** @ClassName ExceptionLogAspect* @Description TODO* @Author maido* @Date 21:11 2022/12/12* @Version 1.0**/
@Slf4j
@Aspect
@Configuration
@Profile(value = {"dev", "test"})
public class ExceptionLogAspect {@Value("${spring.application.name}")private String appName;@Value("${spring.profiles.active}")private String profile;private static AtomicInteger indexAtomic = new AtomicInteger(0);public static List devList = new ArrayList<>();static {devList.add("2d5a9cdcf81fcee5217fc1f2e54fa195e49be806890dccff11c2adf80be624a1");devList.add("f8350591d9375c7d59327c9847d2dbd6f57b49f56ce29b5ba55fff9aa399cb7a");devList.add("c7ba03a7c91e28b6939b85c1d7a1df5fa710fd2050e0cdd9a59fa5553598e124");devList.add("12b3e0cde929b61a18dba4cba42a3a7faaeaf5100d3d12990f767e8a1912487e");devList.add("fb78ad45cc0eccdab32bfce839abac2020fd61e6837f28690ab6ca5494db1a5a");devList.add("33828b1bd0602eddadcc91c4e01d063777812a410d52c3df6a5a2c07b214299f");devList.add("0b44dc7e46b2c80cc84c5d49be75b9a9c1642bc1b73a9d144ce6ce6c176e2310");devList.add("5785eafe2cfbf3065bc3c0f827becacd0245ed520d978189f56beab9248b62e9");devList.add("0d0d145ccadee55f5ed4ca0b274c0b1e4d7019a195d284f3531f72b31c10f6c3");}/*** 轮训获取token*/public String getToken() {if (indexAtomic.get() >= devList.size()) {indexAtomic.set(0);}return (String) devList.get(indexAtomic.getAndIncrement());}/*** 切入点*/@Pointcut("within(@org.springframework.web.bind.annotation.RestController *) ||" +"within(@org.springframework.stereotype.Controller *)")public void log() {}@AfterThrowing(value = "log()", throwing = "e")public void sendErrorLog(JoinPoint joinPoint, Exception e) {if(isSkip(e)){return;}// 获取推送内容String content = buildMsg(e, joinPoint);// https://oapi.dingtalk.com/robot/send?access_token=d2b5318b162ddda8b366fc9e786090a7b58b30f8f79e72ea6612b7829a930257String result = HttpUtil.post("https://oapi.dingtalk.com/robot/send?access_token=" + getToken(),content);log.info("error log push to dingding result : {}" ,result);}/*** 是否跳过*/private boolean isSkip(Exception e) {List skipList = new ArrayList<>();String exceptionName = e.getClass().getName();if(skipList.contains(exceptionName)){return true;}return false;}private String buildMsg(Exception e,JoinPoint joinPoint) {StringBuilder sb = new StringBuilder("{ \"msgtype\":\"markdown\",\"markdown\":{\"title\":\"错误日志报警\",\"text\":\"");sb.append("错误日志通知").append("\n\n");sb.append("时间: ").append(DateUtil.format(new Date(), "yyyy-MM-dd HH:mm:ss")).append("\n\n");sb.append("项目名称: ").append(appName).append("\n\n");sb.append("环境: ").append(profile).append("\n\n");sb.append("服务IP: ").append(NetUtil.getLocalhostStr()).append("\n\n");sb.append("请求方法名: ").append(joinPoint.getSignature().getDeclaringTypeName()).append(".").append(joinPoint.getSignature().getName()).append("\n\n");sb.append("请求参数: ").append(JSONObject.toJSONString(joinPoint.getArgs()).replace("}","").replace("{","").replace("[","").replace("]","").replace("\"","")).append("\n\n");sb.append("具体信息: ").append(e.getClass().getName()).append(" ");if (e.getStackTrace() != null) {StackTraceElement[] stackTraces = e.getStackTrace();sb.append(e.getMessage()).append("\n");int i = 0;for (StackTraceElement stackTrace : stackTraces) {//只打印00行的堆栈if (i < 20) {sb.append(stackTrace.toString()).append("\n");} else {break;}i++;}} else {sb.append(e.getMessage());}sb.append("\"},\"at\":{\"atMobiles\":[],\"isAtAll\":false}}");return sb.toString();}}
{"at": {"atMobiles":["180xxxxxx"],"atUserIds":["user123"],"isAtAll": false},"text": {"content":"我就是我, @XXX 是不一样的烟火"},"msgtype":"text"
}
| 参数 | 参数类型 | 是否必填 | 说明 |
|---|---|---|---|
| msgtype | String | 是 | 消息类型,此时固定为:text。 |
| content | String | 是 | 消息内容。 |
| atMobiles | Array | 否 | 被@人的手机号。注意 在content里添加@人的手机号,且只有在群内的成员才可被@,非群内成员手机号会被脱敏。 |
| atUserIds | Array | 否 | 被@人的用户userid。注意 在content里添加@人的userid。 |
| isAtAll | Boolean | 否 | 是否@所有人。 |
{"msgtype": "link", "link": {"text": "这个即将发布的新版本,创始人xx称它为红树林。而在此之前,每当面临重大升级,产品经理们都会取一个应景的代号,这一次,为什么是红树林", "title": "时代的火车向前开", "picUrl": "", "messageUrl": "https://www.dingtalk.com/s?__biz=MzA4NjMwMTA2Ng==&mid=2650316842&idx=1&sn=60da3ea2b29f1dcc43a7c8e4a7c97a16&scene=2&srcid=09189AnRJEdIiWVaKltFzNTw&from=timeline&isappinstalled=0&key=&ascene=2&uin=&devicetype=android-23&version=26031933&nettype=WIFI"}
}
| 参数 | 参数类型 | 是否必填 | 说明 |
|---|---|---|---|
| msgtype | String | 是 | 消息类型,此时固定为:link。 |
| title | String | 是 | 消息标题。 |
| text | String | 是 | 消息内容。如果太长只会部分展示。 |
| messageUrl | String | 是 | 点击消息跳转的URL,打开方式如下:移动端,在钉钉客户端内打开PC端默认侧边栏打开希望在外部浏览器打开,请参考消息链接说明 |
| picUrl | String | 否 | 图片URL。 |

{"msgtype": "markdown","markdown": {"title":"杭州天气","text": "#### 杭州天气 @150XXXXXXXX \n > 9度,西北风1级,空气良89,相对温度73%\n > \n > ###### 10点20分发布 [天气](https://www.dingtalk.com) \n"},"at": {"atMobiles": ["150XXXXXXXX"],"atUserIds": ["user123"],"isAtAll": false}}
| 参数 | 类型 | 是否必填 | 说明 |
|---|---|---|---|
| msgtype | String | 是 | 消息类型,此时固定为:markdown。 |
| title | String | 是 | 首屏会话透出的展示内容。 |
| text | String | 是 | markdown格式的消息。 |
| atMobiles | Array | 否 | 被@人的手机号。注意 在text内容里要有@人的手机号,只有在群内的成员才可被@,非群内成员手机号会被脱敏。 |
| atUserIds | Array | 否 | 被@人的用户userid。注意 在content里添加@人的userid。 |
| isAtAll | Boolean | 否 | 是否@所有人。 |
下一篇:2022美亚个人赛复盘