JAVA implémente une alarme de journal anormale basée sur Slack

1. Présentation de la fonction

Dans notre développement quotidien, s'il y a une anomalie dans l'environnement en ligne du système, le développeur ne peut pas savoir à temps pour le réparer, ce qui peut entraîner de lourdes pertes. Par conséquent, il est très nécessaire d'ajouter la fonction d'alarme anormale à l'arrière. -fin de service, et pour développer une fonction Un service complet d'alarme anormale peut prendre beaucoup de temps.Aujourd'hui, je vais vous apporter une méthode pour réaliser une alarme de journal anormale basée sur Slack.

Logique de mise en œuvre : dans des circonstances normales, le code traitera l'endroit anormal. La chose la plus élémentaire est d'imprimer le journal. Cet article se rendra compte que lorsque le journal est imprimé, les informations anormales seront envoyées au canal Slack en même temps. Personnel de développement ou d'exploitation et de maintenance Créez un compte Slack, rejoignez un canal et recevez des alertes en temps réel en cas d'informations anormales.

2. Introduction à Slack

Slack It est un outil de communication en temps réel basé sur le Web disponible sous la forme d'une application unique pour les ordinateurs de bureau/portables, les appareils mobiles, ainsi qu'une application Web. Fondamentalement, c'est votre chat privé et votre salle de collaboration. Pour de nombreuses entreprises, il a remplacé les e-mails/forums privés/salles de discussion comme principal canal de communication interne basé sur du texte. On peut comprendre qu'il s'agit d'un groupe de discussion + intégration d'outils à grande échelle + intégration de fichiers + recherche unifiée . Fin 2014, Slack a intégré 65 outils et services tels que la messagerie électronique, les SMS, Google Drives, Twitter, Trello, Asana, GitHub, etc., qui peuvent rassembler diverses communications et collaborations d'entreprise fragmentées. Plusieurs notions importantes :

Espaces de travail : Allez plutôt dans les espaces de travail, les utilisateurs peuvent rejoindre ou créer différents espaces de travail, souvent le nom et l'URL de l'espace de travail seront le nom de l'entreprise.

Canaux : Les canaux peuvent être divisés en différentes équipes ou sujets, et peuvent également être compris comme l'équivalent de WeChat, où les membres d'un canal partagent des informations dans le canal.

3. Préparation préliminaire

configuration lâche

  1. Créez un compte, connectez-vous, vous pouvez utiliser l'application ou vous connecter à la version Web avec un navigateur
  2. Créez votre propre espace de travail et invitez d'autres personnes dans votre espace de travail.
  3. Créez un canal, invitez des collègues à le rejoindre, vous pouvez envoyer des informations au canal à ce moment-là, et tous ceux qui rejoignent le canal peuvent voir les informations
  4. Ajoutez l'application Incoming WebHook à l'espace de travail , sélectionnez le canal, enregistrez l'URL du Webhook, puis envoyez des messages au canal via le programme de mise en œuvre du Webhook.image.png

image.png

image.png

image.png

pom.xml

<dependencies>
    <dependency>
        <groupId>org.apache.httpcomponents</groupId>
        <artifactId>httpclient</artifactId>
        <version>4.5.2</version>
    </dependency>
    <dependency>
        <groupId>com.alibaba</groupId>
        <artifactId>fastjson</artifactId>
        <version>1.2.83</version>
    </dependency>
    <dependency>
        <groupId>commons-configuration</groupId>
        <artifactId>commons-configuration</artifactId>
        <version>1.10</version>
    </dependency>
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.10</version>
        <scope>test</scope>
    </dependency>
</dependencies>
复制代码

Quatrièmement, la mise en œuvre spécifique

1. Implémenter Slack pour envoyer des messages

Classe d'outil de message SlackUtil pour Slack

package com.yy.operation;



import com.yy.common.CommonThreadFactory;
import com.yy.common.ConnUtil;
import org.apache.commons.lang.StringUtils;

import java.text.MessageFormat;
import java.text.SimpleDateFormat;
import java.util.concurrent.*;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
 * @author :Max
 * @date :Created in 2022/8/26 下午12:54
 * @description:
 */

public class SlackUtil {

    private static final Logger logger = Logger.getLogger(SlackUtil.class.getCanonicalName());

    private static SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    private static final String SEND_USER_NAME ="运维机器人";

    private static int MAX_RETRY =3;


    /**
     * 线程池 抛弃策略DiscardPolicy:这种策略,会默默的把新来的这个任务给丢弃;不会得到通知
      */
    private static ExecutorService executor = new ThreadPoolExecutor(10,30,60,TimeUnit.MILLISECONDS,new ArrayBlockingQueue<Runnable>(200),new CommonThreadFactory("Slack"), new ThreadPoolExecutor.DiscardPolicy());

    private static String MSG_FORMAT ="payload='{'"channel": "{0}", "username": "{1}", "text": "{2}", "icon_emoji": ":ghost:"'}'" ;

    /**
     * 保存的Webhook URL ,需要初始化
     */
    private static String WEBHOOK_URL ;
    private static boolean SLACK_ABLE;

    public static void setSlackConfig(String webhookUrl){
        WEBHOOK_URL = webhookUrl;
        SLACK_ABLE = true;
    }

    /**
     * slack异步发消息,保证不能影响到主功能
      * @param channel
     * @param msg
     */
    public static void send(final String channel, final String msg){
        if(!SLACK_ABLE){
            return;
        }
        if(StringUtils.isBlank(msg)){
            return;
        }
        executor.submit(new Runnable() {
            @Override
            public void run() {
                try {
                    SlackUtil.send(channel,sdf.format(System.currentTimeMillis())+"   "+msg,MAX_RETRY);
                } catch (Exception e) {
                    logger.log(Level.SEVERE, e.getMessage(), e);
                }
            }
        });
    }


    /**
     * 如果slask发消息失败,会最多尝试发三次,三次都失败,会打印异常信息
      * @param channel
     * @param msg
     * @param retry
     * @throws Exception
     */
    public static void send(String channel, String msg, int retry) throws Exception {
        if(msg.indexOf(""")>=0 ||msg.indexOf("{")>=0 ||msg.indexOf("}")>=0){
            msg =msg.replace(""","'").replace("{","[").replace("}","]");
        }
        String payload = MessageFormat.format(MSG_FORMAT, channel,SEND_USER_NAME,msg);
        String result = ConnUtil.getContentByPostWithUrlencode(WEBHOOK_URL,payload);
        logger.info("result:"+result);
        if(StringUtils.isEmpty(result) ||!result.startsWith("ok")){
            --retry;
            if(retry>0){
                try {
                    TimeUnit.SECONDS.sleep(retry*5);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                send(channel,msg,retry);
            }else{
                throw new Exception("Fail to send slack:"+result+"\nmsg:"+msg);
            }
        }
    }


}
复制代码

Faire une demande au webhook via Urlencode

package com.yy.common;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.HttpClientBuilder;

import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
 * @author :Max
 * @date :Created in 2022/8/26 下午1:44
 * @description:
 */

public class ConnUtil {

    private static final Logger logger = Logger.getLogger(ConnUtil.class.getCanonicalName());

    public static String getContentByPostWithUrlencode(String url,String msg){

        StringEntity entity = new StringEntity(msg, "UTF-8");
        entity.setContentEncoding("UTF-8");
        entity.setContentType(" application/x-www-form-urlencoded");

        HttpClient httpClient = HttpClientBuilder.create().build();
        HttpPost request = new HttpPost(url);
        request.setEntity(entity);

        HttpResponse response = null;
        try {
            response = httpClient.execute(request);
            HttpEntity responseEntity = response.getEntity();
            if (responseEntity != null) {
                InputStream instream = responseEntity.getContent();
                BufferedReader reader = new BufferedReader(new InputStreamReader(instream));
                StringBuffer contents = new StringBuffer();
                String line = null;
                while ((line = reader.readLine()) != null) {
                    contents.append(line);
                    contents.append("\n");
                }
                return contents.toString();
            }
        } catch (Exception ex) {
            logger.log(Level.SEVERE, ex.getMessage(), ex);
        }
        return null;
    }

}
复制代码

Test SlackUtil

package com.yy.test;

import com.yy.common.SlackChannelEnum;
import com.yy.operation.SlackUtil;
import org.junit.Assert;
import org.junit.Test;

import java.util.concurrent.TimeUnit;

/**
 * @author :Max
 * @date :Created in 2022/8/28 下午2:37
 * @description:
 */

public class SlackTest {

    static {
        SlackUtil.setSlackConfig("https://hooks.slack.com/services/*******");
    }

    @Test
    public void test(){
        SlackUtil.send(SlackChannelEnum.EXCEPTION.channel,"test ~");
        try {
            TimeUnit.MINUTES.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        Assert.assertTrue(true);
    }
}
复制代码

Envoyer avec succès, vous pouvez voir les informations dans le canal

image.png

2. Réécrire la classe du journal d'impression

Traitement courant du journal des exceptions

public class LoggerTest {
    
    private static final Logger logger = Logger.getLogger(LoggerTest.class.getCanonicalName());

    @Test
    public void test() {
        try {
            int i = 1 / 0;
        } catch (Exception e) {
            logger.log(Level.SEVERE, e.getMessage(), e);
        }
    }
}
复制代码

Remplacer la méthode d'encapsulation du journal d'impression

package com.yy.operation;

import com.yy.common.SlackChannelEnum;
import org.apache.commons.lang.StringUtils;

import java.io.PrintWriter;
import java.io.StringWriter;
import java.net.Inet4Address;
import java.net.InetAddress;
import java.text.MessageFormat;
import java.util.logging.Level;
import java.util.logging.LogRecord;
import java.util.logging.Logger;

/**
 * @author  Max
 * @date :Created in 2022/8/4 下午5:14
 * @description:
 */

public class CommonLogger {

    private Logger logger;


    private CommonLogger(String className) {
        logger = Logger.getLogger(className);
    }

    private static String SERVER;

    private static String EXCEPTION_ALARM_FORMAT = "EXCEPTION 发生异常!\n环境 :{0}\n信息 :{1}\n详情 :{2}";

    private static String WARNING_ALARM_FORMAT = "WARNING 发生告警!\n环境 :{0}\n信息 :{1}";

    private static String SEVERE_ALARM_FORMAT = "SEVERE 发生告警!\n环境 :{0}\n信息 :{1}";

    private static String LOG_ALARM_FORMAT = "LOG 发生告警!\n环境 :{0}\n信息 :{1}";

    private static String USER_BEHAVIOR_FORMAT = "CUSTOMER \n环境 :{0}\n信息 :{1}";

    static {
        try{
            InetAddress ip4 = Inet4Address.getLocalHost();
            SERVER = ip4.getHostAddress();

        }catch (Exception e){
            SERVER ="undefined server";
        }
    }

    public static CommonLogger getLogger(String name) {
        return new CommonLogger(name);
    }

    /**
     * Print exception information, send slack
     *
     * @param level
     * @param msg
     * @param e
     */
    public void log(Level level, String msg, Throwable e) {
        if(StringUtils.isBlank(msg)){
            return;
        }
        msg =dolog(level,msg, e);
        msg = MessageFormat.format(EXCEPTION_ALARM_FORMAT, SERVER, formatMsg(msg), getErrmessage(e));
        SlackUtil.send(SlackChannelEnum.EXCEPTION.channel, msg);
    }

    /**
     * Print user behavior information, send slack
     *
     * @param msg
     */
    public void userBehaviorInfo(String msg) {
        if(StringUtils.isBlank(msg)){
            return;
        }
        msg =dolog(Level.INFO,msg);
        msg = MessageFormat.format(USER_BEHAVIOR_FORMAT, SERVER, formatMsg(msg));
        SlackUtil.send(SlackChannelEnum.EXCEPTION.channel, msg);
    }

    public String formatMsg(String msg){
        StringBuilder source =new StringBuilder(logger.getName());
        msg=transferMsgSource(source,msg);
        return source.toString()+" "+msg;
    }

    /**
     * Print warning severe information, send slack
     *
     * @param msg
     */
    public void severe(String msg) {
        if(StringUtils.isBlank(msg)){
            return;
        }
        msg = dolog(Level.SEVERE,msg);
        msg = MessageFormat.format(SEVERE_ALARM_FORMAT, SERVER, formatMsg(msg));
        SlackUtil.send(SlackChannelEnum.EXCEPTION.channel, msg);
    }

    /**
     * Print warning severe information, send slack
     *
     * @param msg
     */
    public void warning(String msg) { 
        if(StringUtils.isBlank(msg)){
            return;
         }
        msg = dolog(Level.WARNING,msg);
        msg = MessageFormat.format(WARNING_ALARM_FORMAT, SERVER, formatMsg(msg));
        SlackUtil.send(SlackChannelEnum.EXCEPTION.channel, msg);
    }

    /**
     * Print warning log information, send slack
     *
     * @param msg
     */
    public void log(Level severe, String msg) {
        if(StringUtils.isBlank(msg)){
            return;
        }
        msg =dolog(severe,msg);
        msg = MessageFormat.format(LOG_ALARM_FORMAT, SERVER, formatMsg(msg));
        SlackUtil.send(SlackChannelEnum.EXCEPTION.channel, msg);
    }

    public static String getErrmessage(Throwable t) {
        return getThrowable(t);
    }

    public void info(String msg) {
        dolog(Level.INFO,msg);
    }

    public void fine(String msg) {
        logger.fine(msg);
    }

    public void setLevel(Level level) {
        logger.setLevel(level);
    }

    public String dolog(Level level, String msg) {
        return dolog(level,msg,null);
    }

    /**
     *
      * @param level
     * @param msg
     * @param thrown
     * @return msg="["+currentThread.getName()+"] "+a.getMethodName()+" "+msg;
     */
    public String dolog(Level level, String msg, Throwable thrown) {

        LogRecord lr = new LogRecord(level, msg);
        lr.setLevel(level);
        if(thrown!=null){
            lr.setThrown(thrown);
        }
        Thread currentThread = Thread.currentThread();
        StackTraceElement[] temp=currentThread.getStackTrace();
        StackTraceElement a=(StackTraceElement)temp[3];
        lr.setThreadID((int) currentThread.getId());
        lr.setSourceClassName(logger.getName());
        lr.setSourceMethodName(a.getMethodName());
        lr.setLoggerName(logger.getName());
        logger.log(lr);
        return "["+currentThread.getName()+"] "+a.getMethodName()+" "+msg;
    }

    public static String getThrowable(Throwable e) {
        String throwable = "";
        if (e != null) {
            StringWriter sw = new StringWriter();
            PrintWriter pw = new PrintWriter(sw);
            pw.println();
            e.printStackTrace(pw);
            pw.close();
            throwable = sw.toString();
        }
        return throwable;
    }

    public static String transferMsgSource(StringBuilder source,String msg){
        if(msg.indexOf(" ")>0){
            String threadName = msg.substring(0,msg.indexOf(" "))+ " ";
            msg=msg.substring(threadName.length());
            source.insert(0,threadName);
            if(msg.indexOf(" ")>0) {
                String method = msg.substring(0, msg.indexOf(" "));
                source.append( "." + method);
                msg = msg.substring(method.length()+1);
            }
        }
        return msg;
    }

}
复制代码
package com.yy.operation;

import java.text.MessageFormat;
import java.util.concurrent.ConcurrentHashMap;
import java.util.logging.Level;
import java.util.logging.Logger;

public class LoggerUtil {
   
   private static Logger curLogger = Logger.getLogger(LoggerUtil.class.getCanonicalName());
   
   private static ConcurrentHashMap<String, CommonLogger> loggers = new ConcurrentHashMap<String, CommonLogger>();
   
   public static CommonLogger getLogger(Class<?> clazz) {
      String className = clazz.getCanonicalName();
      CommonLogger logger = loggers.get(className);
      if (logger == null) {
         logger = CommonLogger.getLogger(className);
         curLogger.fine(MessageFormat.format("Register logger for {0}", className));
         loggers.put(className, logger);
      }
      return logger;
   }
}
复制代码

classe de journal de test

Lorsque la classe de journal est modifiée, le code appelé n'a pas besoin d'être modifié et la fonction d'alarme anormale est intégrée à un faible coût.

public class LoggerTest {

    private static final Logger logger = Logger.getLogger(LoggerTest.class.getCanonicalName());

    @Test
    public void test() {
        try {
            int i = 1 / 0;
        } catch (Exception e) {
            logger.log(Level.SEVERE, e.getMessage(), e);
        }
    }
}
复制代码

Résultats des tests, informations anormales imprimées dans le canal, ce qui est pratique pour le personnel de développement, d'exploitation et de maintenance à localiser

image.png

Cinquièmement, optimisez l'idée d'expansion

  1. Il peut non seulement imprimer des journaux anormaux, mais également imprimer certains comportements clés des utilisateurs, tels que la recharge, etc. Plusieurs canaux peuvent être configurés pour envoyer des messages avec différents sujets.
  2. Le pool de threads peut être optimisé
  3. Si les développeurs ne peuvent pas voir Slack à temps, ils peuvent également intégrer l'e-mail. L'application Mailclark (facturée séparément) peut être ajoutée à Slack. Après la configuration, les informations du canal d'activation peuvent être automatiquement envoyées à n'importe quelle boîte aux lettres, et le destinataire ne le fait pas. besoin de créer un compte slack. Pour une configuration spécifique, veuillez vous référer au lien .

Tout le monde est le bienvenu pour donner des suggestions d'optimisation, et échanger beaucoup !

autre code

package com.yy.common;

import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * @author :Max
 * @date :Created in 2022/8/26 下午1:51
 * @description:
 */

public class CommonThreadFactory implements ThreadFactory {
    private static final AtomicInteger poolNumber = new AtomicInteger(1);
    private final ThreadGroup group;
    private final AtomicInteger threadNumber = new AtomicInteger(1);
    private final String threadNamePrefix;
    private final String nameSpecific;
    private final boolean isDaemon;

    public CommonThreadFactory(String nameSpecific) {
        this(nameSpecific, false);
    }

    public CommonThreadFactory(String nameSpecific, boolean isDaemon) {
        SecurityManager s = System.getSecurityManager();
        this.group = (s != null) ? s.getThreadGroup() :
                Thread.currentThread().getThreadGroup();
        this.threadNamePrefix = "eg-pool-" + poolNumber.getAndIncrement() + "-thread";
        this.nameSpecific = nameSpecific;
        this.isDaemon = isDaemon;
    }

    @Override
    public Thread newThread(Runnable r) {
        Thread t = new Thread(group, r, String.format("%s-%d-%s",
                this.threadNamePrefix, threadNumber.getAndIncrement(), this.nameSpecific), 0);
        t.setDaemon(isDaemon);
        t.setPriority(Thread.NORM_PRIORITY);
        return t;
    }
}
复制代码
public enum SlackChannelEnum {
    EXCEPTION("#test-example");
    public String channel;

    SlackChannelEnum(String channel) {
        this.channel = channel;
    }
}
复制代码

Je suppose que tu aimes

Origine juejin.im/post/7136858841756467230
conseillé
Classement