| name | elasticsearch-authz | ||||
|---|---|---|---|---|---|
| description | Manage Elasticsearch RBAC: native users, roles, role mappings, document- and field-level security. Use when creating users or roles, assigning privileges, or mapping external realms like LDAP/SAML. | ||||
| compatibility | Requires network access to Elasticsearch. Kibana endpoint needed for Kibana role API. Serverless role assignment requires the cloud-access-management skill. | ||||
| metadata |
|
Manage Elasticsearch role-based access control: native users, roles, role assignment, and role mappings for external realms.
For authentication methods and API key management, see the elasticsearch-authn skill.
For detailed API endpoints, see references/api-reference.md.
Deployment note: Feature availability differs between self-managed, ECH, and Serverless. See Deployment Compatibility for details.
- Create a native user with a specific set of privileges
- Define a custom role with least-privilege index and cluster access
- Assign one or more roles to an existing user
- Create a role with Kibana feature or space privileges
- Configure a role mapping for external realm users (SAML, LDAP, PKI)
- Derive role assignments dynamically from user attributes (Mustache templates)
- Restrict document visibility per user or department (document-level security)
- Hide sensitive fields like PII from certain roles (field-level security)
- Implement attribute-based access control (ABAC) using templated role queries
- Translate a natural-language access request into user, role, and role mapping tasks
| Item | Description |
|---|---|
| Elasticsearch URL | Cluster endpoint (e.g. https://localhost:9200 or a Cloud deployment URL) |
| Kibana URL | Required only when setting Kibana feature/space privileges |
| Authentication | Valid credentials (see the elasticsearch-authn skill) |
| Cluster privileges | manage_security is required for user and role management operations |
Prompt the user for any missing values.
When the user describes access in natural language (e.g. "create a user that has read-only access to logs-*"), break
the request into discrete tasks before executing. Follow this workflow:
Extract from the prompt:
| Component | Question to answer |
|---|---|
| Who | New native user, existing user, or external realm user (LDAP, SAML, etc.) |
| What | Which indices, data streams, or Kibana features |
| Access level | Read, write, manage, or a specific set of privileges |
| Scope | All documents/fields, or restricted by region, department, sensitivity? |
| Kibana? | Does the request mention any Kibana feature (dashboards, Discover, etc.) |
| Deployment? | Self-managed, ECH, or Serverless? Serverless has a different user model. |
Before creating a new role, check if an existing role already grants the required access:
curl "${ELASTICSEARCH_URL}/_security/role" <auth_flags>If a matching role exists, skip role creation and reuse it.
Derive a role name and display name from the request. Use the Elasticsearch API for pure index/cluster roles. Use the Kibana API if Kibana features are involved (see Choosing the right API).
| Scenario | Action |
|---|---|
| New native user | Create the user with the role and a strong generated password. (Self-managed / ECH only.) |
| Existing native user | Fetch current roles, append the new role, update the user with the full array. (Self-managed / ECH only.) |
| External realm user | Create a role mapping that matches the user's realm attributes to the role. (Self-managed / ECH only.) |
| Serverless user | Use the cloud-access-management skill. Assign a predefined role or create a custom role first, then assign it via the Cloud API. |
Prompt: "Create a user analyst with read-only access to logs-* and metrics-* and view dashboards in Kibana."
- Identify: new user
analyst, indiceslogs-*/metrics-*, dashboards, read access. - Check roles:
GET /_security/role— no match. - Create role via Kibana API (dashboards involved):
logs-metrics-dashboard-viewer. - Create user:
POST /_security/user/analystwithroles: ["logs-metrics-dashboard-viewer"].
Confirm each step with the user if the request is ambiguous.
Native user management applies to self-managed and ECH deployments. On Serverless, users are managed at the organization level — skip this section.
curl -X POST "${ELASTICSEARCH_URL}/_security/user/${USERNAME}" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"password": "'"${PASSWORD}"'",
"roles": ["'"${ROLE_NAME}"'"],
"full_name": "'"${FULL_NAME}"'",
"email": "'"${EMAIL}"'",
"enabled": true
}'Use PUT /_security/user/${USERNAME} with the fields to change. Omit password to keep the existing one.
curl -X POST "${ELASTICSEARCH_URL}/_security/user/${USERNAME}/_password" \
<auth_flags> -H "Content-Type: application/json" \
-d '{"password": "'"${NEW_PASSWORD}"'"}'
curl -X PUT "${ELASTICSEARCH_URL}/_security/user/${USERNAME}/_disable" <auth_flags>
curl -X PUT "${ELASTICSEARCH_URL}/_security/user/${USERNAME}/_enable" <auth_flags>
curl "${ELASTICSEARCH_URL}/_security/user/${USERNAME}" <auth_flags>
curl -X DELETE "${ELASTICSEARCH_URL}/_security/user/${USERNAME}" <auth_flags>Use the Elasticsearch API (PUT /_security/role/{name}) when the role only needs cluster and indices
privileges. This is the default — no Kibana endpoint is required.
Use the Kibana role API (PUT /api/security/role/{name}) when the role includes any Kibana feature or space
privileges. The Elasticsearch API cannot set Kibana feature grants, space scoping, or base privileges, so if the user
mentions Kibana features like Discover, Dashboards, Maps, Visualize, Canvas, or any other Kibana application, the Kibana
API is required.
If the Kibana endpoint is not available or API key authentication to Kibana fails, fall back to the Elasticsearch API
for the cluster and indices portion and warn the user that Kibana privileges could not be set. Prompt for a Kibana
URL or alternative credentials before giving up.
Default choice when the role has only index and cluster privileges:
curl -X PUT "${ELASTICSEARCH_URL}/_security/role/${ROLE_NAME}" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"description": "'"${ROLE_DISPLAY_NAME}"'",
"cluster": [],
"indices": [
{
"names": ["'"${INDEX_PATTERN}"'"],
"privileges": ["read", "view_index_metadata"]
}
]
}'Required when the role includes Kibana feature or space privileges:
curl -X PUT "${KIBANA_URL}/api/security/role/${ROLE_NAME}" \
<auth_flags> \
-H "kbn-xsrf: true" \
-H "Content-Type: application/json" \
-d '{
"description": "'"${ROLE_DISPLAY_NAME}"'",
"elasticsearch": {
"cluster": [],
"indices": [
{
"names": ["'"${INDEX_PATTERN}"'"],
"privileges": ["read", "view_index_metadata"]
}
]
},
"kibana": [
{
"base": [],
"feature": {
"discover": ["read"],
"dashboard": ["read"]
},
"spaces": ["*"]
}
]
}'curl "${ELASTICSEARCH_URL}/_security/role/${ROLE_NAME}" <auth_flags>
curl "${ELASTICSEARCH_URL}/_security/role" <auth_flags>
curl -X DELETE "${ELASTICSEARCH_URL}/_security/role/${ROLE_NAME}" <auth_flags>Roles can restrict access at the document and field level within an index, going beyond index-level privileges.
Restrict which fields a role can see. Use grant to whitelist or except to blacklist fields:
curl -X PUT "${ELASTICSEARCH_URL}/_security/role/pii-redacted-reader" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"description": "PII Redacted Reader",
"indices": [
{
"names": ["customers-*"],
"privileges": ["read"],
"field_security": {
"grant": ["*"],
"except": ["ssn", "credit_card", "date_of_birth"]
}
}
]
}'Users with this role see all fields except the PII fields. FLS is enforced on search, get, and aggregation results.
Restrict which documents a role can see by attaching a query filter:
curl -X PUT "${ELASTICSEARCH_URL}/_security/role/emea-logs-reader" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"description": "EMEA Logs Reader",
"indices": [
{
"names": ["logs-*"],
"privileges": ["read"],
"query": "{\"term\": {\"region\": \"emea\"}}"
}
]
}'The query field is a JSON string containing a Query DSL filter. Users with this role only see documents where region
equals emea.
DLS queries support Mustache templates that inject user metadata at query time, enabling attribute-based access control
(ABAC) on top of RBAC. Store user-specific attributes in the user's metadata field, then reference them in the role
query template with {{_user.metadata.<key>}}.
curl -X PUT "${ELASTICSEARCH_URL}/_security/role/department-reader" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"description": "Department Reader",
"indices": [
{
"names": ["records-*"],
"privileges": ["read"],
"query": "{\"template\": {\"source\": \"{\\\"term\\\": {\\\"department\\\": \\\"{{_user.metadata.department}}\\\"}}\"}}"
}
]
}'A user with "metadata": {"department": "engineering"} only sees documents where department equals engineering. The
same role works for all departments — no per-department role needed.
For multi-valued attributes (e.g. a list of required programs), use terms_set with minimum_should_match_field to
ensure the user holds all required attributes listed on the document. This enables complex ABAC policies — combining
security levels, program lists, and certification dates — using a single role. See
references/api-reference.md for full terms_set ABAC examples including combined
multi-condition policies and user metadata setup.
A single index privilege entry can include both query (DLS) and field_security (FLS). See the
HR department example for a practical combined use case.
When users hold multiple roles, DLS queries are combined with OR and FLS grants are unioned. A broad role without DLS/FLS can unintentionally widen access. When combining roles, always verify effective permissions and ensure no unrestricted role overrides DLS/FLS intent.
Self-managed and ECH only. On Serverless, use the cloud-access-management skill — see Serverless User Access.
Update the user with the new roles array:
curl -X PUT "${ELASTICSEARCH_URL}/_security/user/${USERNAME}" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"roles": ["role-a", "role-b"]
}'The roles array is replaced entirely — include all roles the user should have. Fetch the user first to see current
roles before updating.
After role or user updates, verify effective access with:
curl -X POST "${ELASTICSEARCH_URL}/_security/user/_has_privileges" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"cluster": ["monitor"],
"index": [
{
"names": ["'"${INDEX_PATTERN}"'"],
"privileges": ["read", "view_index_metadata"]
}
]
}'Role mappings are not available on Serverless (both ES API and Kibana UI are disabled). Use the cloud-access-management skill instead — see Serverless User Access.
Role mappings assign external-realm users (LDAP, AD, SAML, PKI) to roles based on attribute rules. Self-managed and ECH only. For supported rule operators and resource fields, see role mapping resource properties.
curl -X PUT "${ELASTICSEARCH_URL}/_security/role_mapping/saml-default-access" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"roles": ["viewer"],
"enabled": true,
"rules": {
"field": { "realm.name": "saml1" }
}
}'curl -X PUT "${ELASTICSEARCH_URL}/_security/role_mapping/ldap-admins" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"roles": ["superuser"],
"enabled": true,
"rules": {
"all": [
{ "field": { "realm.name": "ldap1" } },
{ "field": { "groups": "cn=admins,ou=groups,dc=example,dc=com" } }
]
}
}'Use role_templates instead of roles to derive role names from user attributes. Scripting must be enabled.
curl -X PUT "${ELASTICSEARCH_URL}/_security/role_mapping/ldap-group-roles" \
<auth_flags> \
-H "Content-Type: application/json" \
-d '{
"role_templates": [
{
"template": { "source": "{{#tojson}}groups{{/tojson}}" },
"format": "json"
}
],
"enabled": true,
"rules": {
"field": { "realm.name": "ldap1" }
}
}'See references/api-reference.md for more Mustache patterns including realm-username derived roles and tiered group access.
curl "${ELASTICSEARCH_URL}/_security/role_mapping/saml-default-access" <auth_flags>
curl "${ELASTICSEARCH_URL}/_security/role_mapping" <auth_flags>
curl -X DELETE "${ELASTICSEARCH_URL}/_security/role_mapping/saml-default-access" <auth_flags>On Serverless, there are no native users or role mappings. Users receive project access through Cloud-level role assignments.
- Predefined roles (e.g.
admin,developer,viewer) cover common access patterns. If one fits, assign it directly via the Cloud API — no custom role creation needed. - Custom roles are required when the user needs fine-grained access (specific indices, Kibana features, DLS/FLS). Create the custom role using the Elasticsearch API or Kibana API (same as self-managed — see Manage Roles), then assign it to the user alongside a predefined base role via the Cloud API.
- Run-as privileges are unavailable in Serverless custom roles.
Use the cloud-access-management skill for the full workflow (inviting users, assigning roles, managing Cloud API keys, and verifying access). This skill handles only role definition; cloud-access-management handles user assignment.
Request: "Create a user joe with read-only access to logs-*."
- Create the role via
PUT /_security/role/logs-readerwith"description": "Logs Reader"andindices: [{ names: ["logs-*"], privileges: ["read", "view_index_metadata"] }]. - Create the user via
POST /_security/user/joewith"roles": ["logs-reader"]and a strong generated password.
Request: "Let users read logs-* and view dashboards in Kibana."
Use the Kibana API (PUT <KIBANA_URL>/api/security/role/logs-dashboard-viewer) with elasticsearch.indices for data
access and kibana[].feature for dashboard and Discover read access on all spaces. See
Create or update a role (Kibana API) for the full request structure.
Request: "Give Alice access to apm-* in addition to her current roles."
GET /_security/user/alice— response shows"roles": ["viewer"].- Create
apm-readerrole withindices: [{ names: ["apm-*"], privileges: ["read", "view_index_metadata"] }]. PUT /_security/user/alicewith"roles": ["viewer", "apm-reader"](include all roles).
Request: "Give alice@example.com read-write access to the colors index and let her use dashboards and Discover."
- Create a custom role via the Kibana API:
PUT <KIBANA_URL>/api/security/role/colors-rw-kibanawithelasticsearch.indicesforread,write,view_index_metadataoncolorsandkibana[].featurefordashboard,discover. - Use the cloud-access-management skill to assign the user the custom role
colors-rw-kibana.
Request: "Each manager should only see HR records from their own department, and PII fields should be hidden."
- Create a user with department metadata:
POST /_security/user/manager_awith"metadata": {"department": "engineering"}. - Create a role with DLS + FLS:
PUT /_security/role/hr-department-viewer
{
"description": "HR Department Viewer",
"indices": [
{
"names": ["hr-*"],
"privileges": ["read"],
"field_security": { "grant": ["*"], "except": ["ssn", "salary", "date_of_birth"] },
"query": "{\"template\": {\"source\": \"{\\\"term\\\": {\\\"department\\\": \\\"{{_user.metadata.department}}\\\"}}\"}}"
}
]
}The same role works for every department — each user sees only their department's records with PII fields removed.
- Never use the
elasticsuperuser for day-to-day operations. Create dedicated minimum-privilege roles and reserveelasticfor initial setup and emergency recovery. - Use
readandview_index_metadatafor read-only data access. Leaveclusterempty unless explicitly required. - Use DLS (
query) and FLS (field_security) to restrict access within an index.
Never use internal action names (e.g. indices:data/read/search). Always use officially documented named privileges.
Prefer fine-grained privileges (manage_ingest_pipelines, monitor) over broad ones (manage, all). See
references/api-reference.md for the full privilege reference tables.
- Use short lowercase names with hyphens:
logs-reader,apm-data-viewer,metrics-writer. - Avoid generic names like
custom-roleornew-role. - Set
descriptionto a short, human-readable display name — not a long sentence. It is shown in the Kibana UI as the role's label. Good:"Logs Reader","APM Data Viewer". Bad:"Read-only access to all logs-* indices for the operations team".
- Generate strong passwords by default: at least 16 characters mixing uppercase, lowercase, digits, and symbols (e.g.
X9k#mP2vL!qR7wZn). Never use placeholder values likechangemeorpassword123. - Prefer disabling users over deleting them to preserve audit trail.
- The
rolesarray on a user is replaced entirely on update. Always fetch current roles before modifying.
- Use static
rolesfor simple, fixed assignments (e.g. all SAML users getviewer). - Use
role_templateswith Mustache only when roles must be derived dynamically from user attributes. - Combine
all,any,field, andexceptrules to express complex conditions without duplicating mappings. - Test new mappings with
enabled: falsefirst, then enable once verified.
See references/deployment-compatibility.md for a feature matrix and detailed notes on self-managed, ECH, and Serverless deployment differences.