Submitting requests on behalf of other users

Submitting requests on behalf of other users

Elasticsearch roles support a run_as privilege that enables an authenticated user to submit requests on behalf of other users. For example, if your external application is trusted to authenticate users, Elasticsearch can authenticate the external application and use the run as mechanism to issue authorized requests as other users without having to re-authenticate each user.

To “run as” (impersonate) another user, the first user (the authenticating user) must be authenticated by a mechanism that supports run-as delegation. The second user (the run_as user) must be authorized by a mechanism that supports delegated run-as lookups by username.

The run_as privilege essentially operates like a secondary form of delegated authorization. Delegated authorization applies to the authenticating user, and the run_as privilege applies to the user who is being impersonated.

Authenticating user

For the authenticating user, the following realms (plus API keys) all support run_as delegation: native, file, Active Directory, JWT, Kerberos, LDAP and PKI.

Service tokens, the Elasticsearch Token Service, SAML 2.0, and OIDC 1.0 do not support run_as delegation.

run_as user

Elasticsearch supports run_as for any realm that supports user lookup. Not all realms support user lookup. Refer to the list of supported realms and ensure that the realm you wish to use is configured in a manner that supports user lookup.

The run_as user must be retrieved from a realm - it is not possible to run as a service account, API key or access token.

To submit requests on behalf of other users, you need to have the run_as privilege in your roles. For example, the following request creates a my_director role that grants permission to submit request on behalf of jacknich or redeniro:

  1. resp = client.security.put_role(
  2. name="my_director",
  3. refresh=True,
  4. cluster=[
  5. "manage"
  6. ],
  7. indices=[
  8. {
  9. "names": [
  10. "index1",
  11. "index2"
  12. ],
  13. "privileges": [
  14. "manage"
  15. ]
  16. }
  17. ],
  18. run_as=[
  19. "jacknich",
  20. "rdeniro"
  21. ],
  22. metadata={
  23. "version": 1
  24. },
  25. )
  26. print(resp)
  1. const response = await client.security.putRole({
  2. name: "my_director",
  3. refresh: "true",
  4. cluster: ["manage"],
  5. indices: [
  6. {
  7. names: ["index1", "index2"],
  8. privileges: ["manage"],
  9. },
  10. ],
  11. run_as: ["jacknich", "rdeniro"],
  12. metadata: {
  13. version: 1,
  14. },
  15. });
  16. console.log(response);
  1. POST /_security/role/my_director?refresh=true
  2. {
  3. "cluster": ["manage"],
  4. "indices": [
  5. {
  6. "names": [ "index1", "index2" ],
  7. "privileges": [ "manage" ]
  8. }
  9. ],
  10. "run_as": [ "jacknich", "rdeniro" ],
  11. "metadata" : {
  12. "version" : 1
  13. }
  14. }

To submit a request as another user, you specify the user in the es-security-runas-user request header. For example:

  1. curl -H "es-security-runas-user: jacknich" -u es-admin -X GET http://localhost:9200/

The run_as user passed in through the es-security-runas-user header must be available from a realm that supports delegated authorization lookup by username. Realms that don’t support user lookup can’t be used by run_as delegation from other realms.

For example, JWT realms can authenticate external users specified in JWTs, and execute requests as a run_as user in the native realm. Elasticsearch will retrieve the indicated runas user and execute the request as that user using their roles.

Apply the run_as privilege to roles

You can apply the run_as privilege when creating roles with the create or update roles API. Users who are assigned a role that contains the run_as privilege inherit all privileges from their role, and can also submit requests on behalf of the indicated users.

Roles for the authenticated user and the run_as user are not merged. If a user authenticates without specifying the run_as parameter, only the authenticated user’s roles are used. If a user authenticates and their roles include the run_as parameter, only the run_as user’s roles are used.

After a user successfully authenticates to Elasticsearch, an authorization process determines whether the user behind an incoming request is allowed to run that request. If the authenticated user has the run_as privilege in their list of permissions and specifies the run-as header, Elasticsearch discards the authenticated user and associated roles. It then looks in each of the configured realms in the realm chain until it finds the username that’s associated with the run_as user, and uses those roles to execute any requests.

Consider an admin role and an analyst role. The admin role has higher privileges, but might also want to submit requests as another user to test and verify their permissions.

First, we’ll create an admin role named my_admin_role. This role has manage privileges on the entire cluster, and on a subset of indices. This role also contains the run_as privilege, which enables any user with this role to submit requests on behalf of the specified analyst_user.

  1. resp = client.security.put_role(
  2. name="my_admin_role",
  3. refresh=True,
  4. cluster=[
  5. "manage"
  6. ],
  7. indices=[
  8. {
  9. "names": [
  10. "index1",
  11. "index2"
  12. ],
  13. "privileges": [
  14. "manage"
  15. ]
  16. }
  17. ],
  18. applications=[
  19. {
  20. "application": "myapp",
  21. "privileges": [
  22. "admin",
  23. "read"
  24. ],
  25. "resources": [
  26. "*"
  27. ]
  28. }
  29. ],
  30. run_as=[
  31. "analyst_user"
  32. ],
  33. metadata={
  34. "version": 1
  35. },
  36. )
  37. print(resp)
  1. const response = await client.security.putRole({
  2. name: "my_admin_role",
  3. refresh: "true",
  4. cluster: ["manage"],
  5. indices: [
  6. {
  7. names: ["index1", "index2"],
  8. privileges: ["manage"],
  9. },
  10. ],
  11. applications: [
  12. {
  13. application: "myapp",
  14. privileges: ["admin", "read"],
  15. resources: ["*"],
  16. },
  17. ],
  18. run_as: ["analyst_user"],
  19. metadata: {
  20. version: 1,
  21. },
  22. });
  23. console.log(response);
  1. POST /_security/role/my_admin_role?refresh=true
  2. {
  3. "cluster": ["manage"],
  4. "indices": [
  5. {
  6. "names": [ "index1", "index2" ],
  7. "privileges": [ "manage" ]
  8. }
  9. ],
  10. "applications": [
  11. {
  12. "application": "myapp",
  13. "privileges": [ "admin", "read" ],
  14. "resources": [ "*" ]
  15. }
  16. ],
  17. "run_as": [ "analyst_user" ],
  18. "metadata" : {
  19. "version" : 1
  20. }
  21. }

Next, we’ll create an analyst role named my_analyst_role, which has more restricted monitor cluster privileges and manage privileges on a subset of indices.

  1. resp = client.security.put_role(
  2. name="my_analyst_role",
  3. refresh=True,
  4. cluster=[
  5. "monitor"
  6. ],
  7. indices=[
  8. {
  9. "names": [
  10. "index1",
  11. "index2"
  12. ],
  13. "privileges": [
  14. "manage"
  15. ]
  16. }
  17. ],
  18. applications=[
  19. {
  20. "application": "myapp",
  21. "privileges": [
  22. "read"
  23. ],
  24. "resources": [
  25. "*"
  26. ]
  27. }
  28. ],
  29. metadata={
  30. "version": 1
  31. },
  32. )
  33. print(resp)
  1. const response = await client.security.putRole({
  2. name: "my_analyst_role",
  3. refresh: "true",
  4. cluster: ["monitor"],
  5. indices: [
  6. {
  7. names: ["index1", "index2"],
  8. privileges: ["manage"],
  9. },
  10. ],
  11. applications: [
  12. {
  13. application: "myapp",
  14. privileges: ["read"],
  15. resources: ["*"],
  16. },
  17. ],
  18. metadata: {
  19. version: 1,
  20. },
  21. });
  22. console.log(response);
  1. POST /_security/role/my_analyst_role?refresh=true
  2. {
  3. "cluster": [ "monitor"],
  4. "indices": [
  5. {
  6. "names": [ "index1", "index2" ],
  7. "privileges": ["manage"]
  8. }
  9. ],
  10. "applications": [
  11. {
  12. "application": "myapp",
  13. "privileges": [ "read" ],
  14. "resources": [ "*" ]
  15. }
  16. ],
  17. "metadata" : {
  18. "version" : 1
  19. }
  20. }

We’ll create an administrator user and assign them the role named my_admin_role, which allows this user to submit requests as the analyst_user.

  1. resp = client.security.put_user(
  2. username="admin_user",
  3. refresh=True,
  4. password="l0ng-r4nd0m-p@ssw0rd",
  5. roles=[
  6. "my_admin_role"
  7. ],
  8. full_name="Eirian Zola",
  9. metadata={
  10. "intelligence": 7
  11. },
  12. )
  13. print(resp)
  1. const response = await client.security.putUser({
  2. username: "admin_user",
  3. refresh: "true",
  4. password: "l0ng-r4nd0m-p@ssw0rd",
  5. roles: ["my_admin_role"],
  6. full_name: "Eirian Zola",
  7. metadata: {
  8. intelligence: 7,
  9. },
  10. });
  11. console.log(response);
  1. POST /_security/user/admin_user?refresh=true
  2. {
  3. "password": "l0ng-r4nd0m-p@ssw0rd",
  4. "roles": [ "my_admin_role" ],
  5. "full_name": "Eirian Zola",
  6. "metadata": { "intelligence" : 7}
  7. }

We can also create an analyst user and assign them the role named my_analyst_role.

  1. resp = client.security.put_user(
  2. username="analyst_user",
  3. refresh=True,
  4. password="l0nger-r4nd0mer-p@ssw0rd",
  5. roles=[
  6. "my_analyst_role"
  7. ],
  8. full_name="Monday Jaffe",
  9. metadata={
  10. "innovation": 8
  11. },
  12. )
  13. print(resp)
  1. const response = await client.security.putUser({
  2. username: "analyst_user",
  3. refresh: "true",
  4. password: "l0nger-r4nd0mer-p@ssw0rd",
  5. roles: ["my_analyst_role"],
  6. full_name: "Monday Jaffe",
  7. metadata: {
  8. innovation: 8,
  9. },
  10. });
  11. console.log(response);
  1. POST /_security/user/analyst_user?refresh=true
  2. {
  3. "password": "l0nger-r4nd0mer-p@ssw0rd",
  4. "roles": [ "my_analyst_role" ],
  5. "full_name": "Monday Jaffe",
  6. "metadata": { "innovation" : 8}
  7. }

You can then authenticate to Elasticsearch as the admin_user or analyst_user. However, the admin_user could optionally submit requests on behalf of the analyst_user. The following request authenticates to Elasticsearch with a Basic authorization token and submits the request as the analyst_user:

  1. curl -s -X GET -H "Authorization: Basic YWRtaW5fdXNlcjpsMG5nLXI0bmQwbS1wQHNzdzByZA==" -H "es-security-runas-user: analyst_user" https://localhost:9200/_security/_authenticate

The response indicates that the analyst_user submitted this request, using the my_analyst_role that’s assigned to that user. When the admin_user submitted the request, Elasticsearch authenticated that user, discarded their roles, and then used the roles of the run_as user.

  1. {"username":"analyst_user","roles":["my_analyst_role"],"full_name":"Monday Jaffe","email":null,
  2. "metadata":{"innovation":8},"enabled":true,"authentication_realm":{"name":"native",
  3. "type":"native"},"lookup_realm":{"name":"native","type":"native"},"authentication_type":"realm"}
  4. %

The authentication_realm and lookup_realm in the response both specify the native realm because both the admin_user and analyst_user are from that realm. If the two users are in different realms, the values for authentication_realm and lookup_realm are different (such as pki and native).