Leverage document-level security from connectors in Search Applications

Leverage document-level security from connectors in Search Applications

This guide explains how to ensure document-level security (DLS) for documents ingested by Elastic connectors, when building a search application.

In this example we will:

  • Set up the SharePoint Online connector to ingest data from SharePoint Online
  • Set up a Search Application using the Elasticsearch index created by the SharePoint Online connector
  • Create Elasticsearch API keys with DLS and workflow restrictions to query your Search Application
  • Build a search experience where authenticated users can search over the data ingested by connectors

Set up connector to sync data with access control

You can run SharePoint Online connector in Elastic Cloud (native) or on a self-managed deployment (self-managed connector). Refer to SharePoint Online connector to learn how to set up the SharePoint Online connector and enable DLS.

To run the self-managed connector, you’ll need to run the connectors service in addition to your Elastic deployment. Refer to Self-managed connectors for details on how to set up a self-managed connector and run the connectors service.

This guide assumes you already have an Elastic deployment, that satisfies the prerequisites for running the connectors service. If you don’t have an Elastic deployment, sign up for a free Elastic Cloud trial.

We use the SharePoint Online connector in this concrete example. Refer to Document level security (DLS) for a list of connectors that support DLS.

Elasticsearch indices overview

When the SharePoint Online connector is set up and you’ve started syncing content, the connector will create two separate Elasticsearch indices:

  • A content index that holds the searchable data in SharePoint Online. We’ll use this index to create our search application.
  • An access control index that includes access control data for each user that has access to SharePoint Online. It will be named .search-acl-filter-<your index name>, where <your index name> is the index name you chose. For example, an index named search-sharepoint would have the ACL filter index .search-acl-filter-search-sharepoint. We’ll use this index to create Elasticsearch API keys that control access to the content index.

Create a Search Application

To build our search experience for our SharePoint Online data, we need to create a Search Application.

Follow these steps to create a Search Application in the Kibana UI:

  1. Navigate to Search > Search Applications from the main menu, or use the global search field.
  2. Select Create.
  3. Name the Search Application.
  4. Select the index used by the SharePoint Online connector.
  5. Select Create.

Alternatively, you can use the Put Search Application API.

Create Elasticsearch API keys

Next we need to create Elasticsearch API keys to restrict queries to the search application. These restrictions will ensure that users can only query documents they have access to. To create this API key, we will leverage information in the access control index created by the connector.

The access control index will contain documents similar to this example:

  1. {
  2. "_index": ".search-acl-filter-search-sharepoint",
  3. "_id": "john@example.co",
  4. "_version": 1,
  5. "_seq_no": 0,
  6. "_primary_term": 1,
  7. "found": true,
  8. "_source": {
  9. "identity": {
  10. "email": "john@example.co",
  11. "access_control": [
  12. "john@example.co",
  13. "Engineering Members"
  14. ]
  15. },
  16. "query": {
  17. "template": {
  18. "params": {
  19. "access_control": [
  20. "john@example.co",
  21. "Engineering Members"
  22. ]
  23. },
  24. "source": """
  25. {
  26. "bool": {
  27. "should": [
  28. {
  29. "bool": {
  30. "must_not": {
  31. "exists": {
  32. "field": "_allow_access_control"
  33. }
  34. }
  35. }
  36. },
  37. {
  38. "terms": {
  39. "_allow_access_control.enum": {{#toJson}}access_control{{/toJson}}
  40. }
  41. }
  42. ]
  43. }
  44. }
  45. """
  46. }
  47. }
  48. }
  49. }

This document contains the Elasticsearch query that describes which documents the user john@example.com has access to. The access control information is stored in the access_control field. In this case the user has access only to documents that contain "john@example.co" or "Engineering Members" in the _allow_access_control field.

The query field contains the DLS query we will use to create an Elasticsearch API key. That key will ensure queries are restricted to the documents john@example.com has access to.

To create the API key, we will use the Create API Key API. The API call will look like this:

  1. resp = client.security.create_api_key(
  2. name="john-api-key",
  3. expiration="1d",
  4. role_descriptors={
  5. "sharepoint-online-role": {
  6. "index": [
  7. {
  8. "names": [
  9. "sharepoint-search-application"
  10. ],
  11. "privileges": [
  12. "read"
  13. ],
  14. "query": {
  15. "template": {
  16. "params": {
  17. "access_control": [
  18. "john@example.co",
  19. "Engineering Members"
  20. ]
  21. },
  22. "source": "\n {\n \"bool\": {\n \"should\": [\n {\n \"bool\": {\n \"must_not\": {\n \"exists\": {\n \"field\": \"_allow_access_control\"\n }\n }\n }\n },\n {\n \"terms\": {\n \"_allow_access_control.enum\": {{#toJson}}access_control{{/toJson}}\n }\n }\n ]\n }\n }\n "
  23. }
  24. }
  25. }
  26. ],
  27. "restriction": {
  28. "workflows": [
  29. "search_application_query"
  30. ]
  31. }
  32. }
  33. },
  34. )
  35. print(resp)
  1. const response = await client.security.createApiKey({
  2. name: "john-api-key",
  3. expiration: "1d",
  4. role_descriptors: {
  5. "sharepoint-online-role": {
  6. index: [
  7. {
  8. names: ["sharepoint-search-application"],
  9. privileges: ["read"],
  10. query: {
  11. template: {
  12. params: {
  13. access_control: ["john@example.co", "Engineering Members"],
  14. },
  15. source:
  16. '\n {\n "bool": {\n "should": [\n {\n "bool": {\n "must_not": {\n "exists": {\n "field": "_allow_access_control"\n }\n }\n }\n },\n {\n "terms": {\n "_allow_access_control.enum": {{#toJson}}access_control{{/toJson}}\n }\n }\n ]\n }\n }\n ',
  17. },
  18. },
  19. },
  20. ],
  21. restriction: {
  22. workflows: ["search_application_query"],
  23. },
  24. },
  25. },
  26. });
  27. console.log(response);
  1. POST /_security/api_key
  2. {
  3. "name": "john-api-key",
  4. "expiration": "1d",
  5. "role_descriptors": {
  6. "sharepoint-online-role": {
  7. "index": [
  8. {
  9. "names": [
  10. "sharepoint-search-application"
  11. ],
  12. "privileges": [
  13. "read"
  14. ],
  15. "query": {
  16. "template": {
  17. "params": {
  18. "access_control": [
  19. "john@example.co",
  20. "Engineering Members"
  21. ]
  22. },
  23. "source": """
  24. {
  25. "bool": {
  26. "should": [
  27. {
  28. "bool": {
  29. "must_not": {
  30. "exists": {
  31. "field": "_allow_access_control"
  32. }
  33. }
  34. }
  35. },
  36. {
  37. "terms": {
  38. "_allow_access_control.enum": {{#toJson}}access_control{{/toJson}}
  39. }
  40. }
  41. ]
  42. }
  43. }
  44. """
  45. }
  46. }
  47. }
  48. ],
  49. "restriction": {
  50. "workflows": [
  51. "search_application_query"
  52. ]
  53. }
  54. }
  55. }
  56. }

The response will look like this:

  1. {
  2. "id": "0rCD3i-MjKsw4g9BpRIBa",
  3. "name": "john-api-key",
  4. "expiration": 1687881715555,
  5. "api_key": "zTxre9L6TcmRIgd2NgLCRg",
  6. "encoded": "Qk05dy1JZ0JhRDNyNGpLQ3MwUmk6elRzdGU5QjZUY21SSWdkMldnQ1RMZw=="
  7. }

The api_key field contains the API key that can be used to query the Search Application with the appropriate DLS restrictions.

Querying multiple indices

This section describes how to generate an API key to query a search application that contains multiple indices with documents ingested by connectors with DLS.

A user might have multiple identities that define which documents they are allowed to read. In this case we want to create a single Elasticsearch API key that can be used to query only the documents this user has access to.

Let’s assume we want to create an API key that combines the following user identities:

  1. GET .search-acl-filter-source1
  2. {
  3. "_id": "example.user@example.com",
  4. "identity": {
  5. "username": "example username",
  6. "email": "example.user@example.com"
  7. },
  8. "query": {
  9. "template": {
  10. "params": {
  11. "access_control": [
  12. "example.user@example.com",
  13. "source1-user-group"]
  14. }
  15. },
  16. "source": "..."
  17. }
  18. }
  1. GET .search-acl-filter-source2
  2. {
  3. "_id": "example.user@example.com",
  4. "identity": {
  5. "username": "example username",
  6. "email": "example.user@example.com"
  7. },
  8. "query": {
  9. "template": {
  10. "params": {
  11. "access_control": [
  12. "example.user@example.com",
  13. "source2-user-group"]
  14. }
  15. },
  16. "source": "..."
  17. }
  18. }

.search-acl-filter-source1 and .search-acl-filter-source2 define the access control identities for source1 and source2.

The following script exemplifies how to generate the Elasticsearch API key that combines multiple user identities:

  1. require("dotenv").config();
  2. const axios = require("axios");
  3. // Elasticsearch URL and creds retrieved from environment variables
  4. const ELASTICSEARCH_URL = process.env.ELASTICSEARCH_URL;
  5. const ELASTICSEARCH_USER = process.env.ELASTICSEARCH_USER;
  6. const ELASTICSEARCH_PASSWORD = process.env.ELASTICSEARCH_PASSWORD;
  7. const config = {
  8. auth: {
  9. username: ELASTICSEARCH_USER,
  10. password: ELASTICSEARCH_PASSWORD,
  11. },
  12. headers: {
  13. "Content-Type": "application/json",
  14. },
  15. };
  16. async function createApiKey({
  17. searchApplication,
  18. userId,
  19. indices = "",
  20. metadata,
  21. expiration = "1d"
  22. }) {
  23. try {
  24. const indices = indices.split(",");
  25. let combinedQuery = { bool: { should: [] } };
  26. for (const index of indices) {
  27. const aclsIndex = `.search-acl-filter-${index}`;
  28. const response = await axios.get(
  29. `${ELASTICSEARCH_URL}/${aclsIndex}/_doc/${userId}`,
  30. config
  31. );
  32. combinedQuery.bool.should.push({
  33. bool: {
  34. must: [
  35. {
  36. term: {
  37. "_index": index,
  38. },
  39. },
  40. response.data._source.query.source,
  41. ],
  42. },
  43. });
  44. }
  45. if (!metadata || Object.keys(metadata).length === 0) {
  46. metadata = { created_by: "create-api-key" };
  47. }
  48. const apiKeyBody = {
  49. name: userId,
  50. expiration,
  51. role_descriptors: {
  52. [`${searchApplication}-role`]: {
  53. index: [
  54. {
  55. names: [searchApplication],
  56. privileges: ["read"],
  57. query: combinedQuery,
  58. },
  59. ],
  60. restriction: {
  61. workflows: ["search_application_query"],
  62. },
  63. },
  64. },
  65. metadata,
  66. };
  67. const apiKeyResponse = await axios.post(
  68. `${ELASTICSEARCH_URL}/_security/api_key`,
  69. apiKeyBody,
  70. config
  71. );
  72. console.log(apiKeyResponse.data);
  73. return apiKeyResponse.data.encoded;
  74. } catch (error) {
  75. console.log(error)
  76. }
  77. }
  78. // example usage:
  79. createApiKey({
  80. searchApplication: "my-search-app",
  81. userId: "example.user@example.com",
  82. indices: "source1,source2",
  83. expiration: "1d",
  84. metadata: {
  85. application: "my-search-app",
  86. namespace: "dev",
  87. foo: "bar",
  88. },
  89. }).then((encodedKey) => console.log(encodedKey));

The example combines multiple identities into a single role descriptor. This is because an Elasticsearch API key can use role restrictions only if it has a single role descriptor.

Implementation in your frontend application

If you’re building a frontend application, use the encoded field to pass the API key to the frontend. Your app can then use the API key to query the search application. The workflow will look something like this:

  1. User signs in to your application.
  2. Your application generates an Elasticsearch API key using the Create API Key API.
  3. The encoded field is returned to the frontend application.
  4. When the user searches for documents, the frontend application passes the encoded field to your search application’s _search endpoint. For example, you might use the Search Application client to make the actual queries using the API key:

    1. const client = SearchApplicationClient(applicationName, endpoint, apiKey, params);

Here’s what this workflow looks like in a sequence diagram:

DLS API key and search application client workflow

When creating an Elasticsearch API key for query Search Applications, you must include the search_application_query restriction. This will ensure the API key can only access the Search Application Search API.

We recommend always setting an expiration time when creating an Elasticsearch API key. When expiration is not set, the Elasticsearch API will never expire.

Workflow guidance

We recommend relying on the connector access control sync to automate and keep documents in sync with changes to the original content source’s user permissions.

In this workflow you will need to handle the generation of the Elasticsearch API key in the backend of your application, in response to browser sign ins.

Once the key is generated, the backend will also need to return that key to the client (browser) to be used in subsequent search requests to your search application.

The API key can be invalidated using the Invalidate API Key API. Additionally, if the user’s permission changes, you’ll need to update or recreate the Elasticsearch API key.

Next steps

Learn how to use the Search Application client to query your Search Application. See Search Applications client.

Learn more