This project is built from the java webapp template from the Royal Danish Library.
The information in this document is aimed at developers that are not proficient in the java webapp template, Jetty, Tomcat deployment or OpenAPI.
After a fresh checkout or after the ds-image-openapi_v1.yaml
specification has changed, the api
and the model
files
must be (re)generated. This is done by calling
mvn package
Jetty is enabled, so testing the webservice can be done by running Start a Jetty web server with the application:
mvn jetty:run
The default port is 9077 and the default Hello World service can be accessed at http://localhost:9077/ds-image/v1/hello where "ds-image" is your artifactID from above.
The Swagger UI is available at http://localhost:9077/ds-image/api/, providing access to the v1
version of the GUI.
Configuration of the project is handled with YAML. It is split into multiple parts:
behaviour
which contains setup for thread pools, limits for arguments etc. This is controlled by the developers.environment
which contains server names, userIDs, passwords etc. This is controlled by operations.local
which contains temporary developer overrides. This is controlled by the individual developer.
Extra configuration files can be added when needed. They are merged in alphanumeric order and later configuration elements overrides previous ones.
Access to the merged configuration is through the static class at src/main/java/dk.kb.image/config/ServiceConfig.java
.
When developing an application, there will typically be 4 files in the conf
-folder:
conf/ds-image-behaviour.yaml
: Application centric config, such as limits for input parameters, rules for transformations and thumbnail sizes
Shared by everyone, controlled by developers, part of the repo.conf/ds-image-environment.yaml.sample
: Sample/template for environment config, such as server names, passwords and thread count. Should contain key-value pairs for all possible environment config elements, but with non-sensitive dummy-values. The sample YAML will not be merged with the other configs.
Shared by everyone, controlled by developers, used as template by operations, part of the repo.conf/ds-image-local.yaml
: Concrete environment config for the individual developer, mirroring
conf/ds-image-environment.yaml.sample
but with locally working values. When needed for experimental purposes, values from the behaviour config can also be overridden here.
Not shared, controlled by the individual developer, not part of the repo.conf/ds-image-logback.xml
: File-oriented Logback config, intended for deploy in Tomcat. Note thatmvn jetty:run
usessrc/test/jetty/logback.xml
.
Shared by everyone, controlled by developers, might be adjusted by operations, part of the repo.
This setup it used when mvn jetty:run
is called. For deployment on a devel-, stage- or production-server,
see the Tomcat section further down.
Note: The environment configuration typically contains sensitive information. Do not put it in open code
repositories! To guard against this, conf/ds-image-environment.yaml
is added to .gitignore
.
Jetty is a servlet container (like Tomcat) that is often used for testing during development.
This project can be started with mvn jetty:run
, which will expose a webserver with the implemented service at port 9077.
If it is started in debug mode from an IDE (normally IntelliJ IDEA), breakpoints and all the usual debug functionality
will be available.
Running under Jetty will result in the configs behaviour and local being used.
Tomcat is the default servlet container for the Royal Danish Library and as deployment to Tomcat must be tested before delivering the project to Operations. As of 2021, Tomcat 9 is used for Java 11 applications.
A WAR-file is generated with mvn package
and can be deployed
directly into Tomcat, although this will log to catalina.out
and use the developer configuration YAML.
Deployment on a shared server or a developer machine:
- Copy
conf/<application-ID>-behaviour.yaml
to the config folder on the server, which is typically$HOME/services/conf
,$HOME/services/<application-ID>/
or$HOME/conf/
. - Copy
conf/<application-ID>-environment.yaml.sample
to the config folder on the server, rename the file to<application-ID>-environment.yaml
and adjust the values to fit the concrete environment. - Copy the generated WAR to the designated folder on the server, probably
$HOME/services/tomcat-apps/
. - Adjust the paths to
docBase
(the path to the recently copied WAR),<application-ID>.yaml
and<application-ID>-logback.xml
in a copy ofconf/ocp/<application-ID>.xml
and copy it to the designated folder on the server, probably$HOME/services/tomcat-apps/
. - Symlink (
ln -s
) the file<application-ID>.xml
to the Tomcat folderconf/Catalina/localhost/
Deployment on a production server at the Royal Danish Library is normally done by Operations and is normally quite
similar to the procedure for test- and developer-machines. The delivery aimed at Operations is the tar-ball generated
as part of the maven build. The tar-ball is the artifact with the suffix -distribution.tar.gz
found in the target folder
and also uploaded to nexus as part of the release procedure. The tar-ball contains:
- The WAR-file
- Readme
- Changelog
<application-ID>-behaviour.yaml
(should be used as-is)<application-ID>.environment.yaml.sample
(sample environment config, to be renamed and adjusted by Operations)<application-ID>-logback.xml
(might be adjusted by Operations)<application-ID>.xml.sample
(sample tomcat context, to be renamed and adjusted by Operations)
Unit tests are run using the surefire plugin (configured in the parent pom).
If you have unit tests that takes long to run, and don't want them to run when at every invocation of mvn package,
annotate the testcase with @Tag("slow")
in the java code.
To run all unit tests including the ones tagged as slow, enable the allTests
maven profile: e.g. mvn clean package -PallTests
.
For smaller projects or standalone web applications, it can be useful to bundle the user interface with the API
implementation: Files and folders added to the src/main/webapp/
folder are served under
http://localhost:9077//.
While it is possible to use JSP, as the sample index.jsp shows, this is considered legacy technology. With an API first approach, the web application will typically be static files and JavaScript.
OpenAPI 1.3 generates interfaces and skeleton code for webservices. It also generates online documentation, which includes sample calls and easy testing of the endpoints.
Everything is defined centrally in the file src/main/openapi/ds-image-openapi_v1.yaml.
IntelliJ IDEA has a plugin for editing OpenAPI files that provides a semi-live preview of the generated GUI and
the online Swagger Editor can be used by copy-pasting the content of ds-image-openapi_v1.yaml
.
The interfaces and models generated from the OpenAPI definition are stored in target/generated-sources/
.
They are recreated on each mvn package
.
Skeleton classes are added to /src/main/java/${project.package}/api/v1/impl/
but only if they are not already present.
A reference to the classes must be present in /src/main/java/${project.package}/webservice/Application
or its equivalent.
A common pattern during initial definition of the ds-image-openapi_v1.yaml
is to delay the implementation phase and recreate
the skeleton implementation files on each build. This can be done by setting generateOperationBody
in the pom.xml
to true
.
Tip: If the ds-image-openapi_v1.yaml
is changed a lot during later development of the application, it might be better to have
<generateOperationBody>true</generateOperationBody>
in pom.xml
and add the implementation code to manually created
classed (initially copied from the OpenAPI-generated skeleton impl classes). When changes to ds-image-openapi_v1.yaml
results in
changed skeleton implementation classes, the changes can be manually ported to the real implementation classes.
Note: The classes in /src/main/java/${project.package}api/impl/
will be instantiated for each REST-call.
Persistence between calls must be handled as statics or outside of the classes.
When an API end point shall return anything else than the default response (HTTP response code 200), this is done by throwing an exception.
See how we map exceptions to responsecodes in ServiceExceptionMapper
See ServiceException and its specializations for samples.
The templates in src/main/templates/ overrides the default Swagger templates. They are needed in order to provide functionality needed by the Royal Danish Library, e.g. delivering a streaming response while announcing a well-defined structure in the Swagger UI.
Normally you do not need to touch the Mustache-files.
The project comes with a single version: v1
. Current best practise is "try and avoid new versions", meaning that
additions to APIs in production (both new methods and in responses) should generally not trigger a new version.
When a project hase been deployed in production and new functionality requires breaking the API contract, developers should
- Create a new OpenAPI YAML, e.g.
ds-image-openapi_v2.yaml
(preferably by copyingds-image-openapi_v1.yaml
and adjusting) and place it alongsideds-image-openapi_v1.yaml
- Edit the old
ds-image-openapi_v1.yaml
and add a note that a new version is available
- Edit the old
- Locate the
openapi-generator-maven-plugin
section inpom.xml
and add setup for the new version (copy-paste the twoexecution
blocks for Version 1 and adjust accordingly) - Create a new
Application
version alongsidewebservice/Application_v1.java
(copy the old one and adjust the*ServiceImpl
import) - Add a new
servlet
and a newservlet-mapping
for the newApplication
insrc/main/webapp/WEB-INF/web.xml
- Add the path to the new OpenAPI YAML to
urls
inwebapp/api/index.html
. The first entry in the array is the default when visiting the main API-page at http://localhost:9077/ds-image/api/
After a mvn package
, a skeleton implementation for the new version of the API class will be created in the source
tree. The standard action is to copy the implementation for the previous version to the new one and adjust from there.
The changelog follows Keep a Changelog guidelines. The trickiest part is to get the diff-links correct at the bottom of CHANGELOG.md as the git tags used for references should be created after the CHANGELOG has been updated.
For GitHub-projects, the syntax is
[Unreleased](https://github.com/kb-dk/ds-image-template/compare/v1.0.0...HEAD)
[1.1.0](https://github.com/kb-dk/ds-image-template/compare/v1.0.0...v1.1.0)
[1.0.0](https://github.com/kb-dk/ds-image-template/releases/tag/v1.0.0)
For BitBucket (KB's internal git), the syntax is
[Unreleased](https://sbprojects.statsbiblioteket.dk/stash/projects/ARK/repos/ds-image-template/compare/commits?targetBranch=refs%2Ftags%2Fv1.1.0&sourceBranch=refs%2Fheads%2Fmaster)
[1.1.0](https://sbprojects.statsbiblioteket.dk/stash/projects/ARK/repos/ds-image-template/compare/commits?targetBranch=refs%2Ftags%2Fv1.0.0&sourceBranch=refs%2Ftags%2Fv1.1.0)
[1.0.0](https://sbprojects.statsbiblioteket.dk/stash/projects/ARK/repos/ds-image-template/commits?until=refs%2Ftags%2Fv1.0.0)
Note that both the ARK
-part and the repo-id ds-image-template
is project-specific.
- Review that the
version
inpom.xml
is fitting.ds-image
uses Semantic Versioning: The typical release will bump theMINOR
version and setPATCH
to 0. Keep the-SNAPSHOT
-part as the Maven release plugin handles that detail. - Ensure that CHANGELOG.md is up to date.
git log
is your friend. Ensure that the about-to-be-released version is noted in the changelog entry - Ensure all local changes are committed and pushed.
- Ensure that your local
.m2/settings.xml
has a currentsbforge-nexus
-setup (contact Kim Christensen @kb or another Maven-wrangler for help) - Follow the instructions on
Guide to using the release plugin
which boils down to
- Run
mvn clean release:prepare
- Check that everything went well, then run
mvn clean release:perform
- Run
git push
If anything goes wrong during release, rollback and delete tags using something likemvn release:rollback ; git tag -d ds-image-1.4.2 ; git push --delete origin ds-image-1.4.2
- Run