Manage Domino taxonomies — namespaces, tags, and entity tagging — via the Taxonomy API. Covers creating, listing, updating, and deleting tags and namespaces; tagging entities (project, model, dataset, app, project_template, netapp_volume); querying entities by tag; tag autocomplete; merging tags; and importing/exporting taxonomy trees as CSV. Use when organizing projects with tags, building hierarchical namespaces, finding all entities with a given tag, bulk-tagging during onboarding, or migrating taxonomy across environments.
80
—
Does it follow best practices?
Impact
—
No eval scenarios have been run
Passed
No known issues
This skill covers Domino's Taxonomy API for organizing entities (projects, models, datasets, apps, project templates, NetApp volumes) under hierarchical tags grouped into namespaces. It documents every public endpoint with curl examples that work today against a Domino cluster where the taxonomy service is enabled.
Activate this skill when the user wants to:
Auth via the local access-token endpoint per the
Skill Authoring Standards.
Never use DOMINO_USER_API_KEY.
TOKEN=$(curl -s http://localhost:8899/access-token)
# Taxonomy is accessible via its internal Kubernetes service — no external URL needed.
BASE="http://taxonomy.domino-platform:80/api/taxonomy/v1"
H="Authorization: Bearer $TOKEN"| Concept | Description |
|---|---|
| Namespace | Top-level group (e.g. Indication, Analysis). Has label, optional description, and allowMultipleAssignments flag. |
| Tag | A label inside a namespace. Can be hierarchical via parentId (e.g. Clinical_Data / SDTM). Has label, namespaceId, optional description and parentId, and status (active / inactive). |
| EntityType | Enum over taggable entities: dataset, project, project_template, model, app, netapp_volume. |
allowMultipleAssignments | When true, an entity can hold multiple tags from the same namespace. When false, applying a new tag from the namespace replaces any existing one. |
| Taxonomy tree | The full nested view: namespaces → root tags → child tags. Returned by GET /taxonomy. |
| Limits | GET /config returns maxDepth (max tag nesting) and maxLabelLength (max characters per label). |
All endpoints are under $BASE (/api/taxonomy/v1). Authenticate every
request with Authorization: Bearer $TOKEN.
| Endpoint | Method | Purpose |
|---|---|---|
/config | GET | Get configuration limits |
/taxonomy | GET | Get full taxonomy tree (nested namespaces + tags) |
/namespaces | GET / POST | List / create namespaces |
/namespaces/{namespaceId} | GET / PUT / DELETE | Get / update / delete a namespace |
/namespaces/bulk-delete | POST | Bulk delete namespaces — see BULK-OPS.md |
/tags | GET / POST | List / create tags |
/tags/{tagId} | GET / PUT / DELETE | Get / update / delete a tag |
/tags/{tagId}/entities | GET | List entities tagged with a tag |
/tags/autocomplete | GET | Autocomplete tag suggestions for a query |
/tags/bulk-delete | POST | Bulk delete tags — see BULK-OPS.md |
/rpc/merge-tags | POST | Merge tags — see BULK-OPS.md |
/entities | GET | Get entities by one or more tag IDs |
/entity-tags | GET / DELETE | Get tags for entities / delete all tags for an entity |
/rpc/tag-entity | POST | Tag an entity |
/rpc/untag-entity | POST | Remove specific tags from an entity |
/rpc/export-to-file | POST | Export taxonomy as CSV — see IMPORT-EXPORT.md |
/rpc/import-from-file | POST | Import taxonomy from CSV — see IMPORT-EXPORT.md |
/rpc/validate-file | POST | Validate a CSV before import — see IMPORT-EXPORT.md |
TOKEN=$(curl -s http://localhost:8899/access-token)
BASE="http://taxonomy.domino-platform:80/api/taxonomy/v1"
H="Authorization: Bearer $TOKEN"
# 1. Discover the tag you want to apply
curl -s -H "$H" "$BASE/tags/autocomplete?q=clinical" | python3 -m json.tool
# 2. Apply it to your project
curl -X POST -H "$H" -H "Content-Type: application/json" \
-d "{\"entityType\":\"project\",\"entityId\":\"$DOMINO_PROJECT_ID\",\"tagIds\":[\"<tag-id>\"]}" \
"$BASE/rpc/tag-entity"
# 3. Verify
curl -s -H "$H" "$BASE/entity-tags?entityType=project&entityIds=$DOMINO_PROJECT_ID"# Create a namespace
NS=$(curl -s -X POST -H "$H" -H "Content-Type: application/json" \
-d '{"label":"Analysis","description":"Type of analysis","allowMultipleAssignments":false}' \
"$BASE/namespaces" | python3 -c "import sys,json; print(json.load(sys.stdin)['id'])")
# Create a parent tag
PARENT=$(curl -s -X POST -H "$H" -H "Content-Type: application/json" \
-d "{\"label\":\"Interim\",\"namespaceId\":\"$NS\"}" \
"$BASE/tags" | python3 -c "import sys,json; print(json.load(sys.stdin)['id'])")
# Create a child tag under it
curl -s -X POST -H "$H" -H "Content-Type: application/json" \
-d "{\"label\":\"Milestone_01\",\"namespaceId\":\"$NS\",\"parentId\":\"$PARENT\"}" \
"$BASE/tags"Tag depth is capped at maxDepth from GET /config (5 on most deployments).
# Single tag
curl -s -H "$H" "$BASE/tags/<tag-id>/entities" | python3 -m json.tool
# Multiple tags (intersection) — tagIds is a REPEATED query parameter, not comma-separated
curl -s -H "$H" "$BASE/entities?tagIds=<tag-id-1>&tagIds=<tag-id-2>&entityType=project"Both endpoints return paginated results with meta.pagination.{total,limit,offset}.
Paginate with ?limit=50&offset=50.
Heads-up:
tagIdson/entitiesmust be repeated per value (?tagIds=A&tagIds=B). Comma-separating returns400 {"message":"invalid tag ID: A,B"}. By contrast,entityIdson/entity-tagsis comma-separated. Yes, the convention is inconsistent.
curl -s -H "$H" "$BASE/tags/autocomplete?q=clin"Response:
{"items":[
{"id":"23ca6d78-...","path":"Clinical_Data"},
{"id":"0b55bf67-...","path":"Clinical_Data / SDTM"},
{"id":"57454f72-...","path":"Clinical_Data / ADaM"}
]}path shows the full hierarchy with / between levels — surface it
verbatim in the UI to disambiguate same-named tags in different branches.
# List
curl -s -H "$H" "$BASE/namespaces?limit=50&offset=0"
# Create (label required)
curl -s -X POST -H "$H" -H "Content-Type: application/json" \
-d '{"label":"Indication","description":"Therapeutic area","allowMultipleAssignments":true}' \
"$BASE/namespaces"
# 201 → returns full Namespace including id, status, timestamps
# Get one
curl -s -H "$H" "$BASE/namespaces/<id>"
# Update (label and status both required)
curl -s -X PUT -H "$H" -H "Content-Type: application/json" \
-d '{"label":"Indication","description":"updated","status":"active","allowMultipleAssignments":true}' \
"$BASE/namespaces/<id>"
# Delete — cascades through the namespace's tags and entity-tag bindings.
# Returns 204 even on a non-empty namespace. There is no "are you sure" prompt.
# Reach for `namespaces/bulk-delete` when removing several at once.
curl -s -X DELETE -H "$H" "$BASE/namespaces/<id>"
# 204 No Content# List with optional filters
curl -s -H "$H" "$BASE/tags?namespaceId=<ns>&limit=50"
# Create (label and namespaceId required, parentId optional for hierarchy)
curl -s -X POST -H "$H" -H "Content-Type: application/json" \
-d '{"label":"SDTM","namespaceId":"<ns>","description":"Study Data Tabulation Model","parentId":"<parent-tag>"}' \
"$BASE/tags"
# Get one — includes entityCount broken down by EntityType
curl -s -H "$H" "$BASE/tags/<id>"
# Update (label required; you can change parentId to re-parent within the same namespace)
curl -s -X PUT -H "$H" -H "Content-Type: application/json" \
-d '{"label":"SDTMv2","description":"new desc","status":"active","parentId":"<new-parent>"}' \
"$BASE/tags/<id>"
# Delete (single)
curl -s -X DELETE -H "$H" "$BASE/tags/<id>"
# 204 No Content# Tag an entity (entityType, entityId, tagIds[] all required)
curl -s -X POST -H "$H" -H "Content-Type: application/json" \
-d '{"entityType":"project","entityId":"<project-id>","tagIds":["<tag-1>","<tag-2>"]}' \
"$BASE/rpc/tag-entity"
# 201 → {"entityId":"...","entityType":"project","entityName":"...","tags":[...]}
#
# Constraint: if `tagIds` contains more than one tag from the SAME namespace,
# that namespace must have `allowMultipleAssignments=true`. Otherwise:
# 400 → {"message":"namespace does not allow multiple tag assignments per entity"}
# To replace a tag in a single-assign namespace, send a separate request — the
# new tag overwrites the prior one for that namespace.
# Remove specific tags from an entity
curl -s -X POST -H "$H" -H "Content-Type: application/json" \
-d '{"entityType":"project","entityId":"<project-id>","tagIds":["<tag-1>"]}' \
"$BASE/rpc/untag-entity"
# 200 → {"entityId":"...","entityType":"project","entityName":"...","removedTagIds":[...]}
# Get tags for one or more entities (entityIds is comma-separated)
curl -s -H "$H" "$BASE/entity-tags?entityType=project&entityIds=<id-1>,<id-2>"
# 200 → {"data":{"<id-1>":[Tag,...], "<id-2>":[Tag,...]}}
# Remove ALL tags from an entity
curl -s -X DELETE -H "$H" "$BASE/entity-tags?entityType=project&entityId=<id>"
# 200 → {"removedCount":N}entityType must be one of: dataset, project, project_template,
model, app, netapp_volume. Other values return 400.
curl -s -H "$H" "$BASE/taxonomy" | python3 -m json.toolReturns an array of TreeNamespace objects, each with nested tags of type
TreeTag:
[{
"id":"821e1455-...",
"label":"Indication",
"description":"Therapeutic area",
"status":"active",
"allowMultipleAssignments":false,
"tags":[
{"id":"...", "label":"Oncology", "status":"active", "children":[
{"id":"...", "label":"Breast_Cancer", "status":"active", "children":[]}
]}
]
}]Use this to render the full taxonomy in a UI in one call.
curl -s -H "$H" "$BASE/config"
# {"maxDepth":5,"maxLabelLength":128}Surface these limits in any UI that lets users create tags or namespaces so the user gets immediate validation feedback.
See BULK-OPS.md for tags/bulk-delete,
namespaces/bulk-delete, and rpc/merge-tags.
See IMPORT-EXPORT.md for rpc/export-to-file,
rpc/import-from-file, and rpc/validate-file — useful for migrating a
taxonomy across Domino environments.
allowMultipleAssignments=false for mutually exclusive concepts
(e.g. "Phase" — a project is in exactly one phase). Use true when an
entity can plausibly carry several values from the same namespace
(e.g. multiple "Indication" tags).parentId rather than encoding hierarchy into labels
(e.g. Clinical_Data/SDTM). This lets autocomplete and tree views render
the structure for you.GET /config once at app startup and cache maxLabelLength /
maxDepth for client-side validation.PUT /tags/{id};
IDs are stable. Workflows that automate tagging should reference tag IDs.GET /taxonomy for full-tree views,
GET /tags/autocomplete?q=... for typeaheads, and
GET /entities?tagIds=A&tagIds=B&entityType=... (one tagIds per value,
not comma-separated) for filtered lists.Public api endpoint not found (404)The taxonomy service is not registered on the gateway you are calling. Two common causes:
taxonomy.domino-platform:80, not
$DOMINO_API_HOST (nucleus-frontend does not proxy taxonomy). Ensure BASE
is set to http://taxonomy.domino-platform:80/api/taxonomy/v1.400 Bad Request on POST /rpc/tag-entityEnsure entityType is exactly one of:
dataset | project | project_template | model | app | netapp_volume.
Casing matters — Project and PROJECT will be rejected.
400 on POST /tagsnamespaceId must reference an existing namespace, and the label must be
non-empty and ≤ maxLabelLength (from /config). If you intend to nest the
tag, parentId must reference a tag in the same namespace.
DELETE /namespaces/{id} is a cascading delete: it removes the namespace,
every tag inside it, and every entity-tag binding pointing at those tags.
There is no "namespace must be empty" check — confirm with the user before
calling it against a shared cluster. namespaces/bulk-delete has the same
cascade behavior across multiple IDs.
Before writing or verifying any API call, use the cluster swagger to confirm current endpoint paths and field names. Use public docs for workflow context and field explanations.
Taxonomy API base: http://taxonomy.domino-platform:80 (internal Kubernetes service, reachable from any workspace, job, or app).
Fetch the taxonomy swagger spec (requires bearer token):
TOKEN=$(curl -s http://localhost:8899/access-token)
# Reachable via internal Kubernetes service (works in any workspace, job, or app):
curl -H "Authorization: Bearer $TOKEN" "http://taxonomy.domino-platform:80/api/taxonomy/swagger/doc.json"
# Browser UI — use the external cluster URL (must be logged in):
# https://<your-cluster>/api/taxonomy/swagger/index.htmlPublic docs (workflow context and field explanations):
47c6e0a
If you maintain this skill, you can claim it as your own. Once claimed, you can manage eval scenarios, bundle related skills, attach documentation or rules, and ensure cross-agent compatibility.