Defining Search Guard roles
Content
Hint: You can also use the Search Guard Configuration GUI for configuring Roles and Permissions.
Search Guard roles are the central place to configure access permissions on:
- Cluster level
- Index level
- Document level
- Field level
- Kibana level
Search Guard roles and their associated permissions are defined in the file sg_roles.yml
. The syntax to define a role is as follows:
<role_name>:
cluster_permissions:
- '<action group or single permission>'
- ...
index_permissions:
- index_patterns:
- <index pattern the allowed actions should be applied to>
- <index pattern the allowed actions should be applied to>
- ...
allowed_actions:
- '<action group or single permission>'
- ...
dls: '<Document level security query>'
fls:
- '<field level security field>'
- '<field level security field>'
- ...
- index_patterns:
- ...
tenant_permissions:
- tenant_patterns:
- <tenant pattern the allowed actions should be applied to>
- <tenant pattern the allowed actions should be applied to>
- ...
allowed_actions:
- SGS_KIBANA_ALL_WRITE
- tenant_patterns:
- ...
Description
Name | Description |
---|---|
cluster_permissions | Permissions that apply on the cluster level, e.g. monitoring the cluster health |
index_permissions | Permissions that apply to one or more index patterns |
allowed_actions | The actions that are allowed for the index or tenant patterns |
dls | The Document-level security filter query that should be applied to the index patterns. Used to filter documents from the result set. |
fls | The fields that should be excluded or included that should be applied to the index patterns. Used to filter fields from the documents in the result set. |
tenant_permissions | Permissions that apply to Kibana tenants. Used to control access to Kibana. |
Cluster-level permissions
The cluster_permissions
entry is used to define permissions on cluster level. Cluster-level permissions are used to allow/disallow actions that affect either the whole cluster, like querying the cluster health or the nodes stats.
They are also used to allow/disallow actions that affect multiple indices, like mget
, msearch
or bulk
requests.
Example:
sg_finance:
cluster_permissions:
- SGS_CLUSTER_COMPOSITE_OPS
index_permissions:
...
Index-level permissions
The index_permissions
entry is used to allow/disallow actions that affect indices matching the configured index patterns.
For example, to apply READ
permissions on a single index called humanresources
the configuration would look like:
<role_name>:
cluster_permissions:
- ...
index_permissions:
- index_patterns:
- "humanresources"
allowed_actions:
- READ
To apply READ
permissions to two indices called humanresources
and finance
you would write:
<role_name>:
cluster_permissions:
- ...
index_permissions:
- index_patterns:
- "humanresources"
- "finance"
allowed_actions:
- READ
To apply READ
and WRITE
permissions to two indices called humanresources
and finance
you would write:
<role_name>:
cluster_permissions:
- ...
index_permissions:
- index_patterns:
- "humanresources"
- "finance"
allowed_actions:
- READ
- WRITE
Dynamic index patterns: Wildcards and regular expressions
When defining index patterns you can use wildcards and regular expressions:
- An asterisk (
*
) will match any character sequence (or an empty sequence)*my*index
will matchmy_first_index
as well asmyindex
but notmyindex1
.
- A question mark (
?
) will match any single character (but NOT empty character)?kibana
will match.kibana
but notkibana
- Regular expressions have to be enclosed in
/
:'/<java regex>/'
'/\S*/'
will match any non whitespace characters
Example:
<role_name>:
cluster_permissions:
- ...
index_permissions:
- index_patterns:
- "logstash-*"
allowed_actions:
- CRUD
Dynamic index patterns: User name substitution
When defining index patterns the placeholder ${user.name}
is allowed to support indices or aliases which contain the name of the user.
<role_name>:
cluster_permissions:
- ...
index_permissions:
- index_patterns:
- "index-${user.name}"
allowed_actions:
- CRUD
Dynamic index and tenant patterns: User attributes
Any authentication and authorization domain can provide additional user attributes that you can use for variable substitution in index patterns and tenant patterns for Kibana multi-tenancy.
For this, the auth domains need to configure a mapping from attributes specific to the particular domain to Search Guard user attributes. See the documentation of the respective auth method for details and examples:
If you’re unsure what attributes are available, you can always access the /_searchguard/authinfo
REST endpoint to check. The endpoint will list all attribute names for the currently logged in user.
Note: The attribute mapping mechanism described here supersedes the old mechanism which would automatically provide all attributes from
the authentication domain under the prefix ${attr....}
. The old mechanism is deprecated but still supported. Please migrate to the new
syntax to ensure compatibility with future Search Guard releases.
JWT Example:
Suppose a Json Web Token contains a claim department
:
{
"name": "John Doe",
"roles": "admin, devops",
"department":
{
"name": "operations",
"number": "17"
}
}
To use it as variable in index patterns and tenant patterns, map it to a Search Guard user attribute in the JWT authenticator configuration:
jwt_auth_domain:
...
http_authenticator:
type: jwt
...
config:
map_claims_to_user_attrs:
`department`: department.name
This maps the department.name
JWT claim to the user attribute department
.
You can then use the department
attribute in index patterns and tenant patterns like:
sg_own_index:
cluster_permissions:
- CLUSTER_COMPOSITE_OPS
index_permissions:
- index_patterns:
- 'dept_${user.attrs.department}':
allowed_actions:
- SGS_CRUD
tenant_permissions:
- tenant_patterns:
- 'dept_${user.attrs.department}'
allowed_actions:
- "SGS_KIBANA_ALL_WRITE"
In this example, Search Guard grants the SGS_CRUD
permissions to the index dept_operations
, and SGS_KIBANA_ALL_WRITE
to the tenant named dept_operations
.
Active Directory / LDAP Example
Suppose the LDAP record of the current user contains an attribute departmentName
with value operations
.
To use it as variable in index patterns and tenant patterns, map it to a Search Guard user attribute in the LDAP authenticator configuration like:
ldap:
...
http_authenticator:
...
authentication_backend:
type: ldap
config:
map_ldap_attrs_to_user_attrs:
department: departmentName
This maps the departmentName
LDAP attribute to the user attribute department
.
You can then use the department
attribute in index patterns and tenant patterns like:
sg_own_index:
cluster_permissions:
- CLUSTER_COMPOSITE_OPS
index_permissions:
- index_patterns:
- 'dept_${user.attrs.department}':
allowed_actions:
- SGS_CRUD
tenant_permissions:
- tenant_patterns:
- 'dept_${user.attrs.department}'
allowed_actions:
- "SGS_KIBANA_ALL_WRITE"
Suppose the LDAP attribute has the value operations
.
In this case, Search Guard grants the SGS_CRUD
permissions to the index dept_operations
,
and SGS_KIBANA_ALL_WRITE
to the tenant named dept_operations
.
Internal users Example
Suppose the internal users entry contains an attribute department
:
jdoe:
hash: ...
attributes:
departmentName: "operations"
In order to use this attribute, you need map it in the user_mapping.attributes
configuration inside sg_authc.yml
:
auth_domains:
- type: basic/internal_users_db
user_mapping.attributes.from:
department: user_entry.attributes.departmentName
You can then use the department
attribute in index patterns and tenant patterns like:
sg_own_index:
cluster_permissions:
- CLUSTER_COMPOSITE_OPS
index_permissions:
- index_patterns:
- 'dept_${user.attrs.department}':
allowed_actions:
- SGS_CRUD
tenant_permissions:
- tenant_patterns:
- 'dept_${user.attrs.department}'
allowed_actions:
- "SGS_KIBANA_ALL_WRITE"
In this example, Search Guard grants the SGS_CRUD
permissions to the index dept_operations
, and SGS_KIBANA_ALL_WRITE
permissions to the Kibana tenant dept_operations
.
Substitution Variable Functionality
Substitution variables are always enclosed in the characters ${
and }
. Inside the brackets, you specify the attribute name, optionally followed by a chain of operations on the attribute value.
The pipe character |
followed by a function name causes the attribute value to be processed by the function. You can arbitrarily chain functions.
Available functions are:
|toJson
: Converts the value to a string in JSON format. If the value is a string, it will be properly quoted and escaped. If the value is a number, it will be left untouched. If the value is an object or array, it will be converted into the corresponding JSON syntax.
|toRegexFragment
: Converts the current value to a regular expression fragment. If the current value is a single string, this function just returns the string. However, any characters that would have a special meaning in regular expressions (such as the period .
) will be escaped in order to be interpreted literally. If the current value is an array, this function returns the values of the array as an regex “or” expression in the form (value1|value2|value3|...)
. The values themselves are also escaped as described before. Use this function inside an index pattern of the form '/dept_${user.attrs.department|toRegexFragment}/'
.
|toString
: Converts the value to a simple string format. If the value is a string, it will be left without quotes.
|toList
: Makes sure that the value is a list (or, in JSON terms, an array). If the value is already a list, it will be left unchanged. If the value is a single value, it will be converted to a list containing the single value. You can use this function to ensure that the substituted value is always a list.
|head
: Extracts the first element of a list. If the current value is not a list, the current value is left unchanged. If the current value is an empty list, the current value will be changed to null
. You can use this function to ensure that the substituted value is always a scalar value.
|tail
: Extracts all but the first element of a list. If the current value is not a list, the current value will be set to an empty list.
Additionally, you can use the ?:
operator to provide a value in case the current value is unset, resp. null
. The value to be used in this case is specified after the ?:
in JSON syntax. You can use the ?:
operator at any place between other operations.
It is recommended to use the ?:
operator for all cases where it is not absolutely sure that a value is always present. If an attribute is unset and no fallback is provided by ?:
, the ES operation triggering the DLS query will be aborted with an error.
Examples:
${user.attr.department?:["17"]|toList|toJson}
: Provides a list/array of departments in JSON format. If the attribute user.attr.department
is not defined, an array containing the string "17"
is provided.
${user.attr.email|head?:"nobody@nowhere"|toJson}
: Extracts the first element from the list stored by the attribute user.attr.email
. If the attribute is unset, nobody@nowhere
will be used as fallback value. Additionally, if the attribute user.attr.email
contains an empty list, the |head
function will change the current value to null
; thus, also in this case the ?:
operator will provide "nobody@nowhere"
as a fallback.
${user.attr.xyz|tail|head?:0|toJson}
: Extracts the second element of a list and converts it to JSON format. If there is no second element, 0 is returned.
Multiple Variables
You can use as many variables, wildcards and regular expressions as needed, for example:
sg_own_index:
cluster_permissions:
- CLUSTER_COMPOSITE_OPS
index_permissions:
- index_patterns:
- 'logfiles-${attr_ldap_department}-${user_name}-*':
allowed_actions:
- SGS_CRUD
Permission Exclusions
Besides using cluster_permissions
and index_permissions
to positively define the permissions a user should have, it is also possible to explicitly define permissions a user may not have.
For this purpose, you can add the entries exclude_cluster_permissions
and exclude_index_permissions
to your role definitions. Permissions defined here are not granted to the user, even if there are cluster_permissions
or index_permissions
properties which would grant these permissions.
This means, that you can use cluster_permissions
and index_permissions
entries to define a broader set of permissions and then use exclude_cluster_permissions
and exclude_index_permissions
to selectively subtract permissions a user is not allowed to have.
For example, a role definition might now look like this:
my_role_using_exclusions:
cluster_permissions:
- *
exclude_cluster_permissions:
- SGS_MANAGE_SNAPSHOTS
index_permissions:
- index_patterns:
- "*"
allowed_actions:
- SGS_CRUD
exclude_index_permissions:
- index_patterns:
- "secret"
actions:
- "*"
A user with this role is granted all cluster permissions except permissions for managing snapshots. Furthermore, the user gets access to all indexes except the index secret
.
Permission exclusions do not only affect the permissions granted in the same role. Rather, permission exclusions also affect permissions granted by other roles. Thus, you can grant permissions in one role and use an additional role to selectively remove some of these permissions.
Cluster Permission Exclusions
Cluster permission exclusions are defined using an entry with the key exclude_cluster_permissions
. Just like it is the case for cluster_permissions
entries, you can provide a list of permissions here. The list entries can contain wildcards or action groups.
Index Permission Exclusions
Index permission exclusions are listed under the key exclude_index_permissions
. For defining such an exclusion, you need to define an index_pattern
and the excluded actions
.
For the index_pattern
entries, you can use variable substitution as described above. It is however important to note that any errors during variable substitution are more critical for permission exclusions. Such an error would be for example trying to use a user attribute which does not exist. If such an error occurs while checking the permissions for an action, Search Guard will fail into the safe direction and immediately deny the execution of the action. Search Guard will also write a warning to the Elasticsearch logs about this error.
Note: In index_pattern
entries for exclude_index_permissions
, the deprecated user attribute syntax ${attr.internal...}
is not supported any more. You need to use the new-style user attributes as described above.
The action
key takes a list of actions to be excluded. The list entries can contain wildcards or action groups. Note that this entry is named actions
- in contrast to the entry allowed_actions
in the normal index_permissions
entry.
Built-in Roles
Search Guard ships with the following built-in (static) roles:
Role name | Description |
---|---|
SGS_ALL_ACCESS | All cluster permissions and all index permissions on all indices |
SGS_READALL | Read permissions on all indices, but no write permissions |
SGS_READALL_AND_MONITOR | Read and monitor permissions on all indices, but no write permissions |
SGS_KIBANA_SERVER | Role for the internal Kibana server user, please refer to the Kibana setup chapter for explanation |
SGS_KIBANA_USER | Minimum permission set for regular Kibana users. In addition to this role, you need to also grant READ permissions on indices the user should be able to access in Kibana. |
SGS_LOGSTASH | Role for logstash and beats users, grants full access to all logstash and beats indices. |
SGS_MANAGE_SNAPSHOTS | Grants full permissions on snapshot, restore and repositories operations |
SGS_OWN_INDEX | Grants full permissions on an index named after the authenticated user’s username. |
SGS_XP_MONITORING | Role for X-Pack Monitoring. Users who wish to use X-Pack Monitoring need this role in addition to the sg_kibana_user role |
SGS_XP_ALERTING | Role for X-Pack Alerting. Users who wish to use X-Pack Alerting need this role in addition to the sg_kibana role |
SGS_XP_MACHINE_LEARNING | Role for X-Pack Machine Learning. Users who wish to use X-Pack Machine Learning need this role in addition to the sg_kibana role |