Production deployment (recommendations) #
Lifecycle #
Application lifecycle #
For applications (consider application name is autotest
)
- Pack app for deployment (in the application folder)
- Copy resulting tarball to the server and run deploy
- Edit (create) environment variables file in
/var/opt/unitybase/autotest/ubConfig.env
.
cd
/opt/unitybase/apps/autotest
and runub -T
to see variables list
- If this is first time setup of application - run initialization
- If this is upgrade
Product lifecycle #
For products (consider product name is docflow
)
- Pack product for deployment (in the application folder)
- Copy resulting tarball to the server and run deploy
-
If this is a first time product setup (there is no application for this product on this server)
- create new app
- edit application environment variables file in
/var/opt/unitybase/docflow-cust1/ubConfig.env
and run application initialization
-
if this is an upgrade - all affected applications should be upgraded
Prepare application/product for deployment #
In the application/product folder run a command
This command installs all application dependencies and creates an archive for future deployment.
since ub@5.20.5 application can perform additional actions before up-pack by adding a lifecycle hook into
APP_FOLDER/lifecycle/ub-pack-before.sh
All files what must be excluded from archive should be noted in .npmignore
. Also, ignored:
- all files and folders starts with '.'
- *.log
- /logs, /stores, /localdb
- ./inetpub/clientRequire folder
In case this is a product, set a
"config": {
"ubapptype": "product"
}
in the package.json.
Application/product version is taken from "version" inside package.json and should be increased before packing to prevent deployment conflict in the future.
For testing purpose ub-pack
can be executed with -m DEV
parameter. In this case instead of npm i --production
packing script uses yarn
, so if workspaces
is defined in the package.json then some dependencies are linked.
In any case all linked dependencies are included into resulting archive as files (not as symlinks).
Deploy app/product #
Archive created by ub-pack
can be deployed using command
For applications (ubapp-*.tar.gz
) deploy script archive previous app version, unpack a new app into /opt/unitybase/apps/$UB_APP
and creates folder structure for application data in the /var/opt/unitybase/$UB_APP
(if missing).
For products (ubproduct-*.tar.gz
) deploy script archive previous product version and unpack a new product
into /opt/unitybase/products/$UB_APP
.
Folder structure #
/var
/opt
/unitybase # applications data (localdb, stores, temporary logs)
/shared # data shared between all applications
osplm.ini # UB EE+ DSTU library settings
/certificates # UB EE+ DSTU certificates storage
CACertificates.p7b # UB EE+ root CA's certificates bundle
/crb-docflow # crb.docflow data
/_temp # temp folder for mdb BLOB store
/cmodels # customer model (customer-specific addition for product developed by customer)
/localdb # local database files (SQLite3, SQL Server localdb etc.)
/stores # BLOB stores
ubConfig.env # Environment variables for application instance
/docs-adminui # docs-adminui application data
/_temp
/cmodels
/localdb
/stores
ubConfig.env
/log
/unitybase # local logs for develepnemt purpose; production logs are written to journald
/opt
/unitybase
/server # ub server (installed by rpm/deb. ub symlinked to `/usr/bin/ub`)
ub # executable
/products # products (should be installed by app developer)
/docflow
ubConfig.json
/docflow@2.1.4
ubConfig.json
/scriptum
/deals
/docflow-bpm
...
/apps # available applications (either products what configured for certain customer or stand-alone app)
/crb-docflow # product based application (in form customer-product)
/inetpub # content of this folder available using `/statics` endpoint
/models # vendor model (customer-specific addition for product developed by product owner)
/crb
crb.js
package.json
ubConfig.json # application config - for products - symlynk to a product config `../../products/docflow/ubConfig.json`
/docs-adminui # stand-alone application (not based on any product)
/inetpub
/models
/req # application specific model
/node_modules # application modules. For products this folder is placed in /products/productName
ubConfig.json # for stand-alone app not a symlynk but a file
/usr/lib/systemd/system
unitybase@.service # UnityBase vendor unit. Do not edit it directly - see overriding topic below
/ubr/bin/ub # symbolic link to /opt/unitybase/server/ub
User and group #
ub-server
installation package creates user unitybase
and group unitybase
.
This user is used to starts a unitybase@
services.
Also /opt/unitybase/apps
is owned by unitybase
user/group, so any member of unitybase
group can create a new app.
Do not log in using unitybase
user! - instead create a user for each person who can manage UnityBase and add it into unitybase
group:
Application as a service #
Product configuration #
In terms of UnityBase product
is a set of models and customizations what can be used by many customers.
The product examples is: DocFlow
, Scriptum
, df-bpm
etc.
Products installed in the /opt/unitybase/products/productname
folder as a superuser (root).
Product folder write permission allowed only for root
, other user can read it content.
Product owner should create an application config template ubConfig.json
what uses environment
variables for its variable parts. For a multi-database products we recommended wrapping a database section to #ifdef
as such:
"connections": [
//#ifdef(%UB_DB_DRIVER%=Oracle)
{ oracle config },
//#endif
//#ifdef(%UB_DB_DRIVER%=PostgreSQL)
{ Postgres config },
//#endif
]
See apps/autotest/ubConfig.json
in ubjs project git repository.
During startup service sets following variables:
Variable name | Value and explanation |
---|---|
UB_APP | Part after @ in the service name (autotest for systemctl start unitybase@autotest ) |
UB_APPDATA | /var/opt/unitybase/$UB_APP/ Note a trailing '/' - this allows to use path relative to cwd for development (when UB_APPDATA is not defined) |
UB_APPHOME | /opt/unitybase/apps/$UB_APP/ |
Additional variables can be added for application by placing it in the %UB_APPDATA%ubConfig.env
.
This file used as env
file for service.
Product template should contain placeholders for application.domain.vendorModels
and application.domain.customerModels
:
"application": {
"domain": {
"models": [....],
"vendorModels": "%UB_VMODELS%",
"customerModels": "%UB_CMODELS%"
}
}
This allows application to add a customer-specific behaviors to a product.
up-app-pack
lifecycle script detect product by check config.ubapptype
in package.json === 'product'.
Starting from UB@5.18.12 server sets a NODE_PATH variable to the real path of application config, so all modules in the
product/node_modules
folder are accessible forvendorModel
usingrequire('moduleName')
.
See application structure below for a list of folders for application data.
Tip for products developers #
During development customers specific models and test data usually placed in the same git repository as a product files:
\docflow
\models
\customer1
\crb
\crb_bpm
\stores
ubConfig.json
(config contains `"vendorModels": "%UB_VMODELS%"`)
To start an application in debug mode for a certain customer (crb) using product template and Oracle as a database run ub as:
All variables can be exported from env file using
Or via command line using -env
ub -env ./ubConfig-dev.env
We also recommend exporting an UB_ENV
variable and put it to the PROMPT_COMMAND
inside .bashrc
to
see a current environment (put a \e[0;35m\$UB_ENV\e[m
in your .bashrc PROMPT_COMMAND=). For example:
Application #
In terms of UnityBase application is either a product what configured for a certain customer, or a "stand-alone" app - an application what not based on any product and used in single environment only.
Product based application examples are:
- DocFlow for Customer#1: application based on DocFlow product with customizations for Customer#1
- Scriptum for Customer#2: application based on
Scriptum
product with customizations for Customer#2
Stand-alone application examples are:
- docs-adminui: AdminUI documentation project. Not based on a product since used on the unitybase.info website only
- autotest: an autotest application for UnityBase
Both type of applications are placed in the sub-folders of /opt/unitybase/apps
folder. Name of the sub-folder is a $UB_APP.
Main part of product-based applications (including node_modules
and config template) are placed in the /products sub-folder
and application data (vendorModels, customerModel, localdb, stores and .env file) - in the /apps sub-folder.
Config is symlinked from products into apps.
Stand-alone application can place all its part into /apps.
Application folders structure #
Application with name appName
place its code in /opt/unitybase/apps/appName
(UB_APPHOME environment variable) and
its data in /var/opt/unitybase/appName
(UB_APPDATA environment variable):
/opt/unitybase/apps
/appName
/inetpub # content of this folder available using `/statics` endpoint
favicon.ico
robots.txt
...
/models # customer models
/crb # vendor model (customer-specific addition for product developed by product owner)
package.json
ubConfig.json # application config - symlynk to a product config `../../products/docflow@2.1.4/ubConfigDocflow-tpl.json`
/var/opt/unitybase/
/appName
/localdb # local database files (SQLite3, SQL Server localdb etc.)
appnameFTS.sqlite3
..
/stores # application BLOB stores
/cmodels
/crbc # customer model (customer-specific addition for product developed by customer)
ubConfig.env # Environment variables for application instance
Product template can use additional environment variables. These variables should be defined in $UB_APPDATA/ubConfig.env
.
Start application #
A command below starts autotest
application and schedule it to start after OS reboot:
sudo systemctl enable --now unitybase@autotest
In fact this is combination of two command:
sudo systemctl enable unitybase@autotest # schedule application to start after boot
sudo systemctl start unitybase@autotest # start application now
Vendor preset for unitybase@
service is configured to restart service on failure with 1 second delay between restarts.
In case service restarts more when 5 times during 30 second it marked as "failed" and should be started
manually (using systemstl start
).
To force systemd to reset service startup failure counters use a command:
Typical operations are:
- start application
- stop application
Logs #
Setup logging #
setup file logging #
UnityBase can write logs to journald (default mode when started using systemd unitybase@.service
unit) or into files.
For big production we recommend to use a file based logging - it is faster and allow to use a build-in Log Viewer
(on UI Administration->Maintenance Tools->Log Viewer
).
To force a file logging for server started as systemd service run ub with --journald false
command line switch
or (preferred) - set an environment variable (un ubConfig.env) UB_JOURNALD=false
To allow remote log access in case ub is behind nginx drop-in for nginx must be defined - see @unitybase/log-view/README.md for details
setup journald logging #
For a systemd < 254 setup /etc/systemd/journald.conf
as in /etc/systemd/journald@unitybase.conf
For a systemd >= 254 /etc/systemd/journald@unitybase.conf
preset is used automatically.
Restart journald to apply new changes
Logging performance #
Logs are written using UDP sockets. UDP/IP receive buffer default should be increased for better performance.
Check the current UDP/IP receive buffer default and limit by typing the following commands:
If the values are less than 26214400 bytes (25MB), you should add the following lines to the /etc/sysctl.conf file:
Changes to /etc/sysctl.conf do not take effect until a reboot. To update the values immediately, type the following commands as root:
Use logs #
File, exported to LogView'er format may contain a several start/stop parts.
To split a file todaysLog.log
into sessions use a command:
Overriding a default application startup rules #
All services #
Or create a drop-in manually
mkdir /etc/systemd/system/unitybase@.service.d
and place there *.conf
file with parameters what needs to be added / override.
See systemd.unit for more information.
Certain my-app instance #
Metrics #
There is two metrics sources
/stat endpoint #
- authorised user can send a GET request to
/stat
endpoint to obtain a metrics.
The response example:
Here:
Stat metric | Value and explanation |
---|---|
sessions_total | Total count of active user session |
threads_total | Worker thread count (threadPoolSize parameter in ubConfig) |
incoming_bytes | Bytes receive over HTTP |
outcoming_bytes | Bytes send over HTTP |
responses_total | Responses count for status codes 0XX(should be 0), 1XX, 2XX, 3XX, 4XX, 5XX. In example above there is 59 2XX responses |
security_exception_count | All security related exceptions count (including session expire errors) |
js_gc_count | Number of times (for all threads) GC has been invoked. Includes both major and minor GC |
js_memory_bytes | Amount of bytes allocated by the JS engines (all threads) |
http_perning_request_count | How many input tasks are currently waiting to be affected by thread pool (for server on Linux. For Windows always 0 - see below about HTTP service request queue |
/metrics endpoint for Prometheus #
Starting from 5.18.21 UnityBase EE expose a various application metrics in Prometheus format.
Feature is configured using metrics
section of ubConfig.
/metrics
endpoint is available without authorization. Access to endpoint is limited on the reverse proxy level using
metrics.allowedFrom
parameter in ubConfig. Default is deny
for all subnet masks.
For Linux server Prometheus metrics uses histograms whenever it makes sense, so per-endpoint (for HTTP), per-thread (for JS GC) and per-entity statistics are available.
IMPORTANT For Linux with non-english locale (not recommended) LC_NUMERIC=C environment variable should be sets for UnityBase server
A truncated output example (with method profile enabled):
Windows performance counter metrics #
Under Windows HTTP level metrics are exposed as HTTP.sys specific performance counter
Most important is HTTP Service Request Queues
If queue length is too long (200 and more) either a thread pool should be increased or profile and tune application logic.
Separate schedulers and user instance #
For hi-load deployment it is recommended to run separate instances for user access and a schedulers task execution.
Examples below consider we already have app named tstks-docflow
- replace it with your app name
- creat a new unit named
unitybase-sched@.service
- in new unit add a new line after existed
EnvironmentFile
. In example below second line is added:
ensure product/application
ubConfig.json
:
- use environment variables
UB_USE_SCHEDULERS
for schedulers enabling andUB_PORT
for service port number"async": true
is defined forfts
- in main application env file
/var/opt/unitybase/tstks-docflow/ubConfig.env
disable schedulers by define
- create addition
/var/opt/unitybase/tstks-docflow/ubConfig-sched.env
file what will enable schedulers and change application port with content below
Our new unitybase-sched@.service
unit will read ubConfig-sched.env
after ubConfig.env
and override scheduler and port settings.
some production envs uses
PORT_OCT
variable instead ofUB_PORT
- in this case instead of UB_PORT=XXXX define PORT_OCT=XX and use an original service PORT_OCT-10 as a value
- start a new service
Storing secrets #
Starting from UB@5.25, the EE version supports encrypted values in ubConfig.json/ubConfig.env. For sensitive information such as database passwords or API keys, you should use secrets instead of plain text.
To generate a secret (encrypt a password value) use command
command above must be executed on the server where the service will be launched and under the same user as for service
The long string from command output can be used as a value inside ubConfig or ENV files instead of plain text myPlainPasswordVal
.
Server understand what value is encrypted and decrypt it before usage.
If the server is compromised, secrets values will not be available to the attacker in plain text. Decryption is still possible, but it will require time and specialized knowledge
The plaintext password value we encrypt should be stored in a safe place outside the server. If you reinstall the operating system, the secrets must be generated anew.