Template for writing semantic.works services in JavaScript using Express 4
Requires: a semantic.works stack, like mu-project.
Create a new folder for your microservice.
In the folder, create your microservice in app.js
:
import { app } from 'mu';
app.get('/hello', function( req, res ) {
res.send('Hello mu-javascript-template');
} );
This service will respond with 'Hello mu-javascript-template' when receiving a GET request on '/hello'.
Add the mu-javascript-template to your docker-compose.yml
with the sources mounted directly.
version: '3.4'
services:
your-microservice-name:
image: semtech/mu-javascript-template
environment:
NODE_ENV: "development"
ports:
- 8888:80
volumes:
- /absolute/path/to/your/sources/:/app/
Next, create the service by running
docker-compose up -d your-microservice-name
A curl
call to the microservice will show you to message
curl http://localhost:8888/hello
# Hello mu-javascript-template
Requires:
- a semantic.works stack, like mu-project
- 'Develop your first microservice'
When developing inside an existing mu.semte.ch stack, it is easiest to set the development mode by setting the NODE_ENV
environment variable to development
and mount the sources directly. This makes it easy to setup links to the database and the dispatcher. Livereload is enabled automatically when running in development mode.
version: ...
services:
...
your-microservice-name:
image: semtech/mu-javascript-template
environment:
NODE_ENV: "development"
volumes:
- /absolute/path/to/your/sources/:/app/
Requires:
- a semantic.works stack, like mu-project
- 'Develop your first microservice'
Add a Dockerfile with the following contents:
FROM semtech/mu-javascript-template
LABEL maintainer="madnificent@gmail.com"
There are various ways to build a Docker image. For a production service we advise to setup automatic builds, but here we will build it locally. You can choose any name, but we will call ours 'say-hello-service'.
From the root of your microservice folder execute the following command:
docker build -t say-hello-service .
Add the newly built service to your application stack in docker-compose.yml
version: ...
services:
...
say-hello:
image: say-hello-service
Launch the new container in your app
docker-compose up -d say-hello
Requires: 'Develop in a mu.semte.ch stack'.
When running in development mode, you can attach the Chrome debugger to your microservice and add breakpoints as you're used to. The chrome debugger requires port 9229 to be forwarded, and your service to run in development mode. After launching your service, open Google Chrome or Chromium, and visit chrome://inspect/. Once the service is launched, a remote target on localhost should pop up.
Update your service definition in docker-compose.yml
as follows:
version: ...
services:
your-microservice-name:
...
ports:
- 9229:9229
Next, recreate the container by executing
docker-compose up -d your-microservice-name
Requires: 'Build a microservice based on mu-javascript-template' or 'Develop in a mu.semte.ch stack'
If you doubt your requests are arriving at your microservice correctly, you can publish it port to access it directly. In the example below, port 8888 is used to access the service directly.
Note this means you will not have the headers set by the identifier and dispatcher.
Update your service definition in docker-compose.yml
as follows:
your-microservice-name:
...
ports:
- 8888:80
Next, recreate the container by executing
docker-compose up -d your-microservice-name
You can install additional dependencies by including a package.json
file next to your app.js
. It works as you would expect: just define the packages in the dependencies
section of the package.json
. They will be installed automatically at build time and in development mode. There is no need to restart the container.
If you are building a reactive service that should execute certain logic based on changes in the database, you want to hook it up to the delta-notifier. Some extra steps need to be taken to properly handle delta's, specifically the route handling delta's will need to use a specific bodyParser.
The default bodyParser provided by the template will only accept application/vnd.api+json
and the delta-notifier is sending application/json
content. Aside from that the body of a delta message may be very large, often several megabytes. By specifying the bodyParser on the route accepting delta messages you can easily modify it when required.
An example
// app.js
import bodyParser from 'body-parser';
// ...
app.post("/delta-updates", bodyParser.json({ limit: '50mb' }), function(req, res) {
//...
}
The mu-javascript-template is built on ExpressJS. Check Express' Getting Started guide to learn how to build a REST API in Express.
The Express application can be imported from the 'mu'
package as follows:
import { app } from 'mu'
Routes can be defined on the application as explained in the Express routing guide. For example:
import { app } from 'mu'
app.get('/hello', function( req, res ) {
res.send('Hello mu-javascript-template');
} );
The template offers some helpers. They can all be imported from the 'mu'
package like
import { app, uuid, sparqlEscapeString } from 'mu'
app.get('/', function( req, res ) {
const id = uuid();
...
} );
You can also import the whole mu
object like
import mu from 'mu';
mu.app.get('/', function( req, res ) {
const id = mu.uuid();
...
} );
The following helper functions are provided by the template
query(query) => Promise
: Function for sending queries to the triplestoreupdate(query) => Promise
: Function for sending updates to the triplestoreuuid() => string
: Generates a random UUID (e.g. to construct new resource URIs)
The following SPARQL escape helpers are provided to construct safe SPARQL query strings
sparqlEscapeString(value) => string
sparqlEscapeUri(value) => string
sparqlEscapeDecimal(value) => string
sparqlEscapeInt(value) => string
sparqlEscapeFloat(value) => string
sparqlEscapeDate(value) => string
sparqlEscapeDateTime(value) => string
sparqlEscapeBool(value) => string
: The given value is evaluated to a boolean value in javascript. E.g. the string value'0'
evaluates tofalse
in javascript.sparqlEscape(value, type) => string
: Function to escape a value in SPARQL according to the given type. Type must be one of'string'
,'uri'
,'int'
,'float'
,'date'
,'dateTime'
,'bool'
.
The template offers an error handler to send error responses in a JSON:API compliant way. The handler can be imported from 'mu'
and need to be loaded at the end.
import { app, errorHandler } from 'mu'
app.get('/hello', function( req, res, next ) {
try {
...
} catch (e) {
next(new Error('Oops, something went wrong.))
}
});
app.use(errorHandler)
The template has second-class support for transpiling TypeScript and CoffeeScript. These are considered second-class and support may be removed in a minor release but not in a patch release.
Overwriting files through the config folder may require you to stick to the original format. There are currently no guarantees on this.
Any file extending to .coffee will be transpiled from coffeescript to javascript file. Sourcemaps are included for debugging.
Any file extending in .ts will be transpiled to a javascript file. Sources are currently not typechecked though this is subject to change. Sourcemaps are included for debugging.
The following environment variables can be configured:
NODE_ENV
(default:production
): either"development"
or"production"
. The environment to start the application in. The application live reloads on changes in"development"
mode.MAX_BODY_SIZE
(default:100kb
): max size of the request body. See ExpressJS documentation.HOST
(default:0.0.0.0
): The hostname you want the service to bind to.PORT
(default:80
): The port you want the service to bind to.
You may let users extend the microservice with code.
When you import content from ./config/some-file
, the sources can be provided by the end user in /config/some-file
(even in production mode).
You may provide default values for each of these files. The sources provided by the app are merged with the sources provided by the microservice, with the app's configuration taking precedence.
The verbosity of logging can be configured through following environment variables:
LOG_SPARQL_ALL
: Logging of all executed SPARQL queries, read as well as update (defaulttrue
)LOG_SPARQL_QUERIES
: Logging of executed SPARQL read queries (default:undefined
). OverrulesLOG_SPARQL_ALL
.LOG_SPARQL_UPDATES
: Logging of executed SPARQL update queries (defaultundefined
). OverrulesLOG_SPARQL_ALL
.DEBUG_AUTH_HEADERS
: Debugging of mu-authorization access-control related headers (defaulttrue
)
Following values are considered true: ["true"
, "TRUE"
, "1"
].
To execute custom bash statements during the image build (e.g. to install aditional system libraries), provide an on-build.sh
script in the root of your service. It will be automatically picked up and executed by the Docker build.