Skip to content

Latest commit

 

History

History
205 lines (146 loc) · 11.2 KB

DEVELOPER.md

File metadata and controls

205 lines (146 loc) · 11.2 KB

Developer documentation

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.

Initial use

After a fresh checkout or after the ds-present-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 8080 and the default Hello World service can be accessed at http://localhost:8080/ds-present/v1/hello where "ds-present" is your artifactID from above.

The Swagger UI is available at http://localhost:8080/ds-present/api/, providing access to the v1 version of the GUI.

Debugging of XSLT's

This service relies heavily on XSLTs for transformation. They are located in src/main/resources/xslt/.

To test, navigate to src/test and use the script transform.sh. View the file HOWTO.md for details.

java webapp template structure

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-present-behaviour.yaml,
conf/ds-present-environment.yaml and conf/ds-present-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.present/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-present-environment.yaml is added to .gitignore.

Jetty

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 8080. If it is started in debug mode from an IDE (normally IntelliJ IDEA), breakpoints and all the usual debug functionality will be available.

Tomcat

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 and conf/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 and logback.xml in conf/ocp/<application-ID>.xml and symlinking (ln -s) the file conf/ocp/<application-ID>.xml to the Tomcat folder conf/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

A full web application

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:8080//.

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 (aka Swagger)

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-present-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-present-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-present-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-present-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-present-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.

OpenAPI and exceptions

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.

Mustache templates

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.

API versioning

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 copying ds-present-openapi_v1.yaml and adjusting) and place it alongside ds-present-openapi_v1.yaml
    • Edit the old ds-present-openapi_v1.yaml and add a note that a new version is available
  • Locate the openapi-generator-maven-plugin section in pom.xml and add setup for the new version (copy-paste the two execution blocks for Version 1 and adjust accordingly)
  • Create a new Application version alongside webservice/Application_v1.java (copy the old one and adjust the *ServiceImpl import)
  • Add a new servlet and a new servlet-mapping for the new Application in src/main/webapp/WEB-INF/web.xml
  • Add the path to the new OpenAPI YAML to urls in webapp/api/index.html. The first entry in the array is the default when visiting the main API-page at http://localhost:8080/ds-present/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.

Changelog

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-present-template/compare/v1.0.0...HEAD)
[1.1.0](https://github.com/kb-dk/ds-present-template/compare/v1.0.0...v1.1.0)
[1.0.0](https://github.com/kb-dk/ds-present-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-present-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-present-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-present-template/commits?until=refs%2Ftags%2Fv1.0.0)

Note that both the ARK-part and the repo-id ds-present-template is project-specific.

Release procedure

  1. Review that the version in pom.xml is fitting. ds-present uses Semantic Versioning: The typical release will bump the MINOR version and set PATCH to 0. Keep the -SNAPSHOT-part as the Maven release plugin handles that detail.
  2. 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
  3. Ensure all local changes are committed and pushed.
  4. Ensure that your local .m2/settings.xml has a current sbforge-nexus-setup (contact Kim Christensen @kb or another Maven-wrangler for help)
  5. 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 like mvn release:rollback ; git tag -d ds-present-1.4.2 ; git push --delete origin ds-present-1.4.2