The authors mention about source | o | alibaba cloud native public number

preface

Arthas is an open source Java diagnostic tool for Alibaba that provides a real-time view of system health. View function call parameters, return values, and exceptions; Online hot update code; Solve class conflict problem in seconds; Locate the class loading path; Generating hot spots; Online application via web diagnostics. Now it has been widely used in major factories, and many products have been extended.

This section describes how to integrate Arthas into the Spring Boot monitoring platform.

SpringBoot Admin

For convenience, SpringBoot Admin is simply called SBA (version: 1.5.x).

If you want to develop a plug-in for version 1.5 OF SBA, you need to download the source package of SBA and Copy it according to the form of spring-boot-admin-server-uI-hystrix. Since JS uses Angular, I have tried for a long time. Having learned how to develop plug-ins, but not Angular, give up 💀

Version: 2.x 2.x SBA plug-in development, the official website has introduced how to develop, JS using Vue, a lot of convenience, because our project is still using 1.5, so we did not use this version, please try by yourself.

You can’t use SBA plug-ins for integration, so what else can you do? 😅

SBA integration

My solution is to Copy Arthas files directly into the Admin service from the Arthas-all project tunnel-server.

Admin Directory structure

1. Arthas directory

This package houses all Arthas Java files.

  • All files in the Endpoint package can be commented out, not much use.
  • ArthasController is a file I created myself to get all the clients registered with Arthas, which will be useful later.
  • Just Copy the rest of the files.
@RequestMapping("/api/arthas") @RestController public class ArthasController { @Autowired private TunnelServer tunnelServer; @RequestMapping(value = "/clients", method = RequestMethod.GET) public Set<String> getClients() { Map<String, AgentInfo> agentInfoMap = tunnelServer.getAgentInfoMap(); return agentInfoMap.keySet(); }}Copy the code

spring-boot-admin-server-ui

This file is created in Resources. Meta-inf. Admin will load the files in this directory on startup.

2. Resources directory

  • Index.html overrides the original SBA home page and adds an Arthas navigation to it

<! DOCTYPE html> <html class="no-js"> <head> <meta charset="utf-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <title>Spring Boot Admin</title> <meta name="description" content=""> <meta name="viewport" content="width=device-width"> <link rel="shortcut icon" type="image/x-icon" href="img/favicon.png"/> <link rel="stylesheet" type="text/css" href="core.css"/> <link rel="stylesheet" type="text/css" href="all-modules.css"/> </head> <body> <header class="navbar header--navbar desktop-only"> <div class="navbar-inner"> <div class="container-fluid"> <div class="spring-logo--container"> <a class="spring-logo" href="#"><span></span></a> </div> <div class="spring-logo--container"> <a class="spring-boot-logo" href="#"><span></span></a> </div> <ul class="nav pull-right"> <! Arthas navigation --> <li class="navbar-link ng-scope"> <a class="ng-binding" href=" Arthas /arthas.html">Arthas</a> </li> <li ng-repeat="view in mainViews" class="navbar-link" ng-class="{active: $state.includes(view.state)}"> <a ui-sref="{{view.state}}" ng-bind-html="view.title"></a> </li> </ul> </div> </div> </header> <div ui-view></div> <footer class="footer"> <ul class="inline"> <li><a href="https://codecentric.github.io/spring-boot-admin/@project.version@" target="_blank">Reference Guide</a></li> <li>-</li> <li><a href="https://github.com/codecentric/spring-boot-admin" target="_blank">Sources</a></li> <li>-</li> < li > Code licensed under the < a href = "http://www.apache.org/licenses/LICENSE-2.0" target = "_blank" > the Apache License 2.0 < / a > < / li >  </ul> </footer> <script src="dependencies.js" type="text/javascript"></script> <script type="text/javascript"> sbaModules = []; </script> <script src="core.js" type="text/javascript"></script> <script src="all-modules.js" type="text/javascript"></script> <script type="text/javascript"> angular.element(document).ready(function () { angular.bootstrap(document, sbaModules.slice(0), { strictDi: true }); }); </script> </body> </html>Copy the code
  • Arthas.html

Create a new page to display the Arthas console page.

There are two hidden text fields in this file that connect to the Arthas server and automatically assign Admin’s Url to Ip when the page loads.

<input type="hidden" ID =" IP "name=" IP" value="127.0.0.1"> <input type="hidden" ID ="port" name="port" value="19898">Copy the code
<! DOCTYPE html> <html class="no-js"> <head> <meta charset="utf-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <title>Spring Boot Admin</title> <meta name="description" content=""> <meta name="viewport" content="width=device-width"> <link rel="shortcut icon" type="image/x-icon" href=".. /img/favicon.png"/> <link rel="stylesheet" type="text/css" href=".. /core.css"/> <link rel="stylesheet" type="text/css" href=".. /all-modules.css"/> <script SRC ="js/jquery-3.3.1.min.js"></script> <script SRC ="js/popper-1.14.6.min.js"></script> <script src="js/xterm.js"></script> <script src="js/web-console.js"></script> <script src="js/arthas.js"></script> <link  href="js/xterm.css" rel="stylesheet" /> <script type="text/javascript"> window.addEventListener('resize', function () { var terminalSize = getTerminalSize(); ws.send(JSON.stringify({ action: 'resize', cols: terminalSize.cols, rows: terminalSize.rows })); xterm.resize(terminalSize.cols, terminalSize.rows); }); </script> </head> <body> <header class="navbar header--navbar desktop-only"> <div class="navbar-inner"> <div class="container-fluid"> <div class="spring-logo--container"> <a class="spring-logo" href="#"><span></span></a> </div> <div class="spring-logo--container"> <a class="spring-boot-logo" href="#"><span></span></a> </div> <ul class="nav pull-right"> <li class="navbar-link ng-scope"> <a class="ng-binding" href="arthas.html">Arthas</a> </li> <li class="navbar-link ng-scope"> <a class="ng-binding" href=".. /">Applications</a> </li> <li class="navbar-link ng-scope"> <a class="ng-binding" href=".. /#/turbine">Turbine</a> </li> <li class="navbar-link ng-scope"> <a class="ng-binding" href=".. /#/events">Journal</a> </li> <li class="navbar-link ng-scope"> <a class="ng-binding" href=".. /#/about">About</a> </li> <li class="navbar-link ng-scope"> <a class="ng-binding" href=".. /#/logout"><i class="fa fa-2x fa-sign-out" aria-hidden="true"></i></a> </li> </ul> </div> </div> </header> <div ui-view> <div class="container-fluid"> <form class="form-inline"> <input type="hidden" id=" IP "name=" IP" value="127.0.0.1"> <input type="hidden" id="port" name="port" value="19898"> Select Application:  <select id="selectServer"></select> <button class="btn" onclick="startConnect()" type="button"><i class="fa fa-connectdevelop"></i> Connect</button> <button class="btn" onclick="disconnect()" type="button"><i class="fa fa-search-minus"></i> Disconnect</button> <button class="btn" onclick="release()" type="button"><i class="fa fa-search-minus"></i> Release</button> </form> <div id="terminal-card"> <div id="terminal"></div> </div> </div> </div> </body> </html>Copy the code
  • Arthas.js Stores the js for page control
var registerApplications = null; var applications = null; $(document).ready(function () { reloadRegisterApplications(); reloadApplications(); }); / * * * to get registered arthas client * / function reloadRegisterApplications () {var result = reqSync ("/API/arthas/clients ", "get"); registerApplications = result; initSelect("#selectServer", registerApplications, ""); */ function reloadApplications() {applications = reqSync("/ API /applications", "get"); / / function initSelect(uiSelect, list, key) {$(uiSelect).html(" "); var server; for (var i = 0; i < list.length; i++) { server = list[i].toLowerCase().split("@"); if ("phantom-admin" === server[0]) continue; $(uiSelect).append("<option value=" + list[i].toLowerCase() + ">" + server[0] + "</option>"); Function release() {var currentServer = $("#selectServer").text(); for (var i = 0; i < applications.length; i++) { serverId = applications[i].id; serverName = applications[i].name.toLowerCase(); console.log(serverId + "/" + serverName); if (currentServer === serverName) { var result = reqSync("/api/applications/" +serverId+ "/env/reset", "post"); alert("env reset success"); } } } function reqSync(url, method) { var result = null; $. Ajax ({url: url, type: method, async: false, headers: {' content-type ': 'application/json; charset=utf8; ', }, success: function (data) { // console.log(data); result = data; }, error: function (data) { console.log("error"); }}); return result; }Copy the code
  • Web-console.js

Modified the connection part of the code, for reference.

var ws; var xterm; $(function () {var url = window.location.href; var ip = getUrlParam('ip'); var port = getUrlParam('port'); var agentId = getUrlParam('agentId'); if (ip ! = '' && ip ! = null) { $('#ip').val(ip); } else { $('#ip').val(window.location.hostname); } if (port ! = '' && port ! = null) { $('#port').val(port); } if (agentId ! = '' && agentId ! = null) { $('#selectServer').val(agentId); } // startConnect(true); }); /** get params in url **/ function getUrlParam (name, url) { if (! url) url = window.location.href; name = name.replace(/[\[\]]/g, '\\$&'); var regex = new RegExp('[?&]' + name + '(=([^&#]*)|&|#|$)'), results = regex.exec(url); if (! results) return null; if (! results[2]) return ''; return decodeURIComponent(results[2].replace(/\+/g, ' ')); } function getCharSize () { var tempDiv = $('<div />').attr({'role': 'listitem'}); var tempSpan = $('<div />').html('qwertyuiopasdfghjklzxcvbnm'); tempDiv.append(tempSpan); $("html body").append(tempDiv); var size = { width: tempSpan.outerWidth() / 26, height: tempSpan.outerHeight(), left: tempDiv.outerWidth() - tempSpan.outerWidth(), top: tempDiv.outerHeight() - tempSpan.outerHeight(), }; tempDiv.remove(); return size; } function getWindowSize () { var e = window; var a = 'inner'; if (! ('innerWidth' in window )) { a = 'client'; e = document.documentElement || document.body; } var terminalDiv = document.getElementById("terminal-card"); var terminalDivRect = terminalDiv.getBoundingClientRect(); return { width: terminalDivRect.width, height: e[a + 'Height'] - terminalDivRect.top }; } function getTerminalSize () { var charSize = getCharSize(); var windowSize = getWindowSize(); console.log('charsize'); console.log(charSize); console.log('windowSize'); console.log(windowSize); return { cols: Math.floor((windowSize.width - charSize.left) / 10), rows: Math.floor((windowSize.height - charSize.top) / 17) }; } /** init websocket **/ function initWs (ip, port, agentId) { var protocol= location.protocol === 'https:' ? 'wss://' : 'ws://'; var path = protocol + ip + ':' + port + '/ws? method=connectArthas&id=' + agentId; ws = new WebSocket(path); } /** init xterm **/ function initXterm (cols, rows) { xterm = new Terminal({ cols: cols, rows: rows, screenReaderMode: true, rendererType: 'canvas', convertEol: true }); Begin connect **/ function startConnect (silent) {var IP = $('# IP ').val(); var port = $('#port').val(); var agentId = $('#selectServer').val(); if (ip == '' || port == '') { alert('Ip or port can not be empty'); return; } if (agentId == '') { if (silent) { return; } alert('AgentId can not be empty'); return; } if (ws ! = null) { alert('Already connected'); return; } // init webSocket initWs(ip, port, agentId); ws.onerror = function () { ws.close(); ws = null; ! silent && alert('Connect error'); }; ws.onclose = function (message) { if (message.code === 2000) { alert(message.reason); }}; ws.onopen = function () { console.log('open'); $('#fullSc').show(); var terminalSize = getTerminalSize() console.log('terminalSize') console.log(terminalSize) // init xterm initXterm(terminalSize.cols, terminalSize.rows) ws.onmessage = function (event) { if (event.type === 'message') { var data = event.data; xterm.write(data); }}; xterm.open(document.getElementById('terminal')); xterm.on('data', function (data) { ws.send(JSON.stringify({action: 'read', data: data})) }); ws.send(JSON.stringify({action: 'resize', cols: terminalSize.cols, rows: terminalSize.rows})); window.setInterval(function () { if (ws ! = null && ws.readyState === 1) { ws.send(JSON.stringify({action: 'read', data: ""})); }}, 30000); } } function disconnect () { try { ws.close(); ws.onmessage = null; ws.onclose = null; ws = null; xterm.destroy(); $('#fullSc').hide(); alert('Connection was closed successfully! '); } catch (e) { alert('No connection, please start connect first.'); } } /** full screen show **/ function xtermFullScreen () { var ele = document.getElementById('terminal-card'); requestFullScreen(ele); } function requestFullScreen (element) { var requestMethod = element.requestFullScreen || element.webkitRequestFullScreen || element.mozRequestFullScreen || element.msRequestFullScreen; if (requestMethod) { requestMethod.call(element); } else if (typeof window.ActiveXObject ! == "undefined") { var wscript = new ActiveXObject("WScript.Shell"); if (wscript ! == null) { wscript.SendKeys("{F11}"); }}}Copy the code
  • Other documents

    • Jquery – 3.3.1. Min. Js new added js
    • Copy from js
    • Popper – 1.14.6. Min. Js
    • web-console.js
    • xterm.css
    • xterm.js
  • bootstrap.yml

Arthas: Server: port: 9898Copy the code

In this case, the admin side configuration is complete.

Client Configuration

  • Add configuration in the configuration center
# arthas.tunnel-server = ws://admin /ws # arthas.tunnel-server = ws://admin /ws Arthas.agent-id = ${spring.application.name}@${random. Value} #arthas switch. Turn off spring.arthas.enabled = false when not neededCopy the code
  • Arthas-spring-boot-starter is introduced in applications that require automatic Attach. The starter needs to be partially modified to remove the part of the registered Arthas. The following is the modified file.

This is to repackage the modified files into jars and upload them to the private server, but some applications will not be able to load ArthasConfigMap, you can put the two files separately into the public package of your project.

@EnableConfigurationProperties({ ArthasProperties.class }) public class ArthasConfiguration { private static final Logger logger = LoggerFactory.getLogger(ArthasConfiguration.class); @ConfigurationProperties(prefix = "arthas") @ConditionalOnMissingBean @Bean public HashMap<String, String> arthasConfigMap() { return new HashMap<String, String>(); }}Copy the code
@ConfigurationProperties(prefix = "arthas") public class ArthasProperties { private String ip; private int telnetPort; private int httpPort; private String tunnelServer; private String agentId; /** * report executed command */ private String statUrl; /** * session timeout seconds */ private long sessionTimeout; private String home; /** * when arthas agent init error will throw exception by default. */ private boolean slientInit = false; public String getHome() { return home; } public void setHome(String home) { this.home = home; } public boolean isSlientInit() { return slientInit; } public void setSlientInit(boolean slientInit) { this.slientInit = slientInit; } public String getIp() { return ip; } public void setIp(String ip) { this.ip = ip; } public int getTelnetPort() { return telnetPort; } public void setTelnetPort(int telnetPort) { this.telnetPort = telnetPort; } public int getHttpPort() { return httpPort; } public void setHttpPort(int httpPort) { this.httpPort = httpPort; } public String getTunnelServer() { return tunnelServer; } public void setTunnelServer(String tunnelServer) { this.tunnelServer = tunnelServer; } public String getAgentId() { return agentId; } public void setAgentId(String agentId) { this.agentId = agentId; } public String getStatUrl() { return statUrl; } public void setStatUrl(String statUrl) { this.statUrl = statUrl; } public long getSessionTimeout() { return sessionTimeout; } public void setSessionTimeout(long sessionTimeout) { this.sessionTimeout = sessionTimeout; }}Copy the code
  • Switch effect

To implement the switch effect, you also need a file that listens for configuration file changes.

What I’m using here is to change the environment variable in SBA, and the service is listening for the change of the variable. When listening for Spring.arthas. Enabled to be true, register arthas, and here’s the code.

@Component public class EnvironmentChangeListener implements ApplicationListener<EnvironmentChangeEvent> { @Autowired private Environment env; @Autowired private Map<String, String> arthasConfigMap; @Autowired private ArthasProperties arthasProperties; @Autowired private ApplicationContext applicationContext; @Override public void onApplicationEvent(EnvironmentChangeEvent event) { Set<String> keys = event.getKeys(); for (String key : keys) { if ("spring.arthas.enabled".equals(key)) { if ("true".equals(env.getProperty(key))) { registerArthas(); } } } } private void registerArthas() { DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) applicationContext.getAutowireCapableBeanFactory(); String bean = "arthasAgent"; if (defaultListableBeanFactory.containsBean(bean)) { ((ArthasAgent)defaultListableBeanFactory.getBean(bean)).init(); return; } defaultListableBeanFactory.registerSingleton(bean, arthasAgentInit()); } private ArthasAgent arthasAgentInit() { arthasConfigMap = StringUtils.removeDashKey(arthasConfigMap); Map<String, String> mapWithPrefix = new HashMap<String, String>(arthasConfigmap.size ()); for (Map.Entry<String, String> entry : arthasConfigMap.entrySet()) { mapWithPrefix.put("arthas." + entry.getKey(), entry.getValue()); } final ArthasAgent arthasAgent = new ArthasAgent(mapWithPrefix, arthasProperties.getHome(), arthasProperties.isSlientInit(), null); arthasAgent.init(); return arthasAgent; }}Copy the code

The end of the

Now that you’re happy to use the SBA modality application, take a look at the final page.

  • Mode process

The process is as follows:

  1. Open Arthas
  2. Select the Application in Select Application
  3. Connect Application
  4. DisConnect DisConnect application
  5. Release Releases the configuration file

Some drawbacks:

  • Introducing an application using a JAR package is invasive, and if Arthas doesn’t start, the application won’t start either.
  • If Docker is used, the JVM memory needs to be adjusted to prevent Arthas from exploding during debugging.
  • Integration without using the SBA plug-in The preceding integration is for reference only.

Arthas prize entries are in progress!

In an effort to get more developers to start using Arthas as a Java diagnostic tool, the Arthas community has teamed up with JetBrains to launch an Arthas essay campaign: ** Talk about your relationship with Arthas over the years. ** activity is still in progress, click to participate, welcome everyone to contribute, participation is likely to win!