opentelemetry-java-instrume.../instrumentation/servlet
Gregor Zeitlinger b9f09cae88
Http server builder (#11651)
2024-10-07 09:41:24 -07:00
..
servlet-2.2/javaagent Don't use HttpServerTest.controller in java tests (#12347) 2024-09-29 17:09:25 -07:00
servlet-3.0 Don't use HttpServerTest.controller in java tests (#12347) 2024-09-29 17:09:25 -07:00
servlet-5.0 convert servlet tests to java (#12013) 2024-09-24 07:53:45 +03:00
servlet-common Http server builder (#11651) 2024-10-07 09:41:24 -07:00
servlet-javax-common/javaagent Refactor HTTP server `server.address` and `server.port` attributes (#9760) 2023-10-30 08:55:30 +01:00
README.md Move capturing enduser.id attribute behind a flag (#9751) 2023-10-31 15:28:32 +02:00

README.md

Instrumentation for Java Servlets

Settings

System property Type Default Description
otel.instrumentation.servlet.experimental-span-attributes Boolean false Enable the capture of experimental span attributes.
otel.instrumentation.servlet.experimental.capture-request-parameters List Empty Request parameters to be captured (experimental).

A word about version

We support Servlet API starting from version 2.2. But various instrumentations apply to different versions of the API.

They are divided into the following sub-modules:

  • servlet-common contains shared code for both javax.servlet and jakarta.servlet packages.
  • Version-specific modules contain the version-specific instrumentations and request/response accessor.
    • servlet-javax-common contains instrumentations common for Servlet API versions [2.2, 5)
    • servlet-2.2 contains instrumentation for Servlet API versions [2.2, 3)
    • servlet-3.0 contains instrumentation for Servlet API versions [3.0, 5)
    • servlet-5.0 contains instrumentation for Servlet API versions [5,)

Implementation details

In order to fully understand how java servlet instrumentation work, let us first take a look at the following stacktrace from Spring PetClinic application. Unimportant frames are redacted, points of interests are highlighted and discussed below.

at org.springframework.samples.petclinic.owner.OwnerController.initCreationForm(OwnerController.java:60)
...
at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87)
at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1040)
at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:943)
at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006)
at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:898)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:634)
at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:741)
...
at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:119)
...
at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49)
at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1128)
at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628)
at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61)
at java.base/java.lang.Thread.run(Thread.java:834)

Everything starts when HTTP request processing reaches the first class from Servlet specification. In the example above this is the OncePerRequestFilter.doFilter(ServletRequest, ServletResponse, FilterChain) method. Let us call this first servlet specific method an "entry point". This is the main target for Servlet3Instrumentation and Servlet2Instrumentation:

public void javax.servlet.Filter#doFilter(ServletRequest, ServletResponse, FilterChain)

public void javax.servlet.http.HttpServlet#service(ServletRequest, ServletResponse).

These instrumentations are located in separate submodules servlet-3.0, servlet-2.2 and servlet-5.0, because they and corresponding tests depend on different versions of the servlet specification.

At last, request processing may reach the specific framework that your application uses. In this case Spring MVC and OwnerController.initCreationForm.

If all instrumentations are enabled, then a new span will be created for every highlighted frame. All spans from Servlet API will have kind=SERVER and name based on request method, such as HTTP GET, or servlet/filter mapping corresponding to the requested path, such as /foo/bar/*. Span created by Spring MVC instrumentation will have kind=INTERNAL and named OwnerController.initCreationForm.

The state described above has one significant problem. Observability backends usually aggregate traces based on their root spans. This means that for applications with a single controller Servlet that services all requests ALL traces from that application will be grouped together because their root spans will all have the same named based on common entry point. In order to alleviate this problem, instrumentations for specific frameworks, such as Spring MVC here, update name of the span corresponding to the entry point. Each framework instrumentation can decide what is the best span name based on framework implementation details. Of course, still adhering to OpenTelemetry semantic conventions.

Additional instrumentations

HttpServletResponseInstrumentation instruments javax.servlet.http.HttpServletResponse.sendError and javax.servlet.http.HttpServletResponse.sendRedirect methods to create new INTERNAL spans around their invocations.