This project is build 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 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
Unit tests and local jetty server uses a H2-database with no installation of software required. Devel/stage/prod environment uses a PostGreSQL server that must be installed and have the database tables created. create_ds_storage.ddl
./deploy2Devel11.sh
The Swagger UI is available at http://localhost:9072/ds-storage/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.
During development the configurations are located in the conf
-folder as conf/ds-storage-behaviour.yaml
,
conf/ds-storage-environment.yaml
and conf/ds-storage-local.yaml
.
When the project is started using Jetty, these are the configurations that are used.
Access to the configuration is through the static class at src/main/java/dk.kb.storage/config/ServiceConfig.java
.
Note: The environment configuration typically contains sensitive information. Do not put it in open code
repositories. To guard against this, conf/ds-storage-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 9072.
If it is started in debug mode from an IDE (normally IntelliJ IDEA), breakpoints and all the usual debug functionality
will be available.
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 is done by
- Creating and adjusting production specific copies of
conf/<application-ID>.yaml
andconf/ocp/logback.xml
in the designated folder on the server. The folder will probably be$HOME/services/conf
,$HOME/services/application-id/
or$HOME/conf/
. - Adjusting the the paths to
docBase
,<application-ID>.yaml
andlogback.xml
inconf/ocp/<application-ID>.xml
and symlinking (ln -s
) the fileconf/ocp/<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
- Changelog
- Configuration samples
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:9072//.
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/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 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 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 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 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.
openapi_v2.yaml
(preferably by copyingopenapi_v1.yaml
and adjusting) and place it alongsideopenapi_v1.yaml
- Edit the old
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:9072/ds-storage/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-storage-template/compare/v1.0.0...HEAD)
[1.1.0](https://github.com/kb-dk/ds-storage-template/compare/v1.0.0...v1.1.0)
[1.0.0](https://github.com/kb-dk/ds-storage-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-storage-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-storage-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-storage-template/commits?until=refs%2Ftags%2Fv1.0.0)
Note that both the ARK
-part and the repo-id ds-storage-template
is project-specific.
- Review that the
version
inpom.xml
is fitting.ds-storage
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-storage-1.4.2 ; git push --delete origin ds-storage-1.4.2
- Run