wf-moray-backend
v1.3.2
Published
A backend for wf built over Moray
Downloads
5
Readme
node-workflow-moray-backend
This repository is part of the Joyent Triton project. See the contribution guidelines and general documentation at the main Triton project page.
A backend for node-workflow built over Moray.
Highlights regarding other wf-backends:
Searching Jobs
- Searching
jobs
byvm_uuid
is faster, given we are using indexedjob.target
attribute, which always matches'(target=*' + vm_uuid + '*)'
for machines' jobs. (Since WORKFLOW-101). - Searching
jobs
on a given time period is also supported using the argumentssince
, for jobs created since the given time anduntil
for jobs created until the specified time. Note both of these attributes values will be included into the search, which is made using LDAP filters. Attributes can take eitherepoch
time values or any valid JavaScript Date string. Note that if the given value cannot be transformed into a valid JavaScript Date, the argument will be silently discarded. (Since WORKFLOW-104).
Please, remember that JavaScript dates must be specified in milliseconds.
Therefore, if you are passing "time since epoch" values, those must be
specified using milliseconds. If you are buiding your date objects, just
call getTime()
method and you're set.
Examples:
The following is a very simple output of a system with very few jobs to help illustrating search options:
[root@headnode (coal) ~]# sdc-workflow /jobs | json -a created_at execution name
2013-05-27T15:38:27.173Z running reboot-7.0.0
2013-05-27T01:12:05.918Z succeeded provision-7.0.6
2013-05-27T01:01:39.919Z succeeded server-sysinfo-1.0.0
2013-05-27T00:56:49.635Z succeeded server-sysinfo-1.0.0
Searching by vm_uuid
will retrieve all the jobs associated with the given machine:
[root@headnode (coal) ~]# sdc-workflow /jobs?vm_uuid=767aa4fd-95b4-4c94-bc6c-71d274fb2899 | json -a created_at execution name
2013-05-27T15:38:27.173Z succeeded reboot-7.0.0
2013-05-27T01:12:05.918Z succeeded provision-7.0.6
Searching jobs "until", "within" or "since" a given time period just
requires some caution regarding the format of the passed in since
and until
arguments.
For example, to pass in the value 2013-05-27T15:38:27.173Z
for one of these
attributes we can either URL encode it, or transform into a time since epoch
integer:
$ node
> encodeURIComponent('2013-05-27T15:38:27.173Z');
'2013-05-27T15%3A38%3A27.173Z'
> new Date('2013-05-27T15:38:27.173Z').getTime();
1369669107173
Either way, we can use for jobs search:
[root@headnode (coal) ~]# sdc-workflow /jobs?since=2013-05-27T15%3A38%3A27.173Z | json -a created_at execution name2013-05-27T15:38:27.173Z succeeded reboot-7.0.0
[root@headnode (coal) ~]# sdc-workflow /jobs?since=1369669107173 | json -a created_at execution name
2013-05-27T15:38:27.173Z succeeded reboot-7.0.0
The same way we search for jobs created since a given time value, we can also search for jobs created until a given time:
[root@headnode (coal) ~]# sdc-workflow /jobs?until=2013-05-27T00%3A56%3A49.635Z | json -a created_at execution name
2013-05-27T00:56:49.635Z succeeded server-sysinfo-1.0.0
And, of course, we can combine both parameters to search jobs created within a given time period:
[root@headnode (coal) ~]# sdc-workflow /jobs?since=2013-05-27T01%3A01%3A39.919Z\&until=2013-05-27T01%3A12%3A05.918Z | json -a created_at execution name
2013-05-27T01:12:05.918Z succeeded provision-7.0.6
2013-05-27T01:01:39.919Z succeeded server-sysinfo-1.0.0
Also, note that it's perfectly possible passing in other valid JavaScript date values
to these parameters, like 2013-05-27
. As far as we can transform the given value
into a valid date, everything will work as expected. If the given value cannot be
properly transformed into a valid Date object, the value will be silently discarded.
Searching workflows
(Since WORKFLOW-81)
While it's possible to search workflows by any workflow property, the only indexed property is workflow's name (this will be improved in the future).
Therefore, it's highly recommended to search workflows just by name. Searching by name, with the current model of workflow definition also allows to search workflows by version.
Some search examples:
Exact match:
# A workflow name matches exactly the provided search filter:
[root@headnode (coal) ~]# sdc-workflow /workflows?name=provision-7.0.6| json -a name
provision-7.0.6
# There isn't a single workflow matching name exactly:
[root@headnode (coal) ~]# sdc-workflow /workflows?name=provision| json -a name
Approximate matches:
# workflow's name begins with provision:
[root@headnode (coal) ~]# sdc-workflow /workflows?name=provision*| json -a name
provision-7.0.6
# workflow's name contains provision:
[root@headnode (coal) ~]# sdc-workflow /workflows?name=*provision*| json -a name
provision-7.0.6
reprovision-7.0.0
# workflow's name finishes with '7.0.0' (actually, this matches wf version)
[root@headnode (coal) ~]# sdc-workflow /workflows?name=*7.0.0| json -a name
create-from-snapshot-7.0.0
start-7.0.0
stop-7.0.0
reboot-7.0.0
reprovision-7.0.0
update-7.0.0
destroy-7.0.0
snapshot-7.0.0
rollback-7.0.0
delete-snapshot-7.0.0
remove-nics-7.0.0
Extra fields
Starting with version 0.9.5
is possible to add custom fields to the buckets
wf_jobs
, wf_workflows
and wf_jobs_info
. While this was theoretically
possible in the past for latest two buckets, it wasn't for wf_jobs
.
Additionally, the possible fields were restricted to the moray supported
indexes.
Now it's also possible to add object
types, which will be stored into moray
as strings, and properly encoded/decoded when saved/retrieved from the backend.
This change has been introduced to automatically deal with JSON.stringified
objects and arrays.
Fields definition must be placed into configuration file, right under:
backend.opts.extra_fields
it's to say:
{
"backend": {
"module": "../lib/workflow-moray-backend",
"opts": {
"moray": {
"srvDomain": "moray.coal.joyent.us",
"cueballOptions": {
"resolvers": [ "binder.coal.joyent.us" ]
}
}
"extra_fields": {
"wf_workflows": {
"custom_object": {
"type": "object",
"index": false,
},
"unique_string": {
"type": "string",
"unique": true
}
"indexed_string": {
"type": "string",
"index": true
}
},
"wf_jobs": {
"vm_uuid": {
"type": "string",
"index": true,
"unique": false
},
"server_uuid": {
"type": "string",
"index": true,
"unique": false
}
},
"wf_jobs_info": {}
},
"version": 1
}
}, ...
}
Let's briefly review the wf_workflows
section on the config fragment above:
- The field
custom_object
will not be added to moray bucket as an index. (index
was set tofalse
). Indeed, this would be the recommended way to proceed for arbitrary length complex objects: do not index them. If you need to search for specific properties, add extra fields with those properties. Anyway, when a JavaScriptArray
orObject
value is given to this field, it'll be encoded usingJSON.stringify
before it's saved into moray and decoded usingJSON.parse
when it's retrieved from moray. - The field
unique_string
will be added to moray as anunique
index. - The field
indexed_string
will be added to moray as anon unique
index.
Couple important things to note:
If you add new fields for an existing setup, you need to change the version number. Otherwise, the moray buckets will not get updated with the new fields. Use a plain integer number for versioning.
If you want to be able to add the extra fields to either
jobs
orworkflows
using the API (and you want it), you must also add these to the api config section as follows:"api": { "port": 8080, "wf_extra_params": ["custom_object", "unique_string", "indexed_string"], "job_extra_params": ["vm_uuid", "server_uuid"] },
Installation
Add [email protected]:joyent/node-workflow-moray-backend.git
to your wf project
dependencies.
Usage
Add the following to the config file of your application using wf:
{
"backend": {
"module": "wf-moray-backend",
"opts": {
"moray": {
"srvDomain": "moray.coal.joyent.us",
"cueballOptions": {
"resolvers": [ "binder.coal.joyent.us" ]
}
},
"extra_fields": {}
}
}
}
Where url
is the url where moray server is listening.
And that should be it. wf
REST API and Runners should take care of
properly loading the module on init.
Issues
See node-workflow issues for general workflow issues, and visit SDC or Manta specific issues.
Testing
There is a script at tools/coal-test-env.sh. Just scp into GZ and executing it should work. What the script does is below for reference.
Then, to run the tests either:
make test
or for extra verbosity:
node test/moray-backend.test.js 2>&1 | bunyan
node test/buckets-definition.test.js 2>&1 | bunyan
Setup a moray_test
db from manatee zone:
[root@headnode (coal) ~]# sdc-login manatee
[root@... (coal:manatee0) ~]# createdb -U postgres moray_test
[root@... (coal:manatee0) ~]# psql -U postgres moray_test
moray_test=# CREATE TABLE buckets_config (
name text PRIMARY KEY,
index text NOT NULL,
pre text NOT NULL,
post text NOT NULL,
options text,
mtime timestamp without time zone DEFAULT now() NOT NULL,
reindex_active text
);
Boot another moray instance pointing to the moray_test
db:
[root@headnode (coal) ~] sdc-login moray
[root@... (coal:moray0) ~]# cd /opt/smartdc/moray/
[root@... (coal:moray0) /opt/smartdc/moray]# cp etc/config.json etc/config.test.json
[root@... (coal:moray0) /opt/smartdc/moray]# cp smf/manifests/moray-2021.xml smf/manifests/moray-test.xml
Then, you need to modify SMF to make this moray test instance to use our moray_test
db
and listen into a different port:
Change service name from
moray
tomoray-test
at the very top of the manifest:<service_bundle type="manifest" name="smartdc-moray-test">
Make sure the
start
methodexec
attribute has the following value:exec="node main.js -f etc/config.json -p %{moray-test/port} &"
Add
MORAY_DB_NAME
to thestart
method environment:Change the value for the
port
variable:<instance name="default" enabled="true"> <property_group name="moray-test" type="application"> <propval name="port" type="astring" value="2222" /> </property_group> </instance>
The manifest diff versus the original will look like:
[root@... (coal:moray0) /opt/smartdc/moray]# diff -u smf/manifests/moray.xml smf/manifests/moray-test.xml
--- smf/manifests/moray.xml 2013-05-26 08:17:06.323940448 +0000
+++ smf/manifests/moray-test.xml 2013-05-26 09:04:17.859967502 +0000
@@ -1,7 +1,7 @@
<?xml version="1.0"?>
<!DOCTYPE service_bundle SYSTEM "/usr/share/lib/xml/dtd/service_bundle.dtd.1">
-<service_bundle type="manifest" name="smartdc-moray">
- <service name="smartdc/application/moray" type="service" version="1">
+<service_bundle type="manifest" name="smartdc-moray-test">
+ <service name="smartdc/application/moray-test" type="service" version="1">
<dependency name="network"
grouping="require_all"
@@ -27,7 +27,7 @@
<exec_method
type="method"
name="start"
- exec="node main.js -f etc/config.json -p %{moray/port} &"
+ exec="node main.js -f etc/config.json -p %{moray-test/port} &"
timeout_seconds="30">
<method_context working_directory="/opt/smartdc/moray">
<method_credential user="nobody"
@@ -38,6 +38,7 @@
value="/opt/smartdc/moray/build/node/bin:/opt/local/bin:/usr/bin:/usr/sbin:/bin"/>
<envvar name="LD_PRELOAD_32"
value="/usr/lib/extendedFILE.so.1" />
+ <envvar name="MORAY_DB_NAME" value="moray_test" />
</method_environment>
</method_context>
</exec_method>
@@ -51,8 +52,8 @@
<property_group name="application" type="application" />
<instance name="default" enabled="true">
- <property_group name="moray" type="application">
- <propval name="port" type="astring" value="2020" />
+ <property_group name="moray-test" type="application">
+ <propval name="port" type="astring" value="2222" />
</property_group>
</instance>
Finally, import the new service:
svccfg import smf/manifests/moray-test.xml
You should have a new moray instance running at the moray zone on port 2222 now. For
COAL this means your moray URL will be http://10.99.99.17:2222
.
LICENSE
Copyright 2017 Joyent, Inc.
This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.