1. Developer Environment Setup

1.1. Prerequisites

The following is required for DINA development:

  • Linux operating system

    • On Windows using Virtual Machine

    • On Windows using WSL2 (Not Recommended - Experimental)

  • Visual Studio Code (Install on Linux or see Remote setup below)

  • Docker (Install on Linux)

    sudo apt install curl
    curl -fsSL https://get.docker.com -o get-docker.sh
    sh get-docker.sh
    sudo usermod -aG docker ${USER}
  • Docker Compose (Install on Linux)

    DOCKER_CONFIG=${DOCKER_CONFIG:-$HOME/.docker}
    mkdir -p $DOCKER_CONFIG/cli-plugins
    curl -SL https://github.com/docker/compose/releases/download/v2.4.1/docker-compose-linux-x86_64 -o $DOCKER_CONFIG/cli-plugins/docker-compose
    chmod +x $DOCKER_CONFIG/cli-plugins/docker-compose
  • Node.js

Install Node Version Manager (NVM): https://github.com/nvm-sh/nvm#installing-and-updating

Install node 18 (codename hydrogen)

nvm install Hydrogen
  • Yarn

    corepack enable
    yarn set version stable
    yarn install
  • Maven

    sudo apt install maven
  • Git

    sudo apt install git

1.2. Clone Repositories

To get started, you can git clone all of the projects to your home directory:

git clone --branch dev https://github.com/AAFC-BICoE/dina-ui.git
git clone --branch dev https://github.com/AAFC-BICoE/object-store-api.git
git clone --branch dev https://github.com/AAFC-BICoE/seqdb-api.git
git clone --branch dev https://github.com/AAFC-BICoE/agent-api.git
git clone --branch dev https://github.com/AAFC-BICoE/dina-user-api.git
git clone --branch dev https://github.com/AAFC-BICoE/natural-history-collection-api.git
git clone --branch dev https://github.com/AAFC-BICoE/loan-transaction-api.git
git clone --branch dev https://github.com/AAFC-BICoE/dina-search-api.git

You can also clone the OpenAPI specification projects:

git clone https://github.com/DINA-Web/loan-transaction-specs.git
git clone https://github.com/DINA-Web/object-store-specs.git
git clone https://github.com/DINA-Web/collection-specs.git
git clone https://github.com/DINA-Web/user-specs.git
git clone https://github.com/DINA-Web/agent-specs.git
git clone https://github.com/DINA-Web/sequence-specs.git

1.3. Visual Studio Code

Visual Studio Code can be installed directly on your Virtual Machine or on your Windows machine using the VSCode Remote instructions below.

Once you have VSCode downloaded, you can run this command to install the recommended extensions:

code --install-extension vscjava.vscode-java-pack
code --install-extension msjsdiag.debugger-for-chrome
code --install-extension ms-azuretools.vscode-docker
code --install-extension gabrielbb.vscode-lombok
code --install-extension esbenp.prettier-vscode
code --install-extension dbaeumer.vscode-eslint
code --install-extension firsttris.vscode-jest-runner
code --install-extension alefragnani.project-manager

1.3.2. VSCode Remote (Optional)

You can avoid running a slow IDE inside your virtual machine by running VS Code on your host machine instead and connecting to the dev environment using VS Code Remote.

  1. Download VSCode for Windows

  2. Install the VSCode SSH Remote Extension

    code --install-extension ms-vscode-remote.remote-ssh
  3. Install Open-SSH Server on your VM

    sudo apt-get install openssh-server
    sudo systemctl enable ssh
    sudo systemctl start ssh
  4. Setup Port Forwarding on your VM

    In order for your Windows VSCode to communicate with your Virtual Machine, you will need to allow for SSH port forwarding.

    From the VirtualBox Manager window, click Settings` > Network Tab > Advanced > Port Forwarding > Green + icon and add the following rule:

    Table 1. SSH Port Forwarding Rule
    Name Protocol Host IP Host Port Guest IP Guest Port

    SSH

    TCP

    Leave Blank

    22

    Leave Blank

    22

    Click OK on both windows to finish setting up the port.

  5. Configure SSH Remote Extension

    Once you have the SSH Remote extension, using the command palette (CTRL + SHIFT + P) search for >Remote-SSH: Add New SSH Host…​.

    In the popup at the top of the screen you can enter the SSH command to connect to your virtual machine (Replacing USERNAME with your Ubuntu username):

    ssh USERNAME@localhost

Now you are connected to your Virtual Machine. You can open projects and use the terminal as if you were in the VM.

1.3.3. API Debugging

To debug an API while using the local deployment, you can use the docker-compose.debug.yml config which can be enabled from the start_stop_dina.sh script.

Once enabled, you will be able to attach your VSCode to an API. The debugging ports can be found in the .env file.

Also remember that if you are running VSCode remotely, you will need to port forward the debugging port.

Here is an example of a launch.json that can be added to an API for VSCode to attach itself to the java debugger for a specific API:

{
  "version": "0.2.0",
  "configurations": [
    {
      "type": "java",
      "name": "Attach to Collection API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5002"
    }
  ]
}

Just ensure that port lines up with the correct API and the port is exposed for VSCode to attach itself.

1.3.4. VSCode Launch Configurations

Here is the complete launch configuration for the DINA API projects using the default ports. This can be edited by going to the "Run and debug" tab in VSCode and clicking the gear icon to edit the launch.json file.

{
  "version": "0.2.0",
  "configurations": [
    {
      "type": "java",
      "name": "Attach to Agent API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5001"
    },
    {
      "type": "java",
      "name": "Attach to Collection API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5002"
    },
    {
      "type": "java",
      "name": "Attach to Loan Transaction API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5003"
    },
    {
      "type": "java",
      "name": "Attach to User API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5004"
    },
    {
      "type": "java",
      "name": "Attach to Object Store API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5005"
    },
    {
      "type": "java",
      "name": "Attach to SeqDB API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5006"
    },
    {
      "type": "java",
      "name": "Attach to Export API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5007"
    },
    {
      "type": "java",
      "name": "Attach to Search CLI API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5008"
    },
    {
      "type": "java",
      "name": "Attach to Search WS API Container",
      "request": "attach",
      "hostName": "localhost",
      "port": "5009"
    }
  ]
}

2. Instance Configuration

By setting environment variables for the DINA UI container, you can configure certain features and options for the runtime instance.

These settings can be configured in the .env file.

2.1. Instance Mode

The INSTANCE_MODE environment variable specifies the operational mode of the runtime instance.

For example, setting PROD configures the instance for production environments. If it’s set to a instance mode other than PROD, it will be displayed in the application’s header to inform users which mode they are currently using.

If using the Dev User mode (without Keycloak), this will need to be set to developer.

By default, developer is used, which is used to indicate the server is running in a local development environment.

2.2. Instance Name

The INSTANCE_NAME environment variable is currently used to generate the feedback URL within the UI.

For instance, setting GRDI indicates that this is the GRDI instance of DINA. The feedback link will display a badge on new issues to identify the instance (GRDI) from which the user is reporting the issue.

By default, the instance name is set to AAFC.

2.3. Supported Languages

The SUPPORTED_LANGUAGES environment variable is a comma-separated list of the ISO language codes that the application will support.

For example:

SUPPORTED_LANGUAGES_ISO=en,fr

Will allow the application to switch between English and French languages.

Supported languages:

ISO Code Language

en

English

fr

French

de

German

2.4. Supported Geoinformational Systems

The SUPPORTED_GEOINFORMATION_SYSTEMS environment variable is a comma-separated list of the geoinformational systems supported by the UI.

By default, OSM for OpenStreetMap is used.

Supported systems:

Code Name

OSM

OpenStreetMap

TGN

The Getty Thesaurus of Geographic Names

When using TGN, a reverse proxy will need to be setup since it’s using HTTP protocol. The following can be added to the traefik-dynamic.yml file to configure this reverse proxy:

http:
  routers:
    tgn:
      tls: true
      rule: "Host(`localhost`, `dina.local`) && PathPrefix(`/TGNService.asmx`)"
      middlewares:
        - addHost
      service: tgnLb
      priority: 10000
  middlewares:
    addHost:
        headers:
          customRequestHeaders:
            Host: vocabsservices.getty.edu
  services:
    tgnLb:
      loadBalancer:
        servers:
          - url: "http://vocabsservices.getty.edu"

The TGN_SEARCH_BASE_URL environment variable is used to configure the base URL for the reverse proxy. If this is not provided, it will just use the localhost (dina.local).

3. Using Local Images

Running local deployment without making any changes will run the docker-hub deployed API modules. For development purposes, you can follow these steps to test your API changes.

For the examples below, we are using Collection-API but the steps are the same for all the other API modules. Check out the dina-ui Development section for live reloading support for UI development.

  1. Clean and build your API changes

    Navigate to the root of the API project you wish to test locally, then run maven clean install. The -DskipTests argument can be used to skip the tests.

    cd ~/collection-api
    mvn clean install -DskipTests
  2. Build the docker image

    Each of the API modules, contain a Dockerfile which can be used to build an image. When running the command to build the image you can provide a tag which will be used in step 3.

    docker build -t collection-api:dev .
  3. Change the image name on the docker-compose.local.yml

    In the docker-compose.local.yml file you can find the API container and change the image to use the tag created in step 2.

      collection-api:
        image: collection-api:dev
        environment:
          keycloak.enabled: "true"
  4. Start/Restart the containers

    Re-run the command you used to start the container originally (with any settings you used like messaging or profiles), you can use the -d collection-api to just start the specific container.

    docker-compose -f docker-compose.base.yml -f docker-compose.local.yml up -d collection-api

You are now running your local changes. It’s important to understand that you are running the jar generated from step 1, so if you make more changes on the API, you will need to repeat the steps to test your latest changes.

3.1. Export Docker image to file

Exporting an image from the local registry:

docker save -o collection-api1.tar collection-api:dev

Loading a Docker image from a file:

docker load < collection-api1.tar

4. DINA-UI Development

To enable dina-ui hot-reloading in dina-local-deployment, you need to configure the dina-ui repo location. This variable is stored in the .env as DINA_UI_REPO_DIRECTORY. By default it looks in the home directory for dina-ui.

Any changes to the pointed DINA_UI_REPO_DIRECTORY will cause the page to automatically reload.

To use, simply add the override docker-compose configuration for dev: -f docker-compose.dev.yml.

docker-compose \
-f docker-compose.base.yml \
-f docker-compose.local.yml \
-f docker-compose.dev.yml up

You can also still use profiles or combine this with RabbitMQ messaging as well.

5. API Endpoint Testing

5.1. Applications

While any API client can be used, Bruno is our recommended API client and requires no 3rd party accounts.

Bruno can be downloaded from: https://www.usebruno.com/downloads

Once downloaded, you can open the DINA collection of API endpoints by going to File > Open collection > Navigate to dina-local-deployment/api-client/DINA-API and open that folder.

You will also need to select the environment variables for the instance you want to connect to, for local dina development you would select DINA Local. The dropdown at the top right of the screen lets you change the environment variables.

For local development, you will need to open the Bruno preferences (CTRL + ,), Under the General tab, you will see an option for "Use custom CA certificate", make sure this is checked on. Click the "Select file" option and navigate to /etc/ssl/certs/ca-certificates.crt. (Be sure that you have ran the Local Certificate steps in order for the dina local certificates to be inside of the ca-certificates.crt file)

Please note that bruno is still under active development.

5.2. Authentication

If using the Bruno API client with our collection loaded, the Authentication settings will be automatically loaded in.

Click the gear icon located at the top right of the screen. Then go to the Auth where you can click the "Get Access Token". Currently you need to do this each time to generate the token but probably in the future this will happen automatically when the token has expired. Each request will need to use the inherit authentication option.

In other API clients, you can setup the authentication using these settings:

Table 2. Authentication Settings
Setting Value

Authentication Type

OAuth 2.0

Grant Type

Authorization Code

Authorization URL / Auth URL

https://dina.local/auth/realms/dina/protocol/openid-connect/auth

Access Token URL

https://dina.local/auth/realms/dina/protocol/openid-connect/token

Client ID

dina-public

Redirect URL / Callback URL

https://dina.local

Scope

openid

Then when sending the request, you should see a popup which will let you login to the DINA application.

5.3. Logout of Session

Sometimes you might need to login as a different user for testing, the instructions for clearing your sessions are below:

To logout of your session on the Bruno application, go to the collection settings by using the gear icon at the top right of the screen, then click Auth to access the authentication settings. At the bottom of the page there is a button to clear the session.

5.4. Headers

Most of the API POST requests require a specific "Content-Type" header. The following headers are setup to be inherited to each request inside of the DINA so they do not need to be setup for each request.

Table 3. Common Headers
Key Value Description

Content-Type

application/vnd.api+json

This is only required for POST requests, by default Postman and Insomnia might put application/json which will cause an error.

Crnk-Compact

true

Optional. Self and related links can make up to 60% of the response payload size and not always are those links of use. In this case the computation of those links is omitted. Further relationships without data are completely omitted.

5.5. Additional Environments

Only included in the /api-client repo is the local development environment file. Other environment files can be downloaded from the DINA Wiki under the "Bruno Environments" section.

6. PostgreSQL

It is possible to query PostgreSQL when the container is running.

Example on the collection database, collection pg schema:

docker exec -i CONTAINER_ID psql -U pguser collection -c "select * from collection.project"

7. Messaging

To enable Messaging based on RabbitMQ, combine the --profile search_api with the override file message-producing-override/docker-compose.override.messageProducer.yml.

docker-compose \
--profile search_api \
-f docker-compose.base.yml \
-f docker-compose.local.yml \
-f message-producing-override/docker-compose.override.messageProducer.yml \
up -d

8. Keycloak

8.1. Dev User

This option will allow the APIs to start without Keycloak but still function with a dev user.

The keycloak/docker-compose.enable-dev-user.yml config needs to be enabled inside of the start_stop_dina.sh:

DINA_CONFIGS+=('keycloak/docker-compose.enable-dev-user.yml')

For the UI to work without keycloak, you need to be using UI dev mode, which can be enabled by uncommenting the following line inside start_stop_dina.sh:

DINA_CONFIGS+=('docker-compose.dev.yml')

Inside of the docker-compose.enable-dev-user.yml file it can be configured which containers the settings are applied, by default it is applied to all.

The dina-ui container expects a comma-separated list of the group and roles.

For example:

/aafc/user, /bicoe/read-only

Then each of the API’s that you wish to use also need to be configured with the same group roles:

"groupRole": {
  "aafc": ["user"],
  "bicoe": ["read-only"]
}

8.2. Settings

Traefik is responsible for TLS termination so Keycloak is configured with KC_PROXY: edge.

Keycloak Admin console is available at https://keycloak.dina.local/auth.

The main reason for having the admin console on a different hostname is to simplify the rules for not exposing it.

9. Local certificates

In order to generate development certificates mkcert will be used.

9.1. Installation (Ubuntu 20)

sudo apt-get install wget libnss3-tools -y
wget https://github.com/FiloSottile/mkcert/releases/download/v1.4.3/mkcert-v1.4.3-linux-amd64
sudo chmod +x mkcert-v1.4.3-linux-amd64
sudo mv mkcert-v1.4.3-linux-amd64 /usr/bin/mkcert

Test mkcert is installed correctly:

mkcert --version

Generate certificates for dina.local:

mkcert -cert-file certs/dina-local-cert.pem -key-file certs/dina-local-key.pem "dina.local" "api.dina.local" "keycloak.dina.local"

Then install the certificates so it’s trusted by Chrome and Firefox:

mkcert --install

10. Minikube

It is possible to use minikube to deploy DINA locally using the helm chart.

10.1. Installing minikube

curl -LO https://storage.googleapis.com/minikube/releases/latest/minikube-linux-amd64
sudo install minikube-linux-amd64 /usr/local/bin/minikube

10.2. Starting minikube

minikube start --addons=ingress --cpus=4 --cni=flannel --install-addons=true --kubernetes-version=stable --memory=6g

Change the cpus and memory according to your local resources.

Note: This is a sample command. The important flags here are:

 --addons=ingress
 --cni=flannel
 --install-addons=true

Note: Useful alias for running kubectl commands: alias k="minikube kubectl --"

10.3. Push local image

To push local (images that are built locally) to minikube :

From the host: minikube cache add myimage:mytag

See minikube Pushing images documentation for more information.

11. Helm

11.1. Installing Helm

curl -fsSL -o get_helm.sh https://raw.githubusercontent.com/helm/helm/main/scripts/get-helm-3
chmod 700 get_helm.sh
./get_helm.sh

12. JMeter

Apache JMeter is used to execute performance/stress test plan.

12.1. Installation

You will need to have java installed on your system. Insure it’s installed using:

$ java -version

Next, download the latest version of JMeter from the JMeter website and extract it to your desired location:

$ wget https://dlcdn.apache.org//jmeter/binaries/apache-jmeter-5.5.zip

$ unzip apache-jmeter-5.5.zip

Then you can open up the binary and run jmeter to view the GUI:

$ cd /apache-jmeter-5.5/bin/

$ ./jmeter

12.2. How to run

You can start JMeter in GUI mode (default) and open the test plan to visualize it.

To run a test plan from command-line:

$ jmeter -n -t collection_api_module_testplan.jmx -l log.jtl

12.3. Module Structure

The JMeter test plan is constructed as follows

  • Module Test Plan

    • setUp Thread Group

    • Endpoint Thread Groups

    • tearDown Thread Group

12.3.1. setUp Thread Group

The setUp Thread Group will be executed prior to any regular thread groups, allowing us to setup access privileges as well as create any records that may be required by regular thread groups.

Refer to setUp Thread Group component reference

12.3.2. Endpoint Thread Group(s)

Each endpoint within the corresponding module will contain a corresponding endpoint thread group. A thread group will be responsible for setting up and testing the respective endpoint’s CRUD operations.

The layout of each thread group is as follows:

  • Endpoint User Defined Variables

    • All constant variables will be declared in this configuration element

  • HTTP Headers

    • Sets the access token and content-type headers, required to perform API requests

  • Module Token Controller

    • A Module Controller utilized to refresh the access token.

      • Module Controllers allow you to perform the actions of other controllers such as using the Token Controller from the setUp Thread Group or the CRUD controllers from other endpoints.

      • All controllers aside from the Module Controllers are Simple Controllers that offer no unique functionality, being used primarily for organizational purposes and to allow Module Controllers to replicate their functionality if needed.

  • Create/Update/Delete (CRUD) Endpoint Controllers

    • These controllers contain all Controller, Samplers, and Assertions associated with their respective CRUD operation.

      • Retrieval is tested within each controller when assertions are being made.

12.3.3. tearDown Thread Group

The tearDown Thread Group is executed after all of the other thread groups have terminated. This allows us to remove the records created within the setUp Thread Group without causing any conflicts with the thread group that user these records.

Refer to tearDown Thread Group component reference

12.4. CRUD Controllers

The CRUD Controllers are the thread group Create, Update, and Delete Controllers.

12.4.1. Create

This controller contains all JMeter elements required for testing record creation for the respective endpoint.

  • Endpoint setUp

    • Sets up random variables for the thread group to ensure unique values between threads.

    • Creates module records if the endpoint requires them for testing relationships fields.

  • Basic/Verbose

    • Creates a Basic record with the minimum fields and a Verbose record with all fields populated

      • Asserts that records have been created and values match with the variables used in the POST request.

  • w/ Empty Attributes

    • Attempts to create a record with no attributes specified.

      • Asserts that the correct error code is returned.

  • w/ Only User Group

    • Attempts to create a record with only the user group specified, missing possible required attributes

      • Asserts that the correct error code is returned.

      • If the user group is the only required attribute, this test is omitted as it is synonymous with the 'w/ Empty Attributes' test.

  • w/ Incorrect Type

    • Attempts to create a record with the type field set to not match the endpoint.

      • Asserts that the correct error code is returned.

12.4.2. Update

This controller contains all JMeter elements required for testing updating of records for the respective endpoint.

  • Basic/Verbose

    • Update the created Basic and Verbose records

      • Asserts that the values from the retrieved record match with the variables used in the PATCH request.

  • Verbose w/ Empty Body

    • Update Verbose entity with no request body.

      • Asserts that no values have been updated as a result of the request.

  • w/ Invalid UUID

    • Attempts to update a record with a UUID that does not correspond to an existing record.

      • Asserts that the correct error code is returned.

12.4.3. Delete

This controller contains all JMeter elements required for testing removal of records for the respective endpoint.

  • Basic/Verbose

    • Removes the records that have been created.

      • Asserts that they have been removed. Audit records may remain.

  • w/ Invalid UUID

    • Attempts to remove a record with a UUID that does not correspond to an existing record.

      • Asserts that the correct error code is returned.

  • Endpoint tearDown

    • Removes any module records if they were created in the 'Endpoint setUp' controller

12.4.4. Assertions

In all of the CRUD controllers, assertions are being made to ensure that the correct results are produced by each operation.

  • Assertions are managed by using an HTTP Request Sampler to retrieve the designated record and verifying the correctness of the fields.

    • The majority of assertions are conducted through a Groovy script by means of a JSR223 Assertion.

    • Assertions for Map attributes utilize the JSON Assertion instead for simpler comparison.

  • In addition to the assertions used to validate data, Response Assertions are used after every HTTP Request Sampler to ensure that the correct response code is also returned.

    • In order to for the Invalid CRUD Controller to return a valid result, a JSR223 Assertion with a Groovy script is used in place of the Response Assertion to verify the correct response code has been returned and set the successful attribute of the HTTP Request Sampler to be true.

13. Change default Docker address pool

The default Docker network addess pool may overlap with other networks (e.g. VPN).

Add to (or create) the file /etc/docker/daemon.json

{
  "bip": "192.168.32.1/24",
  "default-address-pools" : [
      {
          "base" : "192.168.33.0/24",
          "size" : 24
      }
   ]
}

bip: Bridge IP (bip) must not overlap with the default-address-pools.