blob: 203edba15d2c278bcae25658b70e95db6dfb60ab [file]
<!DOCTYPE html>
<!--
| Generated by Apache Maven Doxia Site Renderer 2.0.0 from src/site/xdoc/docs/spring-boot-starter.xml at 2026-05-18
| Rendered using Apache Maven Fluido Skin 2.0.0-M11
-->
<html xmlns="http://www.w3.org/1999/xhtml" lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="generator" content="Apache Maven Doxia Site Renderer 2.0.0" />
<title>Apache Axis2 Spring Boot Starter – Apache Axis2</title>
<link rel="stylesheet" href="../css/apache-maven-fluido-2.0.0-M11.min.css" />
<link rel="stylesheet" href="../css/site.css" />
<link rel="stylesheet" href="../css/print.css" media="print" />
<script src="../js/apache-maven-fluido-2.0.0-M11.min.js"></script>
<meta http-equiv="content-type" content="" /> </head>
<body>
<div class="container-fluid container-fluid-top">
<header>
<div id="banner">
<div class="pull-left"><div id="bannerLeft"><h1><a href="https://www.apache.org/"><img class="class java.lang.Object" src="https://www.apache.org/images/asf_logo_wide.png" /> Apache Axis2</a></h1></div></div>
<div class="pull-right"><div id="bannerRight"><h1><a href="https://axis.apache.org/axis2/java/core/"><img class="class java.lang.Object" src="https://axis.apache.org/axis2/java/core/images/axis.jpg" /></a></h1></div></div>
<div class="clear"><hr/></div>
</div>
<div id="breadcrumbs">
<ul class="breadcrumb">
<li id="publishDate">Last Published: 2026-05-17<span class="divider">|</span>
</li>
<li id="projectVersion">Version: 2.0.1<span class="divider">|</span></li>
<li><a href="https://www.apache.org" class="externalLink">Apache</a><span class="divider">/</span></li>
<li><a href="../index.html">Axis2/Java</a><span class="divider">/</span></li>
<li class="active">Apache Axis2 Spring Boot Starter</li>
</ul>
</div>
</header>
<div class="row-fluid">
<header id="leftColumn" class="span2">
<nav class="well sidebar-nav">
<ul class="nav nav-list">
<li class="nav-header">Axis2/Java</li>
<li><a href="../index.html">Home</a></li>
<li><a href="../download.html">Downloads</a></li>
<li><a href="javascript:void(0)"><span class="icon-chevron-down"></span>Release Notes</a>
<ul class="nav nav-list">
<li><a href="../release-notes/1.6.1.html">1.6.1</a></li>
<li><a href="../release-notes/1.6.2.html">1.6.2</a></li>
<li><a href="../release-notes/1.6.3.html">1.6.3</a></li>
<li><a href="../release-notes/1.6.4.html">1.6.4</a></li>
<li><a href="../release-notes/1.7.0.html">1.7.0</a></li>
<li><a href="../release-notes/1.7.1.html">1.7.1</a></li>
<li><a href="../release-notes/1.7.2.html">1.7.2</a></li>
<li><a href="../release-notes/1.7.3.html">1.7.3</a></li>
<li><a href="../release-notes/1.7.4.html">1.7.4</a></li>
<li><a href="../release-notes/1.7.5.html">1.7.5</a></li>
<li><a href="../release-notes/1.7.6.html">1.7.6</a></li>
<li><a href="../release-notes/1.7.7.html">1.7.7</a></li>
<li><a href="../release-notes/1.7.8.html">1.7.8</a></li>
<li><a href="../release-notes/1.7.9.html">1.7.9</a></li>
<li><a href="../release-notes/1.8.0.html">1.8.0</a></li>
<li><a href="../release-notes/1.8.1.html">1.8.1</a></li>
<li><a href="../release-notes/1.8.2.html">1.8.2</a></li>
<li><a href="../release-notes/2.0.0.html">2.0.0</a></li>
<li><a href="../release-notes/2.0.1.html">2.0.1</a></li>
</ul></li>
<li><a href="../modules/index.html">Modules</a></li>
<li><a href="../tools/index.html">Tools</a></li>
<li class="nav-header">Documentation</li>
<li><a href="../docs/toc.html">Table of Contents</a></li>
<li><a href="../docs/installationguide.html">Installation Guide</a></li>
<li><a href="../docs/quickstartguide.html">QuickStart Guide</a></li>
<li><a href="../docs/userguide.html">User Guide</a></li>
<li><a href="../docs/jaxws-guide.html">JAXWS Guide</a></li>
<li><a href="../docs/pojoguide.html">POJO Guide</a></li>
<li><a href="../docs/spring.html">Spring Guide</a></li>
<li><a href="../docs/webadminguide.html">Web Administrator&apos;s Guide</a></li>
<li><a href="../docs/migration.html">Migration Guide (from Axis1)</a></li>
<li class="nav-header">Resources</li>
<li><a href="../faq.html">FAQ</a></li>
<li><a href="https://github.com/apache/axis-axis2-java-core" class="externalLink">Source Code</a></li>
<li class="nav-header">Get Involved</li>
<li><a href="../overview.html">Overview</a></li>
<li><a href="../mail-lists.html">Mailing Lists</a></li>
<li><a href="../release-process.html">Release Process</a></li>
<li><a href="../guidelines.html">Developer Guidelines</a></li>
<li><a href="../siteHowTo.html">Build the Site</a></li>
<li class="nav-header">Project Information</li>
<li><a href="https://github.com/apache/axis-axis2-java-core/graphs/contributors" class="externalLink">Contributors</a></li>
<li><a href="https://issues.apache.org/jira/projects/AXIS2/issues" class="externalLink">Issues</a></li>
<li class="nav-header">Apache</li>
<li><a href="https://www.apache.org/licenses/LICENSE-2.0.html" class="externalLink">License</a></li>
<li><a href="https://www.apache.org/foundation/sponsorship.html" class="externalLink">Sponsorship</a></li>
<li><a href="https://www.apache.org/foundation/thanks.html" class="externalLink">Thanks</a></li>
<li><a href="https://www.apache.org/security/" class="externalLink">Security</a></li>
</ul>
</nav>
<div class="well sidebar-nav">
<div id="poweredBy">
<div class="clear"></div>
<div class="clear"></div>
<a href="https://maven.apache.org/" class="builtBy" target="_blank"><img class="builtBy" alt="Built by Maven" src="../images/logos/maven-feather.png" /></a>
</div>
</div>
</header>
<main id="bodyColumn" class="span10">
<html>
<section><a id="Apache_Axis2_Spring_Boot_Starter"></a>
<h1>Apache Axis2 Spring Boot Starter</h1>
<p><strong>axis2-spring-boot-starter</strong> reduces Axis2 + Spring Boot integration
from a multi-day configuration project to a single Maven dependency. It handles
servlet registration, repository configuration, and OpenAPI/MCP endpoint activation
with sensible defaults &#x2014; for both SOAP and JSON-RPC services.</p>
<section><section><a id="What_Axis2_brings_to_Spring_Boot"></a>
<h3>What Axis2 brings to Spring Boot</h3>
<ul>
<li><strong>Large JSON streaming (10MB&#x2013;1GB+)</strong> &#x2014; the
<a href="json-streaming-formatter.html">streaming JSON formatter</a>
flushes HTTP/2 DATA frames every 64KB, keeping server memory flat
regardless of response size. No reactive stack required.</li>
<li><strong>Native HTTP/2 with backpressure</strong> &#x2014; the
<a href="http2-transport-additions.html">H2TransportSender</a>
provides connection multiplexing, flow control, and ALPN negotiation
as a drop-in transport layer. See also the
<a href="http2-integration-guide.html">HTTP/2 overview</a>.</li>
<li><strong>AI-ready services (MCP)</strong> &#x2014; Axis2 auto-generates an
<a href="json-rpc-mcp-guide.html">MCP tool catalog</a> from deployed
services. AI assistants discover and call your services as tools
without hand-authored definitions. See
<a href="mcp-examples.html">MCP examples</a> for live demos.</li>
<li><strong>Response field selection</strong> &#x2014; the
<a href="json-streaming-formatter.html#Field_Selection"><code>?fields=</code></a>
query parameter lets clients request only the fields they need,
reducing payload size and AI token consumption at the
<a href="json-streaming-formatter.html">serialization layer</a>
with zero application code.</li>
<li><strong>OpenAPI schemas from Hibernate mappings</strong> &#x2014; the
<a href="openapi-jpa-schema.html">JPA/Hibernate schema generator</a>
produces read and write JSON Schemas directly from
<code>@Entity</code> annotations or <code>.hbm.xml</code> files,
including a <a href="openapi-jpa-schema.html#batch_generation">batch CLI tool</a>
for large codebases. No hand-authored API contracts &#x2014; the schema stays
in sync with the database model by construction.</li>
<li><strong>Built-in pagination</strong> &#x2014; the
<a href="json-pagination.html"><code>PaginatedResponse</code></a>
wrapper provides offset/limit pagination with
<a href="json-pagination.html#safety">server-enforced safety limits</a>,
<code>totalCount</code> for
<a href="json-pagination.html#frontend_patterns">page controls</a>,
and <code>hasMore</code> for infinite scroll &#x2014; mapping directly to
JPA/Hibernate's <code>setFirstResult</code>/<code>setMaxResults</code>.</li>
<li><strong>SOAP + JSON from the same services</strong> &#x2014; serve both
SOAP/XML and JSON-RPC from the same Spring beans, selected by
<a href="#soap_vs_json">configuration</a>. Useful for enterprises
maintaining both legacy and modern clients.</li>
</ul>
<p>This starter supports both <strong>WAR deployment</strong> to an external container
(Tomcat 11, WildFly 32/39) and <strong>embedded Tomcat</strong> for local development.
For embedded mode, set <code>axis2.repo</code> in <code>application.properties</code>
to the exploded WAR directory from your build.
See <a href="#deployment_model">Deployment Model</a> for details.</p>
<ul>
<li><a href="#deployment_model">0. Deployment Model (WAR + Embedded)</a></li>
<li><a href="#quickstart">1. Quick Start</a></li>
<li><a href="#how_axisservlet_works">2. How AxisServlet Works (SOAP vs JSON)</a></li>
<li><a href="#properties">3. Configuration Properties</a></li>
<li><a href="#soap_vs_json">4. SOAP Mode vs JSON Mode</a></li>
<li><a href="#what_autoconfigured">5. What Gets Autoconfigured</a></li>
<li><a href="#what_stays">6. What Stays in Your Application</a></li>
<li><a href="#openapi_mcp">7. OpenAPI and MCP Support</a></li>
<li><a href="#migration">8. Migration from Manual Configuration</a></li>
<li><a href="#containers">9. Container Support</a></li>
<li><a href="#http2">10. HTTP/2 and Streaming</a></li>
</ul>
<a id="deployment_model"></a>
</section></section><section><a id="a0._Deployment_Model"></a>
<h2>0. Deployment Model</h2>
<p>Axis2 supports two Spring Boot deployment modes:</p>
<table class="bodyTableBorder">
<tr class="a">
<th>Mode</th>
<th>How</th>
<th>Use Case</th></tr>
<tr class="b">
<td><strong>External container (WAR)</strong></td>
<td>Package as WAR, deploy to Tomcat 11 or WildFly 32+</td>
<td>Production &#x2014; the recommended deployment model</td>
</tr>
<tr class="a">
<td><strong>Embedded Tomcat</strong></td>
<td><code>mvn spring-boot:run -Pembedded</code></td>
<td>Local development and testing</td>
</tr>
</table>
<section><a id="External_container_.28production.29"></a>
<h3>External container (production)</h3>
<p>For production, deploy as a WAR to an external servlet container.
Your Spring Boot application must extend
<code>SpringBootServletInitializer</code> and be packaged as a WAR.
External containers provide operational advantages for long-running
Axis2 deployments:</p>
<ul>
<li>Container-managed TLS, thread pools, and connection pools</li>
<li>Hot-redeploy via exploded WAR without JVM restart</li>
<li>Central certificate management (Tomcat's <code>server.xml</code>, WildFly's Elytron)</li>
<li>HTTP/2 with ALPN (via container connector, not application code)</li>
</ul>
</section><section><a id="Embedded_Tomcat_.28development.29"></a>
<h3>Embedded Tomcat (development)</h3>
<p>For local development and testing, the sample application supports
Spring Boot's embedded Tomcat via a Maven profile. Axis2's
<code>WarBasedAxisConfigurator</code> requires a filesystem layout
(<code>WEB-INF/conf/axis2.xml</code>, <code>WEB-INF/services/*.aar</code>,
<code>WEB-INF/modules/*.mar</code>) that embedded Tomcat does not
provide from its temp docbase. The <code>-Daxis2.repo</code> system
property tells the configurator where to find the exploded WAR
from the build:</p>
<pre>
# Build first (creates the exploded WAR with services and modules)
mvn clean install -DskipTests
# Run with embedded Tomcat
mvn spring-boot:run -Pembedded \
-Dspring-boot.run.arguments=&quot;--server.port=9090&quot; \
-Dspring-boot.run.jvmArguments=&quot;-Daxis2.repo=target/deploy/axis2-json-api/WEB-INF&quot;
</pre>
<p>The sample application requires authentication. Log in first to
obtain a token, then pass it as a Bearer token on subsequent
requests:</p>
<pre>
# 1. Log in (sample credentials)
TOKEN=$(curl -s http://localhost:9090/services/loginService \
-H 'Content-Type: application/json' \
-d '{&quot;doLogin&quot;:[{&quot;arg0&quot;:{&quot;email&quot;:&quot;java-dev@axis.apache.org&quot;,&quot;credentials&quot;:&quot;userguide&quot;}}]}' \
| python3 -c &quot;import sys,json; print(json.load(sys.stdin)['response']['token'])&quot;)
# 2. Call a service with the token
curl -s http://localhost:9090/services/FinancialBenchmarkService \
-H 'Content-Type: application/json' \
-H &quot;Authorization: Bearer $TOKEN&quot; \
-d '{&quot;portfolioVariance&quot;:[{&quot;arg0&quot;:{&quot;weights&quot;:[0.5,0.5],&quot;covarianceMatrix&quot;:[[0.04,0.006],[0.006,0.09]]}}]}'
</pre>
<p>OpenAPI and MCP endpoints do not require authentication:</p>
<pre>
curl -s http://localhost:9090/openapi.json
curl -s http://localhost:9090/openapi-mcp.json
</pre>
<p>For production, external containers remain the recommended deployment:</p>
<ul>
<li>Container-managed thread pools, connection pools, and monitoring (Actuator/Micrometer)</li>
<li>Hot-redeploy via exploded WAR without JVM restart</li>
<li>Central TLS/certificate management (Tomcat's <code>server.xml</code>, WildFly's Elytron)</li>
<li>Established operational tooling (WildFly CLI, Tomcat Manager, deployment scanners)</li>
</ul>
</section><section><a id="Which_external_containers.3F"></a>
<h3>Which external containers?</h3>
<table class="bodyTableBorder">
<tr class="a">
<th>Container</th>
<th>JDK</th>
<th>Tested</th></tr>
<tr class="b">
<td>Apache Tomcat 11</td>
<td>OpenJDK 17, 21, 25</td>
<td>Yes (21/25 live-tested; 17 is the minimum supported by Tomcat 11 and Spring Boot 3.x)</td></tr>
<tr class="a">
<td>WildFly 32</td>
<td>OpenJDK 17, 21</td>
<td>Yes (production-tested on WildFly 32 + OpenJDK 17)</td></tr>
<tr class="b">
<td>WildFly 39</td>
<td>OpenJDK 21, 25</td>
<td>Yes (live-tested with OpenJDK 25)</td></tr>
</table>
<p><strong>Minimum JDK version: OpenJDK 17.</strong> Axis2 core and the starter
compile and pass CI tests with <code>-Dmaven.compiler.release=17</code>.
Spring Boot 3.x and Tomcat 11 both require Java 17 as a minimum. The sample
applications target Java 21 source level but Axis2 itself imposes no version
above 17.</p>
<p>Other Jakarta EE 9+ containers (WebSphere Liberty, Payara) should work but
are not tested.</p>
</section><section><a id="Embedded_mode_limitations"></a>
<h3>Embedded mode limitations</h3>
<p>Embedded Tomcat requires a pre-built exploded WAR on the filesystem
(pointed to by <code>axis2.repo</code>). This means you must run
<code>mvn install</code> before <code>mvn spring-boot:run</code> so
that the <code>.aar</code> service archives and <code>axis2.xml</code>
are staged in the build output directory. Changes to service metadata
(operations, message receivers, MCP descriptions) require a rebuild.</p>
<p>Note that the service <em>code</em> itself is a Spring
<code>@Component</code> bean on the classpath &#x2014; the <code>.aar</code>
file contains only metadata (<code>services.xml</code>) that tells
Axis2 which Spring bean to call, which operations to expose, and
which message receivers to use. The <code>SpringBeanName</code>
parameter in <code>services.xml</code> bridges Axis2 service
dispatch to Spring's application context. A future improvement
could scan for <code>services.xml</code> resources on the classpath,
eliminating the filesystem requirement entirely.</p>
<a id="quickstart"></a>
</section></section><section><a id="a1._Quick_Start"></a>
<h2>1. Quick Start</h2>
<p>Add the starter to your <code>pom.xml</code>:</p>
<pre>
&lt;dependency&gt;
&lt;groupId&gt;org.apache.axis2&lt;/groupId&gt;
&lt;artifactId&gt;axis2-spring-boot-starter&lt;/artifactId&gt;
&lt;version&gt;2.0.1&lt;/version&gt;
&lt;/dependency&gt;
&lt;!-- Optional: OpenAPI/MCP endpoints --&gt;
&lt;dependency&gt;
&lt;groupId&gt;org.apache.axis2&lt;/groupId&gt;
&lt;artifactId&gt;axis2-openapi&lt;/artifactId&gt;
&lt;version&gt;2.0.1&lt;/version&gt;
&lt;/dependency&gt;
</pre>
<p>This single dependency replaces ~30 individual Axis2 dependencies
(<code>axis2-kernel</code>, <code>axis2-transport-http</code>,
<code>axis2-json</code>, <code>axis2-spring</code>, <code>axis2-jaxws</code>,
etc.) and eliminates the need for a hand-coded <code>Axis2WebAppInitializer</code>
or <code>OpenApiServlet</code> class.</p>
<p>The starter auto-registers <code>AxisServlet</code> at <code>/services/*</code>.
If you also add the optional <code>axis2-openapi</code> dependency (shown above),
the starter additionally registers endpoints for OpenAPI spec generation
(<code>/openapi.json</code>), Swagger UI (<code>/swagger-ui</code>), and MCP
tool discovery (<code>/openapi-mcp.json</code>). See
<a href="#openapi_mcp">Section 7</a> for details.</p>
<a id="how_axisservlet_works"></a>
</section><section><a id="a2._How_AxisServlet_Works_.28SOAP_vs_JSON.29"></a>
<h2>2. How AxisServlet Works (SOAP vs JSON)</h2>
<p><code>org.apache.axis2.transport.http.AxisServlet</code> is the single entry
point for <strong>all</strong> Axis2 service calls &#x2014; SOAP and JSON-RPC alike.
The servlet itself is protocol-agnostic. It does not know or care whether the
incoming request is SOAP 1.1, SOAP 1.2, or JSON-RPC.</p>
<p>The protocol handling is determined entirely by <code>axis2.xml</code>:</p>
<table class="bodyTableBorder">
<tr class="a">
<th>Component</th>
<th>SOAP</th>
<th>JSON-RPC</th></tr>
<tr class="b">
<td><strong>Message Receivers</strong><br />Parse the incoming payload and invoke the service method</td>
<td><code>RawXMLINOutMessageReceiver</code><br /><code>RawXMLINOnlyMessageReceiver</code><br />(read SOAP XML envelopes)</td>
<td><code>JsonRpcMessageReceiver</code><br /><code>JsonInOnlyRPCMessageReceiver</code><br />(read JSON-RPC <code>{&quot;op&quot;:[{&quot;arg0&quot;:{}}]}</code> envelope)</td>
</tr>
<tr class="a">
<td><strong>Dispatchers</strong><br />Route the request to the correct service and operation</td>
<td>Multi-handler chain:<br /><code>SOAPActionBasedDispatcher</code><br /><code>RequestURIBasedDispatcher</code><br /><code>SOAPMessageBodyBasedDispatcher</code><br /><code>HTTPLocationBasedDispatcher</code><br />(routes by SOAPAction header, WS-Addressing, message body)</td>
<td>Single handler:<br /><code>JSONBasedDefaultDispatcher</code><br />(routes by URL path only)</td>
</tr>
<tr class="b">
<td><strong><code>enableJSONOnly</code></strong></td>
<td><code>false</code> (the default since Axis2 1.0, 2006)</td>
<td><code>true</code> (skips SOAP envelope processing)</td>
</tr>
</table>
<p><strong>SOAP and JSON cannot coexist in a single deployment.</strong>
The message receivers and dispatchers are incompatible &#x2014; a SOAP receiver cannot
parse JSON, and the JSON dispatcher cannot route by SOAPAction. This is not a
starter limitation; it is how Axis2's processing pipeline works.</p>
<section><a id="AxisServlet_Initialization_Sequence"></a>
<h3>AxisServlet Initialization Sequence</h3>
<p>When AxisServlet starts (<code>init()</code>), it:</p>
<ol style="list-style-type: decimal;">
<li>Creates a <code>WarBasedAxisConfigurator</code></li>
<li>Locates the repository (<code>WEB-INF/</code>) using, in order:
<ol style="list-style-type: decimal;">
<li><code>axis2.repository.path</code> servlet init-parameter (if set explicitly)</li>
<li><code>axis2.repository.url</code> servlet init-parameter (if set explicitly)</li>
<li><code>ServletContext.getRealPath(&quot;/WEB-INF&quot;)</code> (automatic fallback)</li>
</ol>
</li>
<li>Reads <code>WEB-INF/conf/axis2.xml</code> to configure the engine</li>
<li>Scans <code>WEB-INF/services/*.aar</code> to deploy services</li>
<li>Scans <code>WEB-INF/modules/*.mar</code> to load modules (e.g., OpenAPI)</li>
<li>Stores the <code>ConfigurationContext</code> in the <code>ServletContext</code>
&#x2014; this is how other servlets (OpenAPI, MCP) access the Axis2 engine</li>
</ol>
<p>The starter sets <code>axis2.repository.path</code> eagerly at startup as a
workaround for WildFly VFS timing issues where <code>getRealPath()</code> can
return <code>null</code> during lazy servlet initialization.</p>
<a id="properties"></a>
</section></section><section><a id="a3._Configuration_Properties"></a>
<h2>3. Configuration Properties</h2>
<table class="bodyTableBorder">
<tr class="a">
<th>Property</th>
<th>Default</th>
<th>Description</th></tr>
<tr class="b">
<td><code>axis2.enabled</code></td>
<td><code>true</code></td>
<td>Set to <code>false</code> to disable Axis2 entirely (e.g., for server roles that don't serve web services)</td></tr>
<tr class="a">
<td><code>axis2.mode</code></td>
<td><code>json</code></td>
<td><code>soap</code> or <code>json</code> &#x2014; selects the built-in axis2.xml template (see Section 4)</td></tr>
<tr class="b">
<td><code>axis2.services-path</code></td>
<td><code>/services</code></td>
<td>URL path for AxisServlet. Servlet is registered at <code>{services-path}/*</code></td></tr>
<tr class="a">
<td><code>axis2.configuration-file</code></td>
<td><em>(empty)</em></td>
<td>Path to a custom axis2.xml. Overrides <code>axis2.mode</code>. Supports <code>classpath:</code> prefix</td></tr>
<tr class="b">
<td><code>axis2.openapi.enabled</code></td>
<td><code>true</code></td>
<td>Register OpenAPI/MCP servlet (requires <code>axis2-openapi</code> on classpath)</td></tr>
<tr class="a">
<td><code>axis2.openapi.paths</code></td>
<td><code>/openapi.json, /openapi.yaml, /swagger-ui, /openapi-mcp.json</code></td>
<td>URL paths for OpenAPI endpoints</td></tr>
</table>
<a id="soap_vs_json"></a>
</section><section><a id="a4._SOAP_Mode_vs_JSON_Mode"></a>
<h2>4. SOAP Mode vs JSON Mode</h2>
<p>The starter ships two built-in <code>axis2.xml</code> templates, selected by
<code>axis2.mode</code>:</p>
<section><a id="JSON_mode_.28axis2.mode.3Djson.2C_the_default.29"></a>
<h3>JSON mode (<code>axis2.mode=json</code>, the default)</h3>
<pre>
# application.properties
axis2.mode=json
</pre>
<p>Uses <code>JsonRpcMessageReceiver</code>, <code>JSONBasedDefaultDispatcher</code>,
and <code>enableJSONOnly=true</code>. Choose this for new services, MCP/AI integration,
and REST/OpenAPI consumers.</p>
</section><section><a id="SOAP_mode_.28axis2.mode.3Dsoap.29"></a>
<h3>SOAP mode (<code>axis2.mode=soap</code>)</h3>
<pre>
# application.properties
axis2.mode=soap
</pre>
<p>Uses <code>RawXMLINOutMessageReceiver</code>, the full SOAP dispatcher stack,
and <code>enableJSONOnly=false</code>. Choose this for WSDL-first services,
WS-Security, WS-Addressing, or interop with .NET/PHP SOAP clients. This is the
classic Axis2 configuration that has been in use since version 1.0 (2006).</p>
</section><section><a id="Custom_axis2.xml"></a>
<h3>Custom axis2.xml</h3>
<pre>
# application.properties
axis2.configuration-file=classpath:my-axis2.xml
</pre>
<p>Overrides both built-in templates. Use this when you need custom phases,
modules, transport configurations, or any axis2.xml settings not covered by
the built-in templates.</p>
<p>The starter only stages axis2.xml if <code>WEB-INF/conf/axis2.xml</code>
does not already exist. If your Maven build pre-stages it (e.g., via
<code>maven-antrun-plugin</code>), the starter will not overwrite it.</p>
<a id="what_autoconfigured"></a>
</section></section><section><a id="a5._What_Gets_Autoconfigured"></a>
<h2>5. What Gets Autoconfigured</h2>
<table class="bodyTableBorder">
<tr class="a">
<th>Component</th>
<th>What it replaces</th>
<th>Condition</th></tr>
<tr class="b">
<td><code>AxisServlet</code> registration at <code>/services/*</code></td>
<td>Hand-coded <code>Axis2WebAppInitializer</code></td>
<td>Always (when <code>axis2.enabled=true</code>)</td>
</tr>
<tr class="a">
<td><code>axis2.xml</code> staging (SOAP or JSON template)</td>
<td><code>axis2.xml</code> copy step in <code>maven-antrun-plugin</code></td>
<td>When <code>WEB-INF/conf/axis2.xml</code> does not already exist</td>
</tr>
<tr class="b">
<td>OpenAPI/MCP servlet at <code>/openapi.json</code>, <code>/swagger-ui</code>, <code>/openapi-mcp.json</code></td>
<td>Hand-coded <code>OpenApiServlet</code> class</td>
<td>When <code>axis2-openapi</code> is on classpath and <code>axis2.openapi.enabled=true</code></td>
</tr>
</table>
<a id="what_stays"></a>
</section><section><a id="a6._What_Stays_in_Your_Application"></a>
<h2>6. What Stays in Your Application</h2>
<p>The starter deliberately does <strong>not</strong> autoconfigure:</p>
<ul>
<li><strong>Security</strong> &#x2014; JWT authentication, mTLS, SAML, CSRF guard filter
chains are application-specific. The starter does not provide any
<code>SecurityFilterChain</code> beans.</li>
<li><strong>Service definitions</strong> &#x2014; <code>services.xml</code> files inside
<code>.aar</code> archives define operation names, Spring bean names,
per-operation message receivers. These are service-specific.</li>
<li><strong><code>.aar</code> file creation</strong> &#x2014; pre-built <code>.aar</code> files
must be staged in <code>WEB-INF/services/</code> by the Maven build
(<code>maven-antrun-plugin</code> or <code>axis2-aar-maven-plugin</code>).</li>
<li><strong><code>.mar</code> module staging</strong> &#x2014; the OpenAPI module JAR must be
renamed to <code>.mar</code> and placed in <code>WEB-INF/modules/</code> by the
Maven build.</li>
<li><strong>Container-specific descriptors</strong> &#x2014; WildFly's
<code>jboss-deployment-structure.xml</code>, <code>jboss-web.xml</code>,
<code>beans.xml</code>.</li>
<li><strong><code>@SpringBootApplication</code> class</strong> &#x2014; the consuming app
still needs its main class extending <code>SpringBootServletInitializer</code>.</li>
</ul>
<a id="openapi_mcp"></a>
</section><section><a id="a7._OpenAPI_and_MCP_Support"></a>
<h2>7. OpenAPI and MCP Support</h2>
<p><code>axis2-openapi</code> is a separate Axis2 module (Maven artifact:
<code>org.apache.axis2:axis2-openapi</code>) that auto-generates an
<a href="https://en.wikipedia.org/wiki/OpenAPI_Specification" class="externalLink">OpenAPI</a>
3.0.1 specification and
<a href="https://en.wikipedia.org/wiki/Model_Context_Protocol" class="externalLink">MCP</a>
(Model Context Protocol) tool catalog from your deployed Axis2 services.
OpenAPI is a standard format for describing REST APIs &#x2014; tools like
<a href="https://swagger.io/tools/swagger-ui/" class="externalLink">Swagger UI</a> use it to
generate interactive documentation. MCP is a protocol that lets AI
assistants (Claude, ChatGPT, etc.) discover and call your services
as tools.</p>
<p>The module auto-generates both from your deployed Axis2 services.
It reads each service's <code>services.xml</code> (inside the <code>.aar</code>
file) at runtime &#x2014; specifically the operation names, message receiver types,
and MCP metadata parameters &#x2014; and produces the specification automatically.
No annotations on your service code are needed.</p>
<p><strong>How it works with the starter:</strong> The starter detects
<code>axis2-openapi</code> on the classpath via Spring Boot's
<code>@ConditionalOnClass(OpenApiModule.class)</code> and automatically
registers a servlet that delegates to Axis2's
<a href="https://github.com/apache/axis-axis2-java-core/blob/master/modules/openapi/src/main/java/org/apache/axis2/openapi/SwaggerUIHandler.java" class="externalLink">SwaggerUIHandler</a>.
No additional code is needed in your application.</p>
<p>To enable it, add the dependency to your <code>pom.xml</code>:</p>
<pre>
&lt;dependency&gt;
&lt;groupId&gt;org.apache.axis2&lt;/groupId&gt;
&lt;artifactId&gt;axis2-openapi&lt;/artifactId&gt;
&lt;version&gt;2.0.1&lt;/version&gt;
&lt;/dependency&gt;
</pre>
<p>Your Maven build must also copy the <code>axis2-openapi</code> JAR as a
<code>.mar</code> (Module Archive) into <code>WEB-INF/modules/</code> so the
Axis2 engine can load it:</p>
<pre>
&lt;!-- In maven-antrun-plugin, prepare-package phase --&gt;
&lt;copy file=&quot;${settings.localRepository}/org/apache/axis2/axis2-openapi/${axis2.version}/axis2-openapi-${axis2.version}.jar&quot;
tofile=&quot;${project.build.directory}/${project.build.finalName}/WEB-INF/modules/openapi-${axis2.version}.mar&quot;
overwrite=&quot;true&quot;/&gt;
</pre>
<p>Once deployed, the following endpoints are available (no auth required):</p>
<ul>
<li><code>GET /openapi.json</code> &#x2014; OpenAPI 3.0.1 specification (JSON)</li>
<li><code>GET /openapi.yaml</code> &#x2014; same in YAML format</li>
<li><code>GET /swagger-ui</code> &#x2014; interactive Swagger UI</li>
<li><code>GET /openapi-mcp.json</code> &#x2014; MCP tool catalog for AI assistants
(includes operation names, input schemas, and payload templates)</li>
</ul>
<p>The MCP catalog is generated from <code>services.xml</code> parameters
like <code>mcpDescription</code> and <code>mcpInputSchema</code>.
See the <a href="json-rpc-mcp-guide.html">JSON-RPC MCP Integration Guide</a>
for the full catalog format and how AI assistants discover and call Axis2
services.</p>
<a id="migration"></a>
</section><section><a id="a8._Migration_from_Manual_Configuration"></a>
<h2>8. Migration from Manual Configuration</h2>
<p>If you have an existing Axis2 + Spring Boot project with manual servlet
registration, follow these steps:</p>
<ol style="list-style-type: decimal;">
<li><strong>Replace dependencies:</strong> Remove the ~30 individual Axis2
dependencies and add the single <code>axis2-spring-boot-starter</code>
dependency.</li>
<li><strong>Delete <code>Axis2WebAppInitializer</code></strong> (or your
equivalent <code>ServletContextInitializer</code> that registers
<code>AxisServlet</code>). The starter handles this.</li>
<li><strong>Delete <code>OpenApiServlet.java</code></strong> if you have one.
The starter provides it when <code>axis2-openapi</code> is on the classpath.</li>
<li><strong>Set <code>axis2.mode</code></strong> in <code>application.properties</code>:
<code>json</code> for JSON-RPC services, <code>soap</code> for SOAP services.</li>
<li><strong>Keep your Maven build steps</strong> that create <code>.aar</code> files
and copy <code>.mar</code> modules &#x2014; the starter does not replace these yet.</li>
<li><strong>Keep your security configuration</strong> &#x2014; the starter does not touch
Spring Security.</li>
</ol>
<p>For working reference implementations, see:</p>
<ul>
<li><code>modules/samples/userguide/src/userguide/springbootdemo-tomcat11/README.md</code></li>
<li><code>modules/samples/userguide/src/userguide/springbootdemo-wildfly/README.md</code></li>
</ul>
<a id="testing"></a>
</section><section><a id="a9._Tested_Configurations"></a>
<h2>9. Tested Configurations</h2>
<section><a id="JSON-RPC_.2F_MCP_.28axis2.mode.3Djson.29"></a>
<h3>JSON-RPC / MCP (axis2.mode=json)</h3>
<p>The starter was tested with the <code>springbootdemo-tomcat11</code> sample
application, replacing ~30 individual Axis2 dependencies with the single
<code>axis2-spring-boot-starter</code> dependency and removing the hand-coded
<code>Axis2WebAppInitializer</code>. All endpoints confirmed working over
HTTPS/HTTP2 on Tomcat 11 with OpenJDK 25:</p>
<table class="bodyTableBorder">
<tr class="a">
<th>Endpoint</th>
<th>Result</th></tr>
<tr class="b">
<td><code>GET /openapi.json</code></td>
<td>Pass &#x2014; OpenAPI 3.0.1 spec returned</td></tr>
<tr class="a">
<td><code>GET /openapi.yaml</code></td>
<td>Pass</td></tr>
<tr class="b">
<td><code>GET /openapi-mcp.json</code></td>
<td>Pass &#x2014; MCP tool catalog with full inputSchema</td></tr>
<tr class="a">
<td><code>GET /swagger-ui</code></td>
<td>Pass &#x2014; HTTP 200</td></tr>
<tr class="b">
<td><code>POST loginService/doLogin</code></td>
<td>Pass &#x2014; JWT token returned</td></tr>
<tr class="a">
<td><code>POST FinancialBenchmarkService/portfolioVariance</code></td>
<td>Pass &#x2014; SUCCESS, variance=0.03168</td></tr>
<tr class="b">
<td><code>POST FinancialBenchmarkService/monteCarlo</code></td>
<td>Pass &#x2014; SUCCESS, 10K simulations</td></tr>
<tr class="a">
<td><code>POST FinancialBenchmarkService/scenarioAnalysis</code></td>
<td>Pass &#x2014; SUCCESS</td></tr>
</table>
</section><section><a id="SOAP_.28axis2.mode.3Dsoap.29"></a>
<h3>SOAP (axis2.mode=soap)</h3>
<p>The SOAP axis2.xml template was validated by unit tests confirming correct
message receivers (<code>RawXMLINOutMessageReceiver</code>,
<code>RawXMLINOnlyMessageReceiver</code>), the full SOAP dispatcher stack
(<code>SOAPActionBasedDispatcher</code>, <code>RequestURIBasedDispatcher</code>,
<code>SOAPMessageBodyBasedDispatcher</code>, <code>HTTPLocationBasedDispatcher</code>),
and <code>enableJSONOnly=false</code>. The SOAP template is based on the
Axis2 configuration that has been in production since version 1.0 (2006) and is
identical in structure to the configuration used by the legacy userguide
<code>example1</code> Echo service (<code>MyService.echo()</code> with
<code>RawXMLINOutMessageReceiver</code>).</p>
</section><section><a id="Unit_test_suite"></a>
<h3>Unit test suite</h3>
<p>The starter includes 9 unit tests covering:</p>
<ul>
<li>Both axis2.xml templates exist on the classpath</li>
<li>JSON template correctness (enableJSONOnly=true, JSONBasedDefaultDispatcher, no SOAP receivers)</li>
<li>SOAP template correctness (enableJSONOnly=false, RawXMLINOut receivers, full SOAP dispatcher stack, no JSON dispatcher)</li>
<li>Default property values</li>
<li>AutoConfiguration.imports contains all required classes</li>
</ul>
</section><section><a id="Container_matrix"></a>
<h3>Container matrix</h3>
<p>See <a href="#deployment_model">Section 0</a> for the full container
compatibility matrix covering WAR deployment (Tomcat 11, WildFly 32/39)
and embedded Tomcat for local development.</p>
<a id="http2"></a>
</section></section><section><a id="a10._HTTP.2F2_and_Streaming"></a>
<h2>10. HTTP/2 and Streaming</h2>
<p>Axis2/Java has <strong>built-in HTTP/2 support at the serialization
layer</strong>, not just the transport layer. Unlike most frameworks where
the servlet container handles HTTP/2 transparently and the serialization
is unaware of the protocol, Axis2 integrates HTTP/2 awareness into the
JSON message formatter pipeline:</p>
<ul>
<li><strong>Dedicated HTTP/2 transport module</strong>
(<code>modules/transport-h2</code>) &#x2014; ALPN negotiation, per-stream
flow control, adaptive buffering, compression optimization, and
HTTP/1.1 fallback. See
<a href="http2-integration-guide.html">HTTP/2 Integration Guide</a>.</li>
<li><strong>Streaming JSON formatters</strong> &#x2014; flush every 64 KB
during serialization, converting a single buffered response into a
stream of HTTP/2 DATA frames. A 50MB response arrives at the client
as ~780 incremental frames, with the first frame arriving within
milliseconds. See
<a href="json-streaming-formatter.html">Streaming JSON Formatter</a>.</li>
<li><strong>Field selection</strong> &#x2014; callers add
<code>?fields=status,result</code> to any service URL and the
response includes only those fields, filtered during serialization
with zero overhead when not used. See
<a href="json-streaming-formatter.html#Field_Selection">Field Selection</a>.</li>
<li><strong>C parity</strong> &#x2014; the same streaming architecture exists
in <a href="https://axis.apache.org/axis2/c/core/" class="externalLink">Axis2/C</a> via
Apache httpd <code>mod_h2</code>. Both
implementations use the same 64 KB flush interval, producing
identical HTTP/2 DATA frame patterns.</li>
</ul>
<p>For a detailed comparison of Axis2's HTTP/2 architecture with
Spring MVC, JAX-RS, gRPC, and other frameworks, see the
<a href="http2-integration-guide.html#How_Axis2_HTTP2_Differs_From_Other_Frameworks">framework comparison</a>
in the HTTP/2 Integration Guide.</p>
</section>
</html> </main>
</div>
</div>
<hr/>
<footer>
<div class="container-fluid">
<div class="row-fluid">
<p>© 2004–2026
<a href="https://www.apache.org/">The Apache Software Foundation</a>
</p>
</div>
</div>
</footer>
</body>
</html>