These instructions are for setting up CouchDB for a development environment. For a production environment, we recommend running Open MCT behind a proxy server (e.g., Nginx or Apache), and securing the CouchDB server properly: https://docs.couchdb.org/en/main/intro/security.html
The following process is the preferred way of using CouchDB as it is automatic and closely resembles a production environment.
Requirement: Get docker compose (or recent version of docker) installed on your machine. We recommend Docker Desktop
- Open a terminal to this current working directory (
cd openmct/src/plugins/persistence/couch
) - Create and start the
couchdb
container:
docker compose -f ./couchdb-compose.yaml up --detach
- Copy
.env.ci
file to file named.env.local
- (Optional) Change the values of
.env.local
if desired - Set the environment variables in bash by sourcing the env file
export $(cat .env.local | xargs)
- Execute the configuration script:
sh ./setup-couchdb.sh
cd
to the workspace root directory (the same directory asindex.html
)- Update
index.html
to use the CouchDB plugin as persistence store:
sh ./src/plugins/persistence/couch/replace-localstorage-with-couchdb-indexhtml.sh
- ✅ Done!
Open MCT will now use your local CouchDB container as its persistence store. Access the CouchDB instance manager by visiting http://localhost:5984/_utils.
To completely remove the CouchDB container and volumes:
docker stop couch-couchdb-1;docker rm couch-couchdb-1;docker volume rm couch_couchdb
We highly recommend using the CouchDB docker compose
method of installation, though it is still possible to install CouchDB through other means.
- Install CouchDB using:
brew install couchdb
. - Edit
/usr/local/etc/local.ini
and add the following settings:
[admins]
admin = youradminpassword
And set the server up for single node:
[couchdb]
single_node=true
Enable CORS
[chttpd]
enable_cors = true
[cors]
origins = http://localhost:8080
If brew
is not available on your mac machine, you'll need to get the CouchDB installed using the official sourcefiles.
- Install CouchDB following these instructions: https://docs.brew.sh/Installation#untar-anywhere.
- Edit
local.ini
in Homebrew's/etc/
directory as directed above in the 'Installing with admin privileges to your computer' section.
Follow the installation instructions from the CouchDB installation guide: https://docs.couchdb.org/en/stable/install/index.html
The simplest way to config a CouchDB instance is to use our provided tooling:
- Copy
.env.ci
file to file named.env.local
- Set the environment variables in bash by sourcing the env file
export $(cat .env.local | xargs)
- Execute the configuration script:
sh ./setup-couchdb.sh
- Start CouchDB by running:
couchdb
. - Add the
_global_changes
database usingcurl
(note theyouradminpassword
should be changed to what you set above 👆):curl -X PUT http://admin:[email protected]:5984/_global_changes
- Navigate to http://localhost:5984/_utils
- Create a database called
openmct
- Navigate to http://127.0.0.1:5984/_utils/#/database/openmct/permissions
- Remove permission restrictions in CouchDB from Open MCT by deleting
_admin
roles for bothAdmin
andMember
.
CouchDB has size limits on both its internal documents, and its httpd interface. If dealing with larger documents in Open MCT (e.g., users adding images to notebook entries), you may to increase this limit. To do this, add the following to the two sections:
[couchdb]
max_document_size = 4294967296 ; approx 4 GB
[chttpd]
max_http_request_size = 4294967296 ; approx 4 GB
If not present, add them under proper sections. The values are in bytes, and can be adjusted to whatever is appropriate for your use case.
The simplest way to config a CouchDB instance is to use our provided tooling:
cd
to the workspace root directory (the same directory asindex.html
)- Update
index.html
to use the CouchDB plugin as persistence store:
sh ./src/plugins/persistence/couch/replace-localstorage-with-couchdb-indexhtml.sh
- Edit
openmct/index.html
comment out the following line:
openmct.install(openmct.plugins.LocalStorage());
Add a line to install the CouchDB plugin for Open MCT:
openmct.install(
openmct.plugins.CouchDB({
databases: [
{
url: 'http://localhost:5984/openmct',
namespace: '',
additionalNamespaces: [],
readOnly: false,
useDesignDocuments: false,
indicator: true
}
]
})
);
When installing the CouchDB plugin for OpenMCT, you can specify a list of databases with configuration options for each. Here's a breakdown of the available options for each database:
-
url
: The URL to the CouchDB instance, specifying the protocol, hostname, and port as needed.- Example:
'http://localhost:5984/openmct'
- Example:
-
namespace
: The namespace associated with this database.- Example:
'openmct-sandbox'
- Example:
-
additionalNamespaces
: Other namespaces that this plugin should respond to requests for.- Example:
['apple-namespace', 'pear-namespace']
- Example:
-
readOnly
: A boolean indicating whether the database should be treated as read-only. If set totrue
, OpenMCT will not attempt to write to this database.- Example:
false
- Example:
-
useDesignDocuments
: Indicates whether design documents should be used to speed up annotation search.- Example:
false
- Example:
-
indicator
: A boolean to specify whether an indicator should show the status of this CouchDB connection in the OpenMCT interface.- Example:
true
- Example:
Note: If using the exampleTags
plugin with non-blank namespaces, you'll need to configure it point to a writable database. For example:
openmct.install(
openmct.plugins.example.ExampleTags({ namespaceToSaveAnnotations: 'openmct-sandbox' })
);
Note: If using the MyItems
plugin, be sure to configure a root for each writable namespace. E.g., if you have two namespaces called apple-namespace
and pear-namespace
:
openmct.install(openmct.plugins.MyItems('Apple Items', 'apple-namespace'));
openmct.install(openmct.plugins.MyItems('Pear Items', 'pear-namespace'));
This will create a root object with the id of mine
in both namespaces upon load if not already created.
- Start Open MCT by running
npm start
in theopenmct
path. - Navigate to http://localhost:8080/ and create a random object in Open MCT (e.g., a 'Clock') and save.
- Navigate to: http://127.0.0.1:5984/_utils/#database/openmct/_all_docs
- Look at the 'JSON' tab and ensure you can see the specific object you created above.
- All done! 🏆
One can delete annotations by running inside this directory (i.e., src/plugins/persistence/couch
):
npm run deleteAnnotations:openmct:PIXEL_SPATIAL
will delete all image tags.
npm run deleteAnnotations:openmct
will delete all tags.
npm run deleteAnnotations:openmct -- --help
will print help options.
For large Open MCT installations, it may be helpful to add additional CouchDB capabilities to bear to improve performance.
Indexing the model.type
field in CouchDB can benefit the performance of queries significantly, particularly if there are a large number of documents in the database. An index can accelerate annotation searches by reducing the number of documents that the database needs to examine.
To create an index for model.type
, you can use the following payload using the API:
{
"index": {
"fields": ["model.type", "model.tags"]
},
"name": "type_tags_index",
"type": "json"
}
This instructs CouchDB to create an index on the model.type
field and the model.tags
field. Once this index is created, queries that include a selector on model.type
and model.tags
(like when searching for tags) can use this index to retrieve results faster.
You can find more detailed information about indexing in CouchDB in the official documentation.
We can also add a design document through the API for retrieving domain objects for specific tags:
{
"_id": "_design/annotation_tags_index",
"views": {
"by_tags": {
"map": "function (doc) { if (doc.model && doc.model.type === 'annotation' && doc.model.tags) { doc.model.tags.forEach(function (tag) { emit(tag, doc._id); }); } }"
}
}
}
and can be retrieved by issuing a GET
to http://localhost:5984/openmct/_design/annotation_tags_index/_view/by_tags?keys=["TAG_ID_TO_SEARCH_FOR"]&include_docs=true
where TAG_ID_TO_SEARCH_FOR
is the tag UUID we're looking for.
and for targets:
{
"_id": "_design/annotation_keystring_index",
"views": {
"by_keystring": {
"map": "function (doc) { if (doc.model && doc.model.type === 'annotation' && doc.model.targets) { doc.model.targets.forEach(function(target) { if(target.keyString) { emit(target.keyString, doc._id); } }); } }"
}
}
}
and can be retrieved by issuing a GET
to http://localhost:5984/openmct/_design/annotation_keystring_index/_view/by_keystring?keys=["KEY_STRING_TO_SEARCH_FOR"]&include_docs=true
where KEY_STRING_TO_SEARCH_FOR
is the UUID we're looking for.
To enable them in Open MCT, we need to configure the plugin useDesignDocuments
like so:
openmct.install(openmct.plugins.CouchDB({url: "http://localhost:5984/openmct", useDesignDocuments: true}));