java web源代码

与其他系统进行交互时,大多数Java Web应用程序都使用标准Java接口。 诸如Web页面或REST服务器之类的基于HTTP的服务是使用接口javax.servlet.Servlet来实现的。 使用JDBC接口java.sql.Statementjava.sql.Connection实现数据库交互。 这些标准几乎是通用的,与基础框架(Spring或Java EE)和Servlet容器(Tomcat,Wildfly等)无关。

本文介绍如何实现Java代理,该Java代理使用Bytecode操作来挂接到这些接口,并收集有关HTTP和数据库调用的频率和持续时间的度量。 演示代码可在https://github.com/fstab/promagent上找到 ,该代码是为Prometheus监视系统检测Java Web应用程序的代理。 但是,本文不是Prometheus特有的,它着重于Java代理,字节码操作和类加载器等基础技术。

1. Java代理

Java代理是可以附加到JVM以便操作Java字节码的Java程序。 例如,Java代理可用于修改接口javax.servlet.Servlet所有实现,以获取有关HTTP调用的数量和持续时间的统计信息。

Java代理作为JAR文件提供。 常规Java程序使用main()方法作为应用程序的入口点,而Java代理具有premain()方法,该方法将在应用程序的main()方法之前调用:

Java代理概述

public class MyAgent {public static void premain(String agentArgs, Instrumentation inst) throws Exception {// ...}
}

虽然可执行的JAR文件有一个MANIFEST.MF文件中指定Main-Class ,代理JAR文件有一个MANIFEST.MF文件中指定的Premain-Class 。 可以使用命令行选项-javaagent:在应用程序启动期间附加代理:

Java代理命令行

java -javaagent:myagent.jar -jar myapp.jar

然后, premain()方法可以调用inst.addTransformer()来注册ClassFileTransformer 。 类文件转换器实现了每当加载Java类时都会调用的transform()方法。 它可以检查和修改任何Java类的字节码,以添加其他功能。

2.字节码操作

有几个可用的库可帮助Java开发人员实现字节码操作。 最低级别的是ASM 。 其他库,例如cglib和javassist提供更高级别的API。 最新,最易于使用的库是Byte Buddy 。 它提供了易于阅读的流畅Java API,用于创建ClassFileTransformer并将其注册到Instrumentation

字节伙伴代理示例

package io.promagent;import net.bytebuddy.agent.builder.AgentBuilder;
import net.bytebuddy.agent.builder.AgentBuilder.Transformer;
import net.bytebuddy.matcher.ElementMatchers;import java.lang.instrument.Instrumentation;import static net.bytebuddy.matcher.ElementMatchers.hasSuperType;
import static net.bytebuddy.matcher.ElementMatchers.named;public class MyAgent {public static void premain(String agentArgs, Instrumentation inst) throws Exception {new AgentBuilder.Default().type(hasSuperType(named("javax.servlet.Servlet"))).transform(new Transformer.ForAdvice().include(MyAgent.class.getClassLoader()).advice(ElementMatchers.named("service"), "io.promagent.MyAdvice")).installOn(inst);}

上面的示例显示了检测所有javax.servlet.Servlet实现的service()方法所需的完整代码。 每当Servlet处理Web请求时,都会调用service()方法。 MyAdvice类定义将注入到Servlet的service()方法中的代码。 这段代码使用@Advice.OnMethodEnter@Advice.OnMethodExit

字节好友建议示例

public class MyAdvice {@Advice.OnMethodEnterpublic static void before(ServletRequest request, ServletResponse response) {System.out.println("before serving the request...");}@Advice.OnMethodExitpublic static void after(ServletRequest request, ServletResponse response) {System.out.println("after serving the request...");}
}

Byte Buddy提供了两种检测方法:建议(如上所示)和拦截器。 区别是微妙的:有了建议, @Advice.OnMethodEnter@Advice.OnMethodExit方法的字节码被复制到被拦截方法的开始和最终块中。 效果与将代码复制并粘贴到要拦截的service()实现中相同。 结果,在完成检测之后,不再使用类MyAdvice 。 截获的service()方法不需要访问MyAdvice类,可以在MyAdvice类不可用的类加载器上下文中执行。

另一方面,拦截器是常规方法调用,它们在被拦截方法的开始和最终块中执行。 这意味着被拦截的方法必须在拦截器可用的上下文中执行。

在以下各节中,我们将看到可能会限制应用程序服务器环境中类的可见性,这就是Promagent使用Advices而非Interceptor的原因。

3.添加依赖项

为了使上面的示例有用,我们需要用代码维护指标并将指标提供给监视系统来替换System.out.println()消息。 例如, Promagent使用Prometheus客户端库来维护和公开Prometheus指标。

JVM自动将使用-javaagent:命令行参数指定的JAR文件添加到应用程序的系统类加载器。 因此,从理论上讲,应该可以创建一个包含代理及其所有依赖项的Uber JAR ,并在-javaagent:命令行参数中使用它。

但是,在应用程序服务器环境中,使所有依赖项在系统类加载器上可用都是有问题的,原因有两个:

  • 某些代理的依赖项可能与应用程序服务器内部使用的库或WAR文件中作为已部署应用程序的一部分提供的库发生冲突。
  • 为了防止冲突,应用程序服务器限制了从系统类加载器对类的访问。 例如,除非使用jboss.modules.system.pkgs系统属性显式公开了受影响的程序包,否则Wildfly模块无法从系统类加载器访问类。 跟踪所有依赖关系并相应地配置模块系统并非易事。

更好的方法是只公开几个Java类,而无需外部依赖系统类加载器,并使用自定义类加载器加载实际的度量实现。 这样可以最大程度地减少潜在的冲突以及运行代理所需的配置。

4.从自定义类加载器加载钩子

在Java中实现自定义类加载器很容易,因为我们可以简单地使用java.net.URLClassLoader并使用指向我们的类所在的JAR文件的路径对其进行初始化。 为了使代理易于使用, Promagent随包含其他JAR文件的JAR文件一起提供。 内部JAR文件在启动时被复制到临时目录,并且使用临时路径配置了自定义类加载器。 这样,用户将获得一个单一的代理JAR,而该代理在内部区分系统类加载器上的类(这些类直接包含在代理JAR中)和定制类加载器上的类(这些类是从JAR中加载的)临时目录)。

实际的工具在称为hook的类中实现。 挂钩是从自定义类加载器加载的。 这样,只要自定义类加载器能够提供这些依赖关系,钩子就可以引用其所需的任何依赖关系。 例如, ServletHook如下所示:

自定义钩子类示例

public class ServletHook {public void before(ServletRequest request, ServletResponse response) {// ...}public void after(ServletRequest request, ServletResponse response) {// ...}
}

该钩子看起来与Byte Buddy建议类似。 区别在于Byte Buddy建议仅是几行代码,这些代码具有最小的依赖性,以便从自定义类加载器加载相应的钩子,并通过反射将其委派给钩子的before()after()方法。 字节好友建议对检测库没有任何依赖关系,因为实际的检测库仅在自定义类加载器中可见。

但是,在加载钩子时有一个细微的陷阱:参数ServletRequestServletResponse将从已检测的Servlet传递。 这意味着,挂钩中的ServletRequestServletResponse类必须使用与拦截的Servlet相同的类加载器进行加载,否则我们无法将Servlet的参数传递到挂钩的before()after()方法中。

解决方案是使用Thread.currentThread().getContextClassLoader()作为自定义类加载器的父级。 这样,可以从上下文类加载器加载的所有类都将从上下文类加载器加载。 这包括ServletRequestServletResponse 。 只有当前上下文中不可用的类(例如钩子本身及其依赖项)才会从自定义JAR文件中加载。 这意味着我们每个上下文需要一个自定义类加载器,因为每个自定义类加载器都委托给另一个上下文类加载器作为其父代。

5.实施全球指标注册

使用到目前为止描述的实现,可以检测单个Web应用程序。 但是,如果应用程序服务器上有多个部署,则每个工具将具有自己的类加载器。 从不同的类加载器加载指标库时,部署无法共享在该指标库中定义的全局静态变量。 例如,不可能跨多个部署使用Prometheus客户端库随附的全局度量标准注册表。 缺少全局注册表,每个部署都需要独立维护和公开其指标。

解决此问题的一种方法是扩展自定义类加载器,并使其委托将共享度量库加载到另一个共享的自定义类加载器。 但是,JVM还附带有一个内置的全局注册表,我们可以将其用作VM范围的指标存储:JMX平台MBean服务器。 将度量标准注册为MBean具有以下好处:

  • 全局注册表:JMX平台MBean服务器提供了VM范围的注册表,使我们能够维护一组全局指标,以对应用程序服务器上的所有部署进行检测。
  • 监视系统的单个导出器:易于实现一个小型Web应用程序,该应用程序从MBean服务器读取所有度量并将其提供给监视系统。 例如, Promagent包含用于将指标导出到Prometheus服务器的WAR部署。
  • JMX工具:由于所有指标都可以作为MBean使用,因此可以使用任何JMX客户端来了解指标的状态。

JMX平台MBean服务器是Java SE的一部分,可以通过静态方法ManagementFactory.getPlatformMBeanServer()进行访问。 在MBean服务器上注册的Java对象称为MBean。 MBean必须在一个接口中定义其公共可访问的API,按照惯例,该接口的名称类似于Java类,并附加了后缀MBean 。 例如,要将Counter类注册为MBean,该类必须实现一个名为CounterMBean的接口。 每个MBean均可通过唯一的ObjectName寻址。 可以使用MBeanServer.invoke()来调用MBean接口中定义的方法。

6.总结

本文概述了如何在不修改Java Web应用程序源代码的情况下对其进行检测。 它基于Promagent ,该工具使用Prometheus指标对Java应用程序进行检测。 但是,本文重点介绍了诸如Java代理, 字节好友字节码操作库之类的基础技术,在诸如Wildfly之类的应用服务器环境中的类加载以及JMX平台MBean服务器。

最好从Promagent示例代码中总结出一些松散的结局,例如如何避免HTTP请求经过多个Servlet链时重复计数。 有关更多示例,值得研究相关项目,例如inspectIT或stagemonitor 。

翻译自: https://www.javacodegeeks.com/2017/07/instrumenting-java-web-applications-without-modifying-source-code.html

java web源代码

java web源代码_检测Java Web应用程序而无需修改其源代码相关推荐

  1. 检测Java Web应用程序而无需修改其源代码

    与其他系统进行交互时,大多数Java Web应用程序都使用标准Java接口. 使用接口javax.servlet.Servlet来实现基于HTTP的服务,例如网页或REST服务器. 使用JDBC接口j ...

  2. java魔塔源代码_魔塔Java开源(素材+源码)

    [实例简介] 魔塔Java开源(素材+源码),内包含魔塔人物以及怪物素材,代码简明,可用作学习参考 [实例截图] [核心代码] MyMota └── MyMota ├── bin │   └── Mo ...

  3. java案例源代码_求java案例源代码 越多越好!

    展开全部 import java.awt.*; import java.awt.event.*; import java.lang.*; import javax.swing.*; public cl ...

  4. java web构建_使用Java构建一个宁静的Web服务

    java web构建 介绍 (Introduction) Due to its exponential growth, REST(Representational State Transfer) ha ...

  5. java web服务_将Java服务公开为Web服务

    java web服务 本教程解决了开发人员面临的最实际的情况. 大多数时候,我们可能需要将某些现有服务公开为Web服务. 在项目生命周期的不同阶段可能会遇到这种情况. 如果这是初始阶段,那么您几乎是安 ...

  6. 如何搭建java web环境_搭建java WEB开发环境和应用

    搭建java WEB开发环境和应用 更新时间:2009年06月07日 02:28:56   作者: 使用Tomcat服务器,使用DBCP数据源搭建Web开发环境 一 使用Tomcat服务器,使用DBC ...

  7. java web 编辑器_基于Java+web的在线Java编辑器 PDF 下载

    主要内容: 近些年,互联网技术飞速发展,越来 越多的人接触到了编程语言,同时更多的 人愿意去了解学习编程语言,但由于以往 的编程语言编译器安装复杂,且部分还需 要配置环境,所以一些想学习编程语言的 人 ...

  8. java反序列化终极工具_检测java反序列化漏洞

    java反序列化终极测试工具是一款检测java反序列化漏洞工具,直接将Jboss.Websphere和weblogic的反序列化漏洞的利用集成到了一起.java反序列化漏洞已经被曝出一段时间了,其强大 ...

  9. java socket 获取网页源代码_通过java.net.Socket 类抓取网页内容

    /** 文件名 : Firstsocket.java 描述 :通过 java.net.Socket 类访问一个web页面,并且返回结果 作者: 慈勤强 cqq1978@yeah.net 参考:http ...

最新文章

  1. 《OpenCV3编程入门》学习笔记7 图像变换(三 )重映射
  2. alpine 编译c语言,Docker alpine中编译和测试Go RocketMQ Client
  3. VMware 虚拟化编程(6) — VixDiskLib 虚拟磁盘库详解之二
  4. JZOJ 5483. 【清华集训2017模拟11.26】简单路径
  5. 在IDEA下导入Maven项目之后 Dependencies报红线
  6. 什么是OLAP和OLTP
  7. Java机器学习,第2部分
  8. 计算机图形学画线_在计算机图形学中直接使用线方程
  9. [html] 页面布局时你使用最多的标签是什么?div吗?在什么情况下会使用到div?
  10. java学习——equals()和==的比较
  11. UI设计字体素材|设计中字母间距指南
  12. ef6 oracle 存储过程,具有多个结果集的存储过程-EF6 | Microsoft Docs
  13. m3u8 视频下载 ——M3U8 GETTER 批量下载版 使用说明
  14. 思科实验4.4.1.2-Packet Tracer - 综合技能练习
  15. 第六版PMBOK豆知识
  16. 多端开发之uniapp开发app
  17. 中国钢铁产业产量分析与市场需求状况研究报告2022版
  18. 【三维目标检测】3DSSD(一)
  19. 【吉大刘大有数据结构绿皮书】向LinkedList类中增加一个函数Contrary,功能为将其所有结点按相反次序链接。
  20. zsh与oh-my-zsh ← 阳志平的个人网站::技术 zsh与oh-my-zsh ← 阳志平的个人网站::技术...

热门文章

  1. 周末狂欢赛1(玩游戏/Game,函数,JOIOI王国)
  2. YBTOJ:伞兵空降(二分图匹配)
  3. P3288-[SCOI2014]方伯伯运椰子【0/1分数规划,负环】
  4. P5305-[GXOI/GZOI2019]旧词【树链剖分,线段树】
  5. CF496E-Distributing Parts【平衡树,贪心】
  6. 【jzoj】2018.2.5NOIP普及组——C组模拟赛
  7. codeforces855 C. Helga Hufflepuff‘s Cup(树形dp)
  8. 星座图(2020特长生 T4)
  9. 【斜率优化】仓库建设(luogu 2120)
  10. Sentinel(二十二)之使用Nacos存储规则