Create or update role mappings API

Create or update role mappings API

New API reference

For the most up-to-date API details, refer to Security APIs.

Creates and updates role mappings.

Request

POST /_security/role_mapping/<name>

PUT /_security/role_mapping/<name>

Prerequisites

  • To use this API, you must have at least the manage_security cluster privilege.

Description

Role mappings define which roles are assigned to each user. Each mapping has rules that identify users and a list of roles that are granted to those users.

The role mapping APIs are generally the preferred way to manage role mappings rather than using role mapping files. The create or update role mappings API cannot update role mappings that are defined in role mapping files.

This API does not create roles. Rather, it maps users to existing roles. Roles can be created by using the create or update roles API or roles files.

For more information, see Mapping users and groups to roles.

Role templates

The most common use for role mappings is to create a mapping from a known value on the user to a fixed role name. For example, all users in the cn=admin,dc=example,dc=com LDAP group should be given the superuser role in Elasticsearch. The roles field is used for this purpose.

For more complex needs, it is possible to use Mustache templates to dynamically determine the names of the roles that should be granted to the user. The role_templates field is used for this purpose.

To use role templates successfully, the relevant scripting feature must be enabled. Otherwise, all attempts to create a role mapping with role templates fail. See Allowed script types setting.

All of the user fields that are available in the role mapping rules are also available in the role templates. Thus it is possible to assign a user to a role that reflects their username, their groups, or the name of the realm to which they authenticated.

By default a template is evaluated to produce a single string that is the name of the role which should be assigned to the user. If the format of the template is set to "json" then the template is expected to produce a JSON string or an array of JSON strings for the role names.

Path parameters

name

(string) The distinct name that identifies the role mapping. The name is used solely as an identifier to facilitate interaction via the API; it does not affect the behavior of the mapping in any way.

Request body

The following parameters can be specified in the body of a PUT or POST request and pertain to adding a role mapping:

enabled

(Required, Boolean) Mappings that have enabled set to false are ignored when role mapping is performed.

metadata

(object) Additional metadata that helps define which roles are assigned to each user. Within the metadata object, keys beginning with _ are reserved for system usage.

roles

(list of strings) A list of role names that are granted to the users that match the role mapping rules. Exactly one of roles or role_templates must be specified.

role_templates

(list of objects) A list of mustache templates that will be evaluated to determine the roles names that should granted to the users that match the role mapping rules. The format of these objects is defined below. Exactly one of roles or role_templates must be specified.

rules

(Required, object) The rules that determine which users should be matched by the mapping. A rule is a logical condition that is expressed by using a JSON DSL. See Role mapping resources.

Examples

The following example assigns the “user” role to all users:

  1. resp = client.security.put_role_mapping(
  2. name="mapping1",
  3. roles=[
  4. "user"
  5. ],
  6. enabled=True,
  7. rules={
  8. "field": {
  9. "username": "*"
  10. }
  11. },
  12. metadata={
  13. "version": 1
  14. },
  15. )
  16. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping1",
  3. roles: ["user"],
  4. enabled: true,
  5. rules: {
  6. field: {
  7. username: "*",
  8. },
  9. },
  10. metadata: {
  11. version: 1,
  12. },
  13. });
  14. console.log(response);
  1. POST /_security/role_mapping/mapping1
  2. {
  3. "roles": [ "user"],
  4. "enabled": true,
  5. "rules": {
  6. "field" : { "username" : "*" }
  7. },
  8. "metadata" : {
  9. "version" : 1
  10. }
  11. }

Mappings that have enabled set to false are ignored when role mapping is performed.

Metadata is optional.

A successful call returns a JSON structure that shows whether the mapping has been created or updated.

  1. {
  2. "role_mapping" : {
  3. "created" : true
  4. }
  5. }

When an existing mapping is updated, created is set to false.

The following example assigns the “user” and “admin” roles to specific users:

  1. resp = client.security.put_role_mapping(
  2. name="mapping2",
  3. roles=[
  4. "user",
  5. "admin"
  6. ],
  7. enabled=True,
  8. rules={
  9. "field": {
  10. "username": [
  11. "esadmin01",
  12. "esadmin02"
  13. ]
  14. }
  15. },
  16. )
  17. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping2",
  3. roles: ["user", "admin"],
  4. enabled: true,
  5. rules: {
  6. field: {
  7. username: ["esadmin01", "esadmin02"],
  8. },
  9. },
  10. });
  11. console.log(response);
  1. POST /_security/role_mapping/mapping2
  2. {
  3. "roles": [ "user", "admin" ],
  4. "enabled": true,
  5. "rules": {
  6. "field" : { "username" : [ "esadmin01", "esadmin02" ] }
  7. }
  8. }

The following example matches users who authenticated against a specific realm:

  1. resp = client.security.put_role_mapping(
  2. name="mapping3",
  3. roles=[
  4. "ldap-user"
  5. ],
  6. enabled=True,
  7. rules={
  8. "field": {
  9. "realm.name": "ldap1"
  10. }
  11. },
  12. )
  13. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping3",
  3. roles: ["ldap-user"],
  4. enabled: true,
  5. rules: {
  6. field: {
  7. "realm.name": "ldap1",
  8. },
  9. },
  10. });
  11. console.log(response);
  1. POST /_security/role_mapping/mapping3
  2. {
  3. "roles": [ "ldap-user" ],
  4. "enabled": true,
  5. "rules": {
  6. "field" : { "realm.name" : "ldap1" }
  7. }
  8. }

The following example matches any user where either the username is esadmin or the user is in the cn=admin,dc=example,dc=com group:

  1. resp = client.security.put_role_mapping(
  2. name="mapping4",
  3. roles=[
  4. "superuser"
  5. ],
  6. enabled=True,
  7. rules={
  8. "any": [
  9. {
  10. "field": {
  11. "username": "esadmin"
  12. }
  13. },
  14. {
  15. "field": {
  16. "groups": "cn=admins,dc=example,dc=com"
  17. }
  18. }
  19. ]
  20. },
  21. )
  22. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping4",
  3. roles: ["superuser"],
  4. enabled: true,
  5. rules: {
  6. any: [
  7. {
  8. field: {
  9. username: "esadmin",
  10. },
  11. },
  12. {
  13. field: {
  14. groups: "cn=admins,dc=example,dc=com",
  15. },
  16. },
  17. ],
  18. },
  19. });
  20. console.log(response);
  1. POST /_security/role_mapping/mapping4
  2. {
  3. "roles": [ "superuser" ],
  4. "enabled": true,
  5. "rules": {
  6. "any": [
  7. {
  8. "field": {
  9. "username": "esadmin"
  10. }
  11. },
  12. {
  13. "field": {
  14. "groups": "cn=admins,dc=example,dc=com"
  15. }
  16. }
  17. ]
  18. }
  19. }

The example above is useful when the group names in your identity management system (such as Active Directory, or a SAML Identity Provider) do not have a 1-to-1 correspondence with the names of roles in Elasticsearch. The role mapping is the means by which you link a group name with a role name.

If there are multiple groups, you can use array syntax for the groups field. This matches any of the groups (rather than all of the groups):

  1. resp = client.security.put_role_mapping(
  2. name="mapping4",
  3. roles=[
  4. "superuser"
  5. ],
  6. enabled=True,
  7. rules={
  8. "any": [
  9. {
  10. "field": {
  11. "username": "esadmin"
  12. }
  13. },
  14. {
  15. "field": {
  16. "groups": [
  17. "cn=admins,dc=example,dc=com",
  18. "cn=other,dc=example,dc=com"
  19. ]
  20. }
  21. }
  22. ]
  23. },
  24. )
  25. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping4",
  3. roles: ["superuser"],
  4. enabled: true,
  5. rules: {
  6. any: [
  7. {
  8. field: {
  9. username: "esadmin",
  10. },
  11. },
  12. {
  13. field: {
  14. groups: ["cn=admins,dc=example,dc=com", "cn=other,dc=example,dc=com"],
  15. },
  16. },
  17. ],
  18. },
  19. });
  20. console.log(response);
  1. POST /_security/role_mapping/mapping4
  2. {
  3. "roles": [ "superuser" ],
  4. "enabled": true,
  5. "rules": {
  6. "any": [
  7. {
  8. "field": {
  9. "username": "esadmin"
  10. }
  11. },
  12. {
  13. "field": {
  14. "groups": [
  15. "cn=admins,dc=example,dc=com",
  16. "cn=other,dc=example,dc=com"
  17. ]
  18. }
  19. }
  20. ]
  21. }
  22. }

However, in rare cases the names of your groups may be an exact match for the names of your Elasticsearch roles. This can be the case when your SAML Identity Provider includes its own “group mapping” feature and can be configured to release Elasticsearch role names in the user’s SAML attributes.

In these cases it is possible to use a template that treats the group names as role names.

Note: This should only be done if you intend to define roles for all of the provided groups. Mapping a user to a large number of unnecessary or undefined roles is inefficient and can have a negative effect on system performance. If you only need to map a subset of the groups, then you should do this using explicit mappings.

  1. resp = client.security.put_role_mapping(
  2. name="mapping5",
  3. role_templates=[
  4. {
  5. "template": {
  6. "source": "{{#tojson}}groups{{/tojson}}"
  7. },
  8. "format": "json"
  9. }
  10. ],
  11. rules={
  12. "field": {
  13. "realm.name": "saml1"
  14. }
  15. },
  16. enabled=True,
  17. )
  18. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping5",
  3. role_templates: [
  4. {
  5. template: {
  6. source: "{{#tojson}}groups{{/tojson}}",
  7. },
  8. format: "json",
  9. },
  10. ],
  11. rules: {
  12. field: {
  13. "realm.name": "saml1",
  14. },
  15. },
  16. enabled: true,
  17. });
  18. console.log(response);
  1. POST /_security/role_mapping/mapping5
  2. {
  3. "role_templates": [
  4. {
  5. "template": { "source": "{{#tojson}}groups{{/tojson}}" },
  6. "format" : "json"
  7. }
  8. ],
  9. "rules": {
  10. "field" : { "realm.name" : "saml1" }
  11. },
  12. "enabled": true
  13. }

The tojson mustache function is used to convert the list of group names into a valid JSON array.

Because the template produces a JSON array, the format must be set to json.

The following example matches users within a specific LDAP sub-tree:

  1. resp = client.security.put_role_mapping(
  2. name="mapping6",
  3. roles=[
  4. "example-user"
  5. ],
  6. enabled=True,
  7. rules={
  8. "field": {
  9. "dn": "*,ou=subtree,dc=example,dc=com"
  10. }
  11. },
  12. )
  13. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping6",
  3. roles: ["example-user"],
  4. enabled: true,
  5. rules: {
  6. field: {
  7. dn: "*,ou=subtree,dc=example,dc=com",
  8. },
  9. },
  10. });
  11. console.log(response);
  1. POST /_security/role_mapping/mapping6
  2. {
  3. "roles": [ "example-user" ],
  4. "enabled": true,
  5. "rules": {
  6. "field" : { "dn" : "*,ou=subtree,dc=example,dc=com" }
  7. }
  8. }

The following example matches users within a particular LDAP sub-tree in a specific realm:

  1. resp = client.security.put_role_mapping(
  2. name="mapping7",
  3. roles=[
  4. "ldap-example-user"
  5. ],
  6. enabled=True,
  7. rules={
  8. "all": [
  9. {
  10. "field": {
  11. "dn": "*,ou=subtree,dc=example,dc=com"
  12. }
  13. },
  14. {
  15. "field": {
  16. "realm.name": "ldap1"
  17. }
  18. }
  19. ]
  20. },
  21. )
  22. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping7",
  3. roles: ["ldap-example-user"],
  4. enabled: true,
  5. rules: {
  6. all: [
  7. {
  8. field: {
  9. dn: "*,ou=subtree,dc=example,dc=com",
  10. },
  11. },
  12. {
  13. field: {
  14. "realm.name": "ldap1",
  15. },
  16. },
  17. ],
  18. },
  19. });
  20. console.log(response);
  1. POST /_security/role_mapping/mapping7
  2. {
  3. "roles": [ "ldap-example-user" ],
  4. "enabled": true,
  5. "rules": {
  6. "all": [
  7. { "field" : { "dn" : "*,ou=subtree,dc=example,dc=com" } },
  8. { "field" : { "realm.name" : "ldap1" } }
  9. ]
  10. }
  11. }

The rules can be more complex and include wildcard matching. For example, the following mapping matches any user where all of these conditions are met:

  • the Distinguished Name matches the pattern *,ou=admin,dc=example,dc=com, or the username is es-admin, or the username is es-system
  • the user in the cn=people,dc=example,dc=com group
  • the user does not have a terminated_date
  1. resp = client.security.put_role_mapping(
  2. name="mapping8",
  3. roles=[
  4. "superuser"
  5. ],
  6. enabled=True,
  7. rules={
  8. "all": [
  9. {
  10. "any": [
  11. {
  12. "field": {
  13. "dn": "*,ou=admin,dc=example,dc=com"
  14. }
  15. },
  16. {
  17. "field": {
  18. "username": [
  19. "es-admin",
  20. "es-system"
  21. ]
  22. }
  23. }
  24. ]
  25. },
  26. {
  27. "field": {
  28. "groups": "cn=people,dc=example,dc=com"
  29. }
  30. },
  31. {
  32. "except": {
  33. "field": {
  34. "metadata.terminated_date": None
  35. }
  36. }
  37. }
  38. ]
  39. },
  40. )
  41. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping8",
  3. roles: ["superuser"],
  4. enabled: true,
  5. rules: {
  6. all: [
  7. {
  8. any: [
  9. {
  10. field: {
  11. dn: "*,ou=admin,dc=example,dc=com",
  12. },
  13. },
  14. {
  15. field: {
  16. username: ["es-admin", "es-system"],
  17. },
  18. },
  19. ],
  20. },
  21. {
  22. field: {
  23. groups: "cn=people,dc=example,dc=com",
  24. },
  25. },
  26. {
  27. except: {
  28. field: {
  29. "metadata.terminated_date": null,
  30. },
  31. },
  32. },
  33. ],
  34. },
  35. });
  36. console.log(response);
  1. POST /_security/role_mapping/mapping8
  2. {
  3. "roles": [ "superuser" ],
  4. "enabled": true,
  5. "rules": {
  6. "all": [
  7. {
  8. "any": [
  9. {
  10. "field": {
  11. "dn": "*,ou=admin,dc=example,dc=com"
  12. }
  13. },
  14. {
  15. "field": {
  16. "username": [ "es-admin", "es-system" ]
  17. }
  18. }
  19. ]
  20. },
  21. {
  22. "field": {
  23. "groups": "cn=people,dc=example,dc=com"
  24. }
  25. },
  26. {
  27. "except": {
  28. "field": {
  29. "metadata.terminated_date": null
  30. }
  31. }
  32. }
  33. ]
  34. }
  35. }

A templated role can be used to automatically map every user to their own custom role. The role itself can be defined through the Roles API or using a custom roles provider.

In this example every user who authenticates using the “cloud-saml” realm will be automatically mapped to two roles - the "saml_user" role and a role that is their username prefixed with _user_. As an example, the user nwong would be assigned the saml_user and _user_nwong roles.

  1. resp = client.security.put_role_mapping(
  2. name="mapping9",
  3. rules={
  4. "field": {
  5. "realm.name": "cloud-saml"
  6. }
  7. },
  8. role_templates=[
  9. {
  10. "template": {
  11. "source": "saml_user"
  12. }
  13. },
  14. {
  15. "template": {
  16. "source": "_user_{{username}}"
  17. }
  18. }
  19. ],
  20. enabled=True,
  21. )
  22. print(resp)
  1. const response = await client.security.putRoleMapping({
  2. name: "mapping9",
  3. rules: {
  4. field: {
  5. "realm.name": "cloud-saml",
  6. },
  7. },
  8. role_templates: [
  9. {
  10. template: {
  11. source: "saml_user",
  12. },
  13. },
  14. {
  15. template: {
  16. source: "_user_{{username}}",
  17. },
  18. },
  19. ],
  20. enabled: true,
  21. });
  22. console.log(response);
  1. POST /_security/role_mapping/mapping9
  2. {
  3. "rules": { "field": { "realm.name": "cloud-saml" } },
  4. "role_templates": [
  5. { "template": { "source" : "saml_user" } },
  6. { "template": { "source" : "_user_{{username}}" } }
  7. ],
  8. "enabled": true
  9. }

Because it is not possible to specify both roles and role_templates in the same role mapping, we can apply a “fixed name” role by using a template that has no substitutions.