Overview
Helidon provides a MicroProfile server implementation (io.helidon.microprofile.server.Server) that encapsulates the Helidon WebServer.
Maven-Coordinates
To enable MicroProfile Server add the helidon-microprofile-core bundle dependency to your project’s pom.xml
(see Managing Dependencies).
<dependency>
<groupId>io.helidon.microprofile.bundles</groupId>
<artifactId>helidon-microprofile-core</artifactId>
</dependency>
MicroProfile Server is already included in the bundle.
If full control over the dependencies is required, and you want to minimize the quantity of the dependencies -
Helidon MicroProfile Server should be used. In this case the following dependencies should be included in your
project’s pom.xml:
<dependency>
<groupId>io.helidon.microprofile.server</groupId>
<artifactId>helidon-microprofile-server</artifactId>
</dependency>
Usage
Helidon Microprofile Server is used to collect and deploy JAX-RS application(s).
When starting Helidon MP, it is recommended to use the io.helidon.Main main class, which will take care of starting Helidon. CDI will then discover all extensions, including the Server extension and start it.
See the Helidon MP Quickstart example. Note that the server lifecycle is bound to CDI.
Usage of the io.helidon.microprofile.server.Server API is discouraged, as Helidon MP uses convention to discover and configure features, which makes the applications easier to understand and maintain.
API
The following table provides a brief description of routing annotations, including its parameters. More information
in Configuring a WebServer route section.
| Annotation | Description |
|---|---|
@RoutingName(
value = ""
required = false
)
|
Binds a JAX-RS Application or Helidon Service to a specific (named) routing on |
@RoutingPath("/path")
|
Path of a Helidon Service to register with routing. |
Configuration
By default, the server uses the MicroProfile Config, but you may also want to use Helidon configuration.
In this example, the configuration is in a file, and it includes Helidon configuration options.
Configuration reference:
This is a standalone configuration type, prefix from configuration root: server
Configuration options
| key | type | default value | description |
|---|---|---|---|
|
int |
|
Accept backlog. |
|
io.helidon.common.concurrency.limits.Limit (service provider interface) Such as: |
|
Concurrency limit to use to limit concurrent execution of incoming requests. The default is to have unlimited concurrency. Note that if maxConcurrentRequests() is configured, this is ignored. |
|
|
Deprecated Configuration of a connection (established from client against our server). @deprecated use connectionOptions() instead |
|
|
|
Options for connections accepted by this listener. This is not used to setup server connection. |
|
|
|
Configure the listener specific io.helidon.http.encoding.ContentEncodingContext. This method discards all previously registered ContentEncodingContext. If no content encoding context is registered, content encoding context of the webserver would be used. |
|
|
boolean |
|
Enable proxy protocol support for this socket. This protocol is supported by
some load balancers/reverse proxies as a means to convey client information that
would otherwise be lost. If enabled, the proxy protocol header must be present
on every new connection established with your server. For more information,
see
the specification. Default is |
|
|
Configuration for this listener’s error handling. |
|
|
io.helidon.webserver.spi.ServerFeature[] (service provider interface) Such as: |
|
Server features allow customization of the server, listeners, or routings. |
|
string |
|
Host of the default socket. Defaults to all host addresses ( |
|
Duration |
|
How often should we check for idleConnectionTimeout().
Defaults to |
|
Duration |
|
How long should we wait before closing a connection that has no traffic on it.
Defaults to |
|
boolean |
|
If set to |
|
int |
|
Limits the number of requests that can be executed at the same time (the number of active virtual threads of requests).
Defaults to Setting this option will always ignore concurrencyLimit() and will use the io.helidon.common.concurrency.limits.FixedLimit. |
|
int |
|
If the entity is expected to be smaller that this number of bytes, it would be buffered in memory to optimize performance when writing it. If bigger, streaming will be used. Note that for some entity types we cannot use streaming, as they are already fully in memory (String, byte[]), for such cases, this option is ignored. Default is 128Kb. |
|
long |
|
Maximal number of bytes an entity may have.
If io.helidon.http.HeaderNames.CONTENT_LENGTH is used, this is checked immediately,
if io.helidon.http.HeaderValues.TRANSFER_ENCODING_CHUNKED is used, we will fail when the
number of bytes read would exceed the max payload size.
Defaults to unlimited ( |
|
int |
|
Limits the number of connections that can be opened at a single point in time.
Defaults to |
|
|
Configure the listener specific io.helidon.http.media.MediaContext. This method discards all previously registered MediaContext. If no media context is registered, media context of the webserver would be used. |
|
|
string |
|
Name of this socket. Defaults to |
|
int |
|
Port of the default socket.
If configured to |
|
io.helidon.webserver.spi.ProtocolConfig[] (service provider interface) Such as: |
|
Configuration of protocols. This may be either protocol selectors, or protocol upgraders from HTTP/1.1. As the order is not important (providers are ordered by weight by default), we can use a configuration as an object, such as: protocols:
providers:
http_1_1:
max-prologue-length: 8192
http_2:
max-frame-size: 4096
websocket:
....
|
|
int |
|
Deprecated Listener receive buffer size. @deprecated use SocketOptions.socketReceiveBufferSize() instead via connectionOptions(). |
|
|
Requested URI discovery context. |
|
|
boolean |
|
Copy and restore response headers before and after passing a request to Jersey for processing. If Jersey fails to handle the request, and the Webserver continues processing the request, it needs to make sure the original headers are restored. Turn off this flag to avoid the extra overhead of copying headers when no handler executes after Jersey returns. |
|
Duration |
|
Grace period in ISO 8601 duration format to allow running tasks to complete before listener’s shutdown.
Default is |
|
boolean |
|
When true the webserver registers a shutdown hook with the JVM Runtime. Defaults to true. Set this to false such that a shutdown hook is not registered. |
|
boolean |
|
If enabled and writeQueueLength() is greater than 1, then start with async writes but possibly switch to sync writes if async queue size is always below a certain threshold. |
|
|
Socket configurations. Note that socket named WebServer.DEFAULT_SOCKET_NAME cannot be used, configure the values on the server directly. |
|
|
|
Listener TLS configuration. |
|
|
int |
|
Initial buffer size in bytes of java.io.BufferedOutputStream created internally to
write data to a socket connection. Default is |
|
int |
|
Number of buffers queued for write operations. |
Examples
Access Log
Access logging in Helidon is done by a dedicated module that can be added to Maven and configured.
To enable Access logging add the following dependency to project’s pom.xml:
<dependency>
<groupId>io.helidon.microprofile</groupId>
<artifactId>helidon-microprofile-access-log</artifactId>
</dependency>
Configuring Access Log in a configuration file
Access log can be configured as follows:
server.port=8080
server.host=0.0.0.0
server.features.access-log.format=helidon
AccessLogFeature (webserver.accesslog) Configuration
access-log
This type provides the following service implementations:
-
io.helidon.webserver.spi.ServerFeatureProvider
Configuration options
| key | type | default value | description | ||||||||||||||||||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
boolean |
|
Whether this feature will be enabled. |
||||||||||||||||||||||||||||||
|
string |
|
The format for log entries (similar to the Apache
|
||||||||||||||||||||||||||||||
|
string |
|
Name of the logger used to obtain access log logger from System.getLogger(String). Defaults to AccessLogFeature.DEFAULT_LOGGER_NAME. |
||||||||||||||||||||||||||||||
|
string[] |
|
List of sockets to register this feature on. If empty, it would get registered on all sockets. The logger used will have the expected logger with a suffix of the socket name. |
||||||||||||||||||||||||||||||
|
double |
|
Weight of the access log feature. We need to log access for anything happening on the server, so weight is high: io.helidon.webserver.accesslog.AccessLogFeature.WEIGHT. |
Configuring TLS
Helidon MP also supports custom TLS configuration.
You can set the following properties:
-
Server truststore
-
Keystore with trusted certificates
-
-
Private key and certificate
-
Server certificate which will be used in TLS handshake
-
#Truststore setup
server.tls.trust.keystore.resource.resource-path=server.p12
server.tls.trust.keystore.passphrase=password
server.tls.trust.keystore.trust-store=true
#Keystore with private key and server certificate
server.tls.private-key.keystore.resource.resource-path=server.p12
server.tls.private-key.keystore.passphrase=password
Or the same configuration done in application.yaml file.
server:
tls:
#Truststore setup
trust:
keystore:
passphrase: "password"
trust-store: true
resource:
# load from classpath
resource-path: "keystore.p12" # (1)
#Keystore with private key and server certificate
private-key:
keystore:
passphrase: "password"
resource:
# load from file system
path: "/path/to/keystore.p12" # (2)
-
File loaded from the classpath.
-
File loaded from the file system.
Configuring additional ports
Helidon MP can expose multiple ports, with the following limitations:
-
The default port is the port that serves your application (JAX-RS applications and resources)
-
Other ports (in this example we configure one "admin" port) can be assigned endpoints that are exposed by Helidon components, currently supported by MP Health and MP Metrics
You can set the configuration in either application.yaml or META-INF/microprofile-config.properties:
-
The port
7011is the default port and will serve your application -
The port
8011is named "admin" (this is an arbitrary name) -
Observability endpoints, such as metrics and health, use the "admin" port through the
features.observe.socketssetting.
application.yamlserver:
port: 7011
host: "localhost"
sockets:
admin:
port: 8011
bind-address: "localhost"
features:
observe:
sockets: "admin"
META-INF/microprofile-config.propertiesserver.port=7011
server.host=localhost
server.sockets.0.name=admin
server.sockets.0.port=8011
server.sockets.0.bind-address=localhost
server.features.observe.sockets=admin
Configuring A WebServer Route
Helidon MP Server will pick up CDI beans that implement the io.helidon.webserver.HttpService
interface and configure them with the underlying WebServer.
This allows configuration of WebServer routes to run alongside a JAX-RS application.
The bean is expected to be either ApplicationScoped or Dependent and will be requested
only once during the boot of the Server.
The bean will support injection of ApplicationScoped and Dependent scoped beans.
You cannot inject RequestScoped beans. Please use WebServer features to handle request
related objects.
Customizing the HTTP service
The service can be customized using annotations and/or configuration to be
-
registered on a specific path
-
registered with a named routing
Assigning an HTTP service to named ports
Helidon has the concept of named routing. These correspond to the named ports configured with WebServer.
You can assign an HTTP service to a named routing (and as a result to a named port) using either an annotation or configuration (or both to override the value from annotation).
Annotation @RoutingName
You can annotate a service bean with this annotation to assign it to a specific named routing, that is (most likely) going to be bound to a specific port.
The annotation has two attributes:
- value that defines the routing name
- required to mark that the routing name MUST be configured in Helidon server
@RoutingName example@ApplicationScoped
@RoutingName(value = "admin", required = true)
@RoutingPath("/admin")
public class AdminService implements HttpService {
@Override
public void routing(HttpRules rules) {
// ...
}
}
The example above will be bound to admin routing (and port) and will fail if such a port
is not configured.
Configuration override of routing name
For each service bean you can define the routing name and its required flag by specifying a configuration
option bean-class-name.routing-name.name and bean-class-name.routing-name.required.
For service beans produced with producer method replace bean-class-name with class-name.producer-method-name.
Example (YAML) configuration for a service bean io.helidon.examples.AdminService that changes the
routing name to management and its required flag to false:
io.helidon.examples.AdminService:
routing-name:
name: "management"
required: false
Configuring an HTTP service path
Each service is registered on a path. If none is configured, then the service would be configured on the root path.
You can configure service path using an annotation or configuration (or both to override value from annotation)
Annotation @RoutingPath
You can configure @RoutingPath to define the path a service is registered on.
Configuration override of routing path
For each HTTP service class you can define the routing path by specifying a configuration
option class-name.routing-path.path. The routing-path configuration can be applied to Jax-RS application.
See Jakarta REST Application for more information.
Example (YAML) configuration for a class io.helidon.example.AdminService that changes the
routing path to /management:
io.helidon.examples.AdminService:
routing-path:
path: "/management"
Serving Static Content
# Location of content on file system
server.features.static-content.path.0.location=/var/www/html
# default is index.html (only in Helidon MicroProfile)
server.features.static-content.path.0.welcome=resource.html
# static content context on webserver - default is "/"
# server.features.static-content.path.0.context=/static-file
# src/main/resources/WEB in your source tree
server.features.static-content.classpath.0.location=/WEB
# default is index.html
server.features.static-content.classpath.0.welcome=resource.html
# static content path - default is "/"
# server.features.static-content.classpath.0.context=/static-cp
It is usually easier to configure list-based options using application.yaml instead, such as:
server:
features:
static-content:
welcome: "welcome.html"
classpath:
- context: "/static"
location: "/WEB"
path:
- context: "/static-file"
location: "./static-content"
See Static Content Feature Configuration Reference for details. The only difference is that we set welcome file to index.html by default.
Re-direct root using server.base-path
To redirect requests for the root path (/) to another path you can use the server.base-path property:
server:
base-path: /static/index.html
For any HTTP request for / this will return a 301 with the Location: header set to the value of server.base-path.
This is often used with Static Content Support to serve a specific index.html when / is requested.
Note that this feature is not for setting a context root for applications. To configure alternate context roots see see Setting Application Path.
Example configuration of routing
A full configuration example (YAML):
server:
port: 8080
sockets:
management:
port: 8090
io.helidon.examples.AdminApplication:
routing-name:
name: "management"
required: true
routing-path:
path: "/management"
Using Requested URI Discovery
Proxies and reverse proxies between an HTTP client and your Helidon application mask important information (for example Host header, originating IP address, protocol) about the request the client sent.
Fortunately, many of these intermediary network nodes set or update either the standard HTTP Forwarded header or the non-standard X-Forwarded-* family of headers to preserve information about the original client request.
Helidon’s requested URI discovery feature allows your application—and Helidon itself—to reconstruct information about the original request using the Forwarded header and the X-Forwarded-* family of headers.
When you prepare the connections in your server you can include the following optional requested URI discovery settings:
-
enabled or disabled
-
which type or types of requested URI discovery to use:
-
FORWARDED- uses theForwardedheader -
X_FORWARDED- uses theX-Forwarded-*headers -
HOST- uses theHostheader
-
-
what intermediate nodes to trust
When your application
receives a request
Helidon iterates through the discovery types you set up for the receiving connection, gathering information from the corresponding header(s) for that type.
If the request does not have the corresponding header(s), or your settings do not trust the intermediate nodes reflected in those headers, then Helidon tries the next discovery type you set up.
Helidon uses the HOST discovery type if you do not set up discovery yourself or if, for a particular request, it cannot assemble the request information using any discovery type you did set up for the socket.
Setting Up Requested URI Discovery
You can use configuration to set up the requested URI discovery behavior.
server.port=8080
server.requested-uri-discovery.types=FORWARDED,X_FORWARDED
server.requested-uri-discovery.trusted-proxies.allow.pattern=lb.*\\.mycorp\\.com
server.requested-uri-discovery.trusted-proxies.deny.exact=lbtest.mycorp.com
This example might apply if mycorp.com had trusted load balancers named lbxxx.mycorp.com except for an untrusted test load balancer lbtest.mycorp.com.
Obtaining the Requested URI Information
Helidon makes the requested URI information available as a property in the request context:
public class MyFilter implements ContainerRequestFilter {
@Override
public void filter(ContainerRequestContext requestContext) {
UriInfo uriInfo = (UriInfo) requestContext.getProperty("io.helidon.jaxrs.requested-uri");
// ...
}
}
See the UriInfo JavaDoc for more information.
|
Note
|
The requestContext.getUriInfo() method returns the Jakarta RESTful web services UriInfo object, not the Helidon-provided requested URI information UriInfo record.
|