Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Search across spaces #67644

Merged
merged 29 commits into from
Jul 14, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
36dd1b4
Allow saved objects to be searched across spaces
legrego Apr 20, 2020
bdb5110
Merge branch 'master' of github.com:elastic/kibana into spaces/search…
legrego Jun 16, 2020
10d4a2a
Start to address first round of feedback
legrego Jun 16, 2020
fa7b57d
Update x-pack/test/saved_object_api_integration/common/suites/find.ts
legrego Jun 17, 2020
602d8ff
Attempting option 3: wildcard is default namespace
legrego Jun 17, 2020
7e26c10
Merge branch 'spaces/search-across-spaces' of github.com:legrego/kiba…
legrego Jun 17, 2020
b0b2d95
Remove namespaces from common test cases
legrego Jun 17, 2020
8eb6bd3
fix type check
legrego Jun 17, 2020
7e50abf
omit namespaces from the export operation
legrego Jun 18, 2020
1ec9e7e
Merge branch 'master' of github.com:elastic/kibana into spaces/search…
legrego Jun 18, 2020
47e5e5e
fix types
legrego Jun 18, 2020
a9f7be0
Merge branch 'master' of github.com:elastic/kibana into spaces/search…
legrego Jun 23, 2020
eafbd10
fix mocked find result
legrego Jun 23, 2020
485f4b5
fix export api test
legrego Jun 23, 2020
5fbbf42
update find test
legrego Jun 24, 2020
0421c1a
Merge branch 'master' into spaces/search-across-spaces
elasticmachine Jun 24, 2020
d5796a8
start addressing feedback
legrego Jun 29, 2020
b6d2799
Merge branch 'spaces/search-across-spaces' of github.com:legrego/kiba…
legrego Jun 29, 2020
f1b8fc8
fixing repository test
legrego Jun 30, 2020
9f16270
Merge branch 'master' of github.com:elastic/kibana into spaces/search…
legrego Jun 30, 2020
078d3b7
simplify bulkCreate
legrego Jul 2, 2020
588f54b
Merge branch 'master' of github.com:elastic/kibana into spaces/search…
legrego Jul 2, 2020
793dc76
start addressing platform feedback
legrego Jul 2, 2020
effc986
Merge branch 'master' into spaces/search-across-spaces
elasticmachine Jul 2, 2020
0f50101
Merge branch 'master' of github.com:elastic/kibana into spaces/search…
legrego Jul 8, 2020
3f6c1bf
changing order of saved objects authorization checks
legrego Jul 8, 2020
6c2df02
Merge branch 'spaces/search-across-spaces' of github.com:legrego/kiba…
legrego Jul 8, 2020
7a095e8
address platform PR feedback
legrego Jul 13, 2020
c134749
Merge branch 'master' of github.com:elastic/kibana into spaces/search…
legrego Jul 13, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
<b>Signature:</b>

```typescript
export interface SavedObjectsFindOptions extends SavedObjectsBaseOptions
export interface SavedObjectsFindOptions extends Omit<SavedObjectsBaseOptions, 'namespace'>
```

## Properties
Expand All @@ -19,6 +19,7 @@ export interface SavedObjectsFindOptions extends SavedObjectsBaseOptions
| [fields](./kibana-plugin-core-server.savedobjectsfindoptions.fields.md) | <code>string[]</code> | An array of fields to include in the results |
| [filter](./kibana-plugin-core-server.savedobjectsfindoptions.filter.md) | <code>string</code> | |
| [hasReference](./kibana-plugin-core-server.savedobjectsfindoptions.hasreference.md) | <code>{</code><br/><code> type: string;</code><br/><code> id: string;</code><br/><code> }</code> | |
| [namespaces](./kibana-plugin-core-server.savedobjectsfindoptions.namespaces.md) | <code>string[]</code> | |
| [page](./kibana-plugin-core-server.savedobjectsfindoptions.page.md) | <code>number</code> | |
| [perPage](./kibana-plugin-core-server.savedobjectsfindoptions.perpage.md) | <code>number</code> | |
| [search](./kibana-plugin-core-server.savedobjectsfindoptions.search.md) | <code>string</code> | Search documents using the Elasticsearch Simple Query String syntax. See Elasticsearch Simple Query String <code>query</code> argument for more information |
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
<!-- Do not edit this file. It is automatically generated by API Documenter. -->

[Home](./index.md) &gt; [kibana-plugin-core-server](./kibana-plugin-core-server.md) &gt; [SavedObjectsFindOptions](./kibana-plugin-core-server.savedobjectsfindoptions.md) &gt; [namespaces](./kibana-plugin-core-server.savedobjectsfindoptions.namespaces.md)

## SavedObjectsFindOptions.namespaces property

<b>Signature:</b>

```typescript
namespaces?: string[];
```
Original file line number Diff line number Diff line change
Expand Up @@ -7,14 +7,14 @@
<b>Signature:</b>

```typescript
find<T = unknown>({ search, defaultSearchOperator, searchFields, hasReference, page, perPage, sortField, sortOrder, fields, namespace, type, filter, }: SavedObjectsFindOptions): Promise<SavedObjectsFindResponse<T>>;
find<T = unknown>({ search, defaultSearchOperator, searchFields, hasReference, page, perPage, sortField, sortOrder, fields, namespaces, type, filter, }: SavedObjectsFindOptions): Promise<SavedObjectsFindResponse<T>>;
```

## Parameters

| Parameter | Type | Description |
| --- | --- | --- |
| { search, defaultSearchOperator, searchFields, hasReference, page, perPage, sortField, sortOrder, fields, namespace, type, filter, } | <code>SavedObjectsFindOptions</code> | |
| { search, defaultSearchOperator, searchFields, hasReference, page, perPage, sortField, sortOrder, fields, namespaces, type, filter, } | <code>SavedObjectsFindOptions</code> | |

<b>Returns:</b>

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ export declare class SavedObjectsRepository
| [delete(type, id, options)](./kibana-plugin-core-server.savedobjectsrepository.delete.md) | | Deletes an object |
| [deleteByNamespace(namespace, options)](./kibana-plugin-core-server.savedobjectsrepository.deletebynamespace.md) | | Deletes all objects from the provided namespace. |
| [deleteFromNamespaces(type, id, namespaces, options)](./kibana-plugin-core-server.savedobjectsrepository.deletefromnamespaces.md) | | Removes one or more namespaces from a given multi-namespace saved object. If no namespaces remain, the saved object is deleted entirely. This method and \[<code>addToNamespaces</code>\][SavedObjectsRepository.addToNamespaces()](./kibana-plugin-core-server.savedobjectsrepository.addtonamespaces.md) are the only ways to change which Spaces a multi-namespace saved object is shared to. |
| [find({ search, defaultSearchOperator, searchFields, hasReference, page, perPage, sortField, sortOrder, fields, namespace, type, filter, })](./kibana-plugin-core-server.savedobjectsrepository.find.md) | | |
| [find({ search, defaultSearchOperator, searchFields, hasReference, page, perPage, sortField, sortOrder, fields, namespaces, type, filter, })](./kibana-plugin-core-server.savedobjectsrepository.find.md) | | |
| [get(type, id, options)](./kibana-plugin-core-server.savedobjectsrepository.get.md) | | Gets a single object |
| [incrementCounter(type, id, counterFieldName, options)](./kibana-plugin-core-server.savedobjectsrepository.incrementcounter.md) | | Increases a counter field by one. Creates the document if one doesn't exist for the given id. |
| [update(type, id, attributes, options)](./kibana-plugin-core-server.savedobjectsrepository.update.md) | | Updates an object |
Expand Down
6 changes: 4 additions & 2 deletions src/core/public/public.api.md
Original file line number Diff line number Diff line change
Expand Up @@ -1126,7 +1126,7 @@ export class SavedObjectsClient {
bulkUpdate<T = unknown>(objects?: SavedObjectsBulkUpdateObject[]): Promise<SavedObjectsBatchResponse<unknown>>;
create: <T = unknown>(type: string, attributes: T, options?: SavedObjectsCreateOptions) => Promise<SimpleSavedObject<T>>;
delete: (type: string, id: string) => Promise<{}>;
find: <T = unknown>(options: Pick<SavedObjectsFindOptions, "search" | "filter" | "type" | "page" | "perPage" | "sortField" | "fields" | "searchFields" | "hasReference" | "defaultSearchOperator">) => Promise<SavedObjectsFindResponsePublic<T>>;
find: <T = unknown>(options: Pick<SavedObjectsFindOptions, "search" | "filter" | "type" | "page" | "perPage" | "sortField" | "fields" | "searchFields" | "hasReference" | "defaultSearchOperator" | "namespaces">) => Promise<SavedObjectsFindResponsePublic<T>>;
get: <T = unknown>(type: string, id: string) => Promise<SimpleSavedObject<T>>;
update<T = unknown>(type: string, id: string, attributes: T, { version, migrationVersion, references }?: SavedObjectsUpdateOptions): Promise<SimpleSavedObject<T>>;
}
Expand All @@ -1144,7 +1144,7 @@ export interface SavedObjectsCreateOptions {
}

// @public (undocumented)
export interface SavedObjectsFindOptions extends SavedObjectsBaseOptions {
export interface SavedObjectsFindOptions extends Omit<SavedObjectsBaseOptions, 'namespace'> {
// (undocumented)
defaultSearchOperator?: 'AND' | 'OR';
fields?: string[];
Expand All @@ -1156,6 +1156,8 @@ export interface SavedObjectsFindOptions extends SavedObjectsBaseOptions {
id: string;
};
// (undocumented)
namespaces?: string[];
// (undocumented)
page?: number;
// (undocumented)
perPage?: number;
Expand Down
1 change: 1 addition & 0 deletions src/core/public/saved_objects/saved_objects_client.ts
Original file line number Diff line number Diff line change
Expand Up @@ -292,6 +292,7 @@ export class SavedObjectsClient {
sortField: 'sort_field',
type: 'type',
filter: 'filter',
namespaces: 'namespaces',
};

const renamedQuery = renameKeys<SavedObjectsFindOptions, any>(renameMap, options);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -105,7 +105,7 @@ describe('getSortedObjectsForExport()', () => {
"calls": Array [
Array [
Object {
"namespace": undefined,
"namespaces": undefined,
"perPage": 500,
"search": undefined,
"type": Array [
Expand Down Expand Up @@ -251,7 +251,7 @@ describe('getSortedObjectsForExport()', () => {
"calls": Array [
Array [
Object {
"namespace": undefined,
"namespaces": undefined,
"perPage": 500,
"search": "foo",
"type": Array [
Expand Down Expand Up @@ -338,7 +338,9 @@ describe('getSortedObjectsForExport()', () => {
"calls": Array [
Array [
Object {
"namespace": "foo",
"namespaces": Array [
"foo",
],
"perPage": 500,
"search": undefined,
"type": Array [
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -109,7 +109,7 @@ async function fetchObjectsToExport({
type: types,
search,
perPage: exportSizeLimit,
namespace,
namespaces: namespace ? [namespace] : undefined,
rudolf marked this conversation as resolved.
Show resolved Hide resolved
});
if (findResponse.total > exportSizeLimit) {
throw Boom.badRequest(`Can't export more than ${exportSizeLimit} objects`);
Expand Down
12 changes: 12 additions & 0 deletions src/core/server/saved_objects/routes/find.ts
Original file line number Diff line number Diff line change
Expand Up @@ -45,11 +45,22 @@ export const registerFindRoute = (router: IRouter) => {
),
fields: schema.maybe(schema.oneOf([schema.string(), schema.arrayOf(schema.string())])),
filter: schema.maybe(schema.string()),
namespaces: schema.maybe(
schema.oneOf([schema.string(), schema.arrayOf(schema.string())])
),
}),
},
},
router.handleLegacyErrors(async (context, req, res) => {
const query = req.query;

let namespaces: string[] | undefined;
if (Array.isArray(req.query.namespaces)) {
namespaces = req.query.namespaces;
} else if (typeof req.query.namespaces === 'string') {
namespaces = [req.query.namespaces];
}
legrego marked this conversation as resolved.
Show resolved Hide resolved

const result = await context.core.savedObjects.client.find({
perPage: query.per_page,
page: query.page,
Expand All @@ -62,6 +73,7 @@ export const registerFindRoute = (router: IRouter) => {
hasReference: query.has_reference,
fields: typeof query.fields === 'string' ? [query.fields] : query.fields,
filter: query.filter,
namespaces,
});

return res.ok({ body: result });
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ describe('GET /api/saved_objects/_find', () => {
notExpandable: true,
attributes: {},
references: [],
namespaces: ['default'],
},
{
type: 'index-pattern',
Expand All @@ -89,6 +90,7 @@ describe('GET /api/saved_objects/_find', () => {
notExpandable: true,
attributes: {},
references: [],
namespaces: ['default'],
},
],
};
Expand Down Expand Up @@ -239,4 +241,38 @@ describe('GET /api/saved_objects/_find', () => {
defaultSearchOperator: 'OR',
});
});

it('accepts the query parameter namespaces as a string', async () => {
await supertest(httpSetup.server.listener)
.get('/api/saved_objects/_find?type=index-pattern&namespaces=foo')
.expect(200);

expect(savedObjectsClient.find).toHaveBeenCalledTimes(1);

const options = savedObjectsClient.find.mock.calls[0][0];
expect(options).toEqual({
perPage: 20,
page: 1,
type: ['index-pattern'],
namespaces: ['foo'],
defaultSearchOperator: 'OR',
});
});

it('accepts the query parameter namespaces as an array', async () => {
await supertest(httpSetup.server.listener)
.get('/api/saved_objects/_find?type=index-pattern&namespaces=default&namespaces=foo')
.expect(200);

expect(savedObjectsClient.find).toHaveBeenCalledTimes(1);

const options = savedObjectsClient.find.mock.calls[0][0];
expect(options).toEqual({
perPage: 20,
page: 1,
type: ['index-pattern'],
namespaces: ['default', 'foo'],
defaultSearchOperator: 'OR',
});
});
});
60 changes: 40 additions & 20 deletions src/core/server/saved_objects/service/lib/repository.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -494,6 +494,7 @@ describe('SavedObjectsRepository', () => {
...obj,
migrationVersion: { [obj.type]: '1.1.1' },
version: mockVersion,
namespaces: obj.namespaces ?? [obj.namespace ?? 'default'],
...mockTimestampFields,
});

Expand Down Expand Up @@ -826,9 +827,23 @@ describe('SavedObjectsRepository', () => {
// Assert that both raw docs from the ES response are deserialized
expect(serializer.rawToSavedObject).toHaveBeenNthCalledWith(1, {
...response.items[0].create,
_source: {
...response.items[0].create._source,
namespaces: response.items[0].create._source.namespaces ?? [
response.items[0].create._source.namespace ?? 'default',
],
},
_id: expect.stringMatching(/^myspace:config:[0-9a-f]{8}-([0-9a-f]{4}-){3}[0-9a-f]{12}$/),
});
expect(serializer.rawToSavedObject).toHaveBeenNthCalledWith(2, response.items[1].create);
expect(serializer.rawToSavedObject).toHaveBeenNthCalledWith(2, {
...response.items[1].create,
_source: {
...response.items[1].create._source,
namespaces: response.items[1].create._source.namespaces ?? [
response.items[1].create._source.namespace ?? 'default',
],
},
});

// Assert that ID's are deserialized to remove the type and namespace
expect(result.saved_objects[0].id).toEqual(
Expand Down Expand Up @@ -985,7 +1000,7 @@ describe('SavedObjectsRepository', () => {
const expectSuccessResult = ({ type, id }, doc) => ({
type,
id,
...(doc._source.namespaces && { namespaces: doc._source.namespaces }),
namespaces: doc._source.namespaces ?? ['default'],
...(doc._source.updated_at && { updated_at: doc._source.updated_at }),
version: encodeHitVersion(doc),
attributes: doc._source[type],
Expand Down Expand Up @@ -1032,7 +1047,7 @@ describe('SavedObjectsRepository', () => {
const result = await bulkGetSuccess([obj1, obj]);
expect(result).toEqual({
saved_objects: [
expect.not.objectContaining({ namespaces: expect.anything() }),
expect.objectContaining({ namespaces: ['default'] }),
expect.objectContaining({ namespaces: expect.any(Array) }),
],
});
Expand Down Expand Up @@ -1651,6 +1666,7 @@ describe('SavedObjectsRepository', () => {
version: mockVersion,
attributes,
references,
namespaces: [namespace ?? 'default'],
migrationVersion: { [type]: '1.1.1' },
});
});
Expand Down Expand Up @@ -1907,7 +1923,7 @@ describe('SavedObjectsRepository', () => {
await deleteByNamespaceSuccess(namespace);
const allTypes = registry.getAllTypes().map((type) => type.name);
expect(getSearchDslNS.getSearchDsl).toHaveBeenCalledWith(mappings, registry, {
namespace,
namespaces: [namespace],
type: allTypes.filter((type) => !registry.isNamespaceAgnostic(type)),
});
});
Expand Down Expand Up @@ -2128,6 +2144,7 @@ describe('SavedObjectsRepository', () => {
version: mockVersion,
attributes: doc._source[doc._source.type],
references: [],
namespaces: doc._source.type === NAMESPACE_AGNOSTIC_TYPE ? undefined : ['default'],
});
});
});
Expand All @@ -2137,7 +2154,7 @@ describe('SavedObjectsRepository', () => {
callAdminCluster.mockReturnValue(namespacedSearchResults);
const count = namespacedSearchResults.hits.hits.length;

const response = await savedObjectsRepository.find({ type, namespace });
const response = await savedObjectsRepository.find({ type, namespaces: [namespace] });

expect(response.total).toBe(count);
expect(response.saved_objects).toHaveLength(count);
Expand All @@ -2150,6 +2167,7 @@ describe('SavedObjectsRepository', () => {
version: mockVersion,
attributes: doc._source[doc._source.type],
references: [],
namespaces: doc._source.type === NAMESPACE_AGNOSTIC_TYPE ? undefined : [namespace],
});
});
});
Expand All @@ -2169,7 +2187,7 @@ describe('SavedObjectsRepository', () => {
describe('search dsl', () => {
it(`passes mappings, registry, search, defaultSearchOperator, searchFields, type, sortField, sortOrder and hasReference to getSearchDsl`, async () => {
const relevantOpts = {
namespace,
namespaces: [namespace],
search: 'foo*',
searchFields: ['foo'],
type: [type],
Expand Down Expand Up @@ -2367,6 +2385,7 @@ describe('SavedObjectsRepository', () => {
title: 'Testing',
},
references: [],
namespaces: ['default'],
});
});

Expand All @@ -2377,10 +2396,10 @@ describe('SavedObjectsRepository', () => {
});
});

it(`doesn't include namespaces if type is not multi-namespace`, async () => {
it(`include namespaces if type is not multi-namespace`, async () => {
const result = await getSuccess(type, id);
expect(result).not.toMatchObject({
namespaces: expect.anything(),
expect(result).toMatchObject({
namespaces: ['default'],
});
});
});
Expand Down Expand Up @@ -2901,10 +2920,10 @@ describe('SavedObjectsRepository', () => {
_id: `${type}:${id}`,
...mockVersionProps,
result: 'updated',
...(registry.isMultiNamespace(type) && {
// don't need the rest of the source for test purposes, just the namespaces attribute
get: { _source: { namespaces: [options?.namespace ?? 'default'] } },
}),
// don't need the rest of the source for test purposes, just the namespace and namespaces attributes
get: {
_source: { namespaces: [options?.namespace ?? 'default'], namespace: options?.namespace },
},
}); // this._writeToCluster('update', ...)
const result = await savedObjectsRepository.update(type, id, attributes, options);
expect(callAdminCluster).toHaveBeenCalledTimes(registry.isMultiNamespace(type) ? 2 : 1);
Expand Down Expand Up @@ -3004,15 +3023,15 @@ describe('SavedObjectsRepository', () => {

it(`includes _sourceIncludes when type is multi-namespace`, async () => {
await updateSuccess(MULTI_NAMESPACE_TYPE, id, attributes);
expectClusterCallArgs({ _sourceIncludes: ['namespaces'] }, 2);
expectClusterCallArgs({ _sourceIncludes: ['namespace', 'namespaces'] }, 2);
});

it(`doesn't include _sourceIncludes when type is not multi-namespace`, async () => {
it(`includes _sourceIncludes when type is not multi-namespace`, async () => {
await updateSuccess(type, id, attributes);
expect(callAdminCluster).toHaveBeenLastCalledWith(
expect.any(String),
expect.not.objectContaining({
_sourceIncludes: expect.anything(),
expect.objectContaining({
_sourceIncludes: ['namespace', 'namespaces'],
})
);
});
Expand Down Expand Up @@ -3086,6 +3105,7 @@ describe('SavedObjectsRepository', () => {
version: mockVersion,
attributes,
references,
namespaces: [namespace],
});
});

Expand All @@ -3096,10 +3116,10 @@ describe('SavedObjectsRepository', () => {
});
});

it(`doesn't include namespaces if type is not multi-namespace`, async () => {
it(`includes namespaces if type is not multi-namespace`, async () => {
const result = await updateSuccess(type, id, attributes);
expect(result).not.toMatchObject({
namespaces: expect.anything(),
expect(result).toMatchObject({
namespaces: ['default'],
});
});
});
Expand Down
Loading