mirror of
https://github.com/elastic/kibana.git
synced 2025-04-24 09:48:58 -04:00
Add multiple namespaces support to PIT search and finder (#109062)
* initial modifications * change approach for openPointInTime and add tests for spaces wrapper changes * fix and add security wrapper tests * fix export security FTR tests * update generated doc * add tests for PIT finder * NIT * improve doc * nits
This commit is contained in:
parent
75cdeae490
commit
5c5e191364
13 changed files with 385 additions and 186 deletions
|
@ -8,7 +8,7 @@
|
|||
<b>Signature:</b>
|
||||
|
||||
```typescript
|
||||
export interface SavedObjectsOpenPointInTimeOptions extends SavedObjectsBaseOptions
|
||||
export interface SavedObjectsOpenPointInTimeOptions
|
||||
```
|
||||
|
||||
## Properties
|
||||
|
@ -16,5 +16,6 @@ export interface SavedObjectsOpenPointInTimeOptions extends SavedObjectsBaseOpti
|
|||
| Property | Type | Description |
|
||||
| --- | --- | --- |
|
||||
| [keepAlive](./kibana-plugin-core-server.savedobjectsopenpointintimeoptions.keepalive.md) | <code>string</code> | Optionally specify how long ES should keep the PIT alive until the next request. Defaults to <code>5m</code>. |
|
||||
| [namespaces](./kibana-plugin-core-server.savedobjectsopenpointintimeoptions.namespaces.md) | <code>string[]</code> | An optional list of namespaces to be used when opening the PIT.<!-- -->When the spaces plugin is enabled: - this will default to the user's current space (as determined by the URL) - if specified, the user's current space will be ignored - <code>['*']</code> will search across all available spaces |
|
||||
| [preference](./kibana-plugin-core-server.savedobjectsopenpointintimeoptions.preference.md) | <code>string</code> | An optional ES preference value to be used for the query. |
|
||||
|
||||
|
|
|
@ -0,0 +1,15 @@
|
|||
<!-- Do not edit this file. It is automatically generated by API Documenter. -->
|
||||
|
||||
[Home](./index.md) > [kibana-plugin-core-server](./kibana-plugin-core-server.md) > [SavedObjectsOpenPointInTimeOptions](./kibana-plugin-core-server.savedobjectsopenpointintimeoptions.md) > [namespaces](./kibana-plugin-core-server.savedobjectsopenpointintimeoptions.namespaces.md)
|
||||
|
||||
## SavedObjectsOpenPointInTimeOptions.namespaces property
|
||||
|
||||
An optional list of namespaces to be used when opening the PIT.
|
||||
|
||||
When the spaces plugin is enabled: - this will default to the user's current space (as determined by the URL) - if specified, the user's current space will be ignored - `['*']` will search across all available spaces
|
||||
|
||||
<b>Signature:</b>
|
||||
|
||||
```typescript
|
||||
namespaces?: string[];
|
||||
```
|
|
@ -7,7 +7,6 @@
|
|||
*/
|
||||
|
||||
import { loggerMock, MockedLogger } from '../../../logging/logger.mock';
|
||||
import type { SavedObjectsClientContract } from '../../types';
|
||||
import type { SavedObjectsFindResult } from '../';
|
||||
import { savedObjectsRepositoryMock } from './repository.mock';
|
||||
|
||||
|
@ -43,38 +42,68 @@ const mockHits = [
|
|||
|
||||
describe('createPointInTimeFinder()', () => {
|
||||
let logger: MockedLogger;
|
||||
let find: jest.Mocked<SavedObjectsClientContract>['find'];
|
||||
let openPointInTimeForType: jest.Mocked<SavedObjectsClientContract>['openPointInTimeForType'];
|
||||
let closePointInTime: jest.Mocked<SavedObjectsClientContract>['closePointInTime'];
|
||||
let repository: ReturnType<typeof savedObjectsRepositoryMock.create>;
|
||||
|
||||
beforeEach(() => {
|
||||
logger = loggerMock.create();
|
||||
const mockRepository = savedObjectsRepositoryMock.create();
|
||||
find = mockRepository.find;
|
||||
openPointInTimeForType = mockRepository.openPointInTimeForType;
|
||||
closePointInTime = mockRepository.closePointInTime;
|
||||
repository = savedObjectsRepositoryMock.create();
|
||||
});
|
||||
|
||||
describe('#find', () => {
|
||||
test('throws if a PIT is already open', async () => {
|
||||
openPointInTimeForType.mockResolvedValueOnce({
|
||||
test('opens a PIT with the correct parameters', async () => {
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'abc123',
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
repository.find.mockResolvedValue({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 1,
|
||||
|
||||
const findOptions: SavedObjectsCreatePointInTimeFinderOptions = {
|
||||
type: ['visualization'],
|
||||
search: 'foo*',
|
||||
perPage: 1,
|
||||
namespaces: ['ns1', 'ns2'],
|
||||
};
|
||||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: repository,
|
||||
});
|
||||
|
||||
expect(repository.openPointInTimeForType).not.toHaveBeenCalled();
|
||||
|
||||
await finder.find().next();
|
||||
|
||||
expect(repository.openPointInTimeForType).toHaveBeenCalledTimes(1);
|
||||
expect(repository.openPointInTimeForType).toHaveBeenCalledWith(findOptions.type, {
|
||||
namespaces: findOptions.namespaces,
|
||||
});
|
||||
});
|
||||
|
||||
test('throws if a PIT is already open', async () => {
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'abc123',
|
||||
});
|
||||
repository.find
|
||||
.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
})
|
||||
.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 1,
|
||||
});
|
||||
|
||||
const findOptions: SavedObjectsCreatePointInTimeFinderOptions = {
|
||||
type: ['visualization'],
|
||||
search: 'foo*',
|
||||
|
@ -83,30 +112,25 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: {
|
||||
find,
|
||||
openPointInTimeForType,
|
||||
closePointInTime,
|
||||
},
|
||||
client: repository,
|
||||
});
|
||||
await finder.find().next();
|
||||
|
||||
expect(find).toHaveBeenCalledTimes(1);
|
||||
find.mockClear();
|
||||
expect(repository.find).toHaveBeenCalledTimes(1);
|
||||
|
||||
expect(async () => {
|
||||
await finder.find().next();
|
||||
}).rejects.toThrowErrorMatchingInlineSnapshot(
|
||||
`"Point In Time has already been opened for this finder instance. Please call \`close()\` before calling \`find()\` again."`
|
||||
);
|
||||
expect(find).toHaveBeenCalledTimes(0);
|
||||
expect(repository.find).toHaveBeenCalledTimes(1);
|
||||
});
|
||||
|
||||
test('works with a single page of results', async () => {
|
||||
openPointInTimeForType.mockResolvedValueOnce({
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'abc123',
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
repository.find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
|
@ -121,11 +145,7 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: {
|
||||
find,
|
||||
openPointInTimeForType,
|
||||
closePointInTime,
|
||||
},
|
||||
client: repository,
|
||||
});
|
||||
const hits: SavedObjectsFindResult[] = [];
|
||||
for await (const result of finder.find()) {
|
||||
|
@ -133,10 +153,10 @@ describe('createPointInTimeFinder()', () => {
|
|||
}
|
||||
|
||||
expect(hits.length).toBe(2);
|
||||
expect(openPointInTimeForType).toHaveBeenCalledTimes(1);
|
||||
expect(closePointInTime).toHaveBeenCalledTimes(1);
|
||||
expect(find).toHaveBeenCalledTimes(1);
|
||||
expect(find).toHaveBeenCalledWith(
|
||||
expect(repository.openPointInTimeForType).toHaveBeenCalledTimes(1);
|
||||
expect(repository.closePointInTime).toHaveBeenCalledTimes(1);
|
||||
expect(repository.find).toHaveBeenCalledTimes(1);
|
||||
expect(repository.find).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
pit: expect.objectContaining({ id: 'abc123', keepAlive: '2m' }),
|
||||
sortField: 'updated_at',
|
||||
|
@ -147,24 +167,25 @@ describe('createPointInTimeFinder()', () => {
|
|||
});
|
||||
|
||||
test('works with multiple pages of results', async () => {
|
||||
openPointInTimeForType.mockResolvedValueOnce({
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'abc123',
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [mockHits[0]],
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [mockHits[1]],
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
repository.find
|
||||
.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [mockHits[0]],
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
})
|
||||
.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [mockHits[1]],
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
});
|
||||
repository.find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [],
|
||||
per_page: 1,
|
||||
|
@ -180,11 +201,7 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: {
|
||||
find,
|
||||
openPointInTimeForType,
|
||||
closePointInTime,
|
||||
},
|
||||
client: repository,
|
||||
});
|
||||
const hits: SavedObjectsFindResult[] = [];
|
||||
for await (const result of finder.find()) {
|
||||
|
@ -192,12 +209,12 @@ describe('createPointInTimeFinder()', () => {
|
|||
}
|
||||
|
||||
expect(hits.length).toBe(2);
|
||||
expect(openPointInTimeForType).toHaveBeenCalledTimes(1);
|
||||
expect(closePointInTime).toHaveBeenCalledTimes(1);
|
||||
expect(repository.openPointInTimeForType).toHaveBeenCalledTimes(1);
|
||||
expect(repository.closePointInTime).toHaveBeenCalledTimes(1);
|
||||
// called 3 times since we need a 3rd request to check if we
|
||||
// are done paginating through results.
|
||||
expect(find).toHaveBeenCalledTimes(3);
|
||||
expect(find).toHaveBeenCalledWith(
|
||||
expect(repository.find).toHaveBeenCalledTimes(3);
|
||||
expect(repository.find).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
pit: expect.objectContaining({ id: 'abc123', keepAlive: '2m' }),
|
||||
sortField: 'updated_at',
|
||||
|
@ -210,10 +227,10 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
describe('#close', () => {
|
||||
test('calls closePointInTime with correct ID', async () => {
|
||||
openPointInTimeForType.mockResolvedValueOnce({
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'test',
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
repository.find.mockResolvedValueOnce({
|
||||
total: 1,
|
||||
saved_objects: [mockHits[0]],
|
||||
pit_id: 'test',
|
||||
|
@ -229,11 +246,7 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: {
|
||||
find,
|
||||
openPointInTimeForType,
|
||||
closePointInTime,
|
||||
},
|
||||
client: repository,
|
||||
});
|
||||
const hits: SavedObjectsFindResult[] = [];
|
||||
for await (const result of finder.find()) {
|
||||
|
@ -241,28 +254,28 @@ describe('createPointInTimeFinder()', () => {
|
|||
await finder.close();
|
||||
}
|
||||
|
||||
expect(closePointInTime).toHaveBeenCalledWith('test');
|
||||
expect(repository.closePointInTime).toHaveBeenCalledWith('test');
|
||||
});
|
||||
|
||||
test('causes generator to stop', async () => {
|
||||
openPointInTimeForType.mockResolvedValueOnce({
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'test',
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
repository.find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [mockHits[0]],
|
||||
pit_id: 'test',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
repository.find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [mockHits[1]],
|
||||
pit_id: 'test',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
repository.find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: [],
|
||||
per_page: 1,
|
||||
|
@ -278,11 +291,7 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: {
|
||||
find,
|
||||
openPointInTimeForType,
|
||||
closePointInTime,
|
||||
},
|
||||
client: repository,
|
||||
});
|
||||
const hits: SavedObjectsFindResult[] = [];
|
||||
for await (const result of finder.find()) {
|
||||
|
@ -290,15 +299,15 @@ describe('createPointInTimeFinder()', () => {
|
|||
await finder.close();
|
||||
}
|
||||
|
||||
expect(closePointInTime).toHaveBeenCalledTimes(1);
|
||||
expect(repository.closePointInTime).toHaveBeenCalledTimes(1);
|
||||
expect(hits.length).toBe(1);
|
||||
});
|
||||
|
||||
test('is called if `find` throws an error', async () => {
|
||||
openPointInTimeForType.mockResolvedValueOnce({
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'test',
|
||||
});
|
||||
find.mockRejectedValueOnce(new Error('oops'));
|
||||
repository.find.mockRejectedValueOnce(new Error('oops'));
|
||||
|
||||
const findOptions: SavedObjectsCreatePointInTimeFinderOptions = {
|
||||
type: ['visualization'],
|
||||
|
@ -308,11 +317,7 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: {
|
||||
find,
|
||||
openPointInTimeForType,
|
||||
closePointInTime,
|
||||
},
|
||||
client: repository,
|
||||
});
|
||||
const hits: SavedObjectsFindResult[] = [];
|
||||
try {
|
||||
|
@ -323,27 +328,28 @@ describe('createPointInTimeFinder()', () => {
|
|||
// intentionally empty
|
||||
}
|
||||
|
||||
expect(closePointInTime).toHaveBeenCalledWith('test');
|
||||
expect(repository.closePointInTime).toHaveBeenCalledWith('test');
|
||||
});
|
||||
|
||||
test('finder can be reused after closing', async () => {
|
||||
openPointInTimeForType.mockResolvedValueOnce({
|
||||
repository.openPointInTimeForType.mockResolvedValueOnce({
|
||||
id: 'abc123',
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
});
|
||||
find.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 1,
|
||||
});
|
||||
repository.find
|
||||
.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 0,
|
||||
})
|
||||
.mockResolvedValueOnce({
|
||||
total: 2,
|
||||
saved_objects: mockHits,
|
||||
pit_id: 'abc123',
|
||||
per_page: 1,
|
||||
page: 1,
|
||||
});
|
||||
|
||||
const findOptions: SavedObjectsCreatePointInTimeFinderOptions = {
|
||||
type: ['visualization'],
|
||||
|
@ -353,11 +359,7 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
const finder = new PointInTimeFinder(findOptions, {
|
||||
logger,
|
||||
client: {
|
||||
find,
|
||||
openPointInTimeForType,
|
||||
closePointInTime,
|
||||
},
|
||||
client: repository,
|
||||
});
|
||||
|
||||
const findA = finder.find();
|
||||
|
@ -370,9 +372,9 @@ describe('createPointInTimeFinder()', () => {
|
|||
|
||||
expect((await findA.next()).done).toBe(true);
|
||||
expect((await findB.next()).done).toBe(true);
|
||||
expect(openPointInTimeForType).toHaveBeenCalledTimes(2);
|
||||
expect(find).toHaveBeenCalledTimes(2);
|
||||
expect(closePointInTime).toHaveBeenCalledTimes(2);
|
||||
expect(repository.openPointInTimeForType).toHaveBeenCalledTimes(2);
|
||||
expect(repository.find).toHaveBeenCalledTimes(2);
|
||||
expect(repository.closePointInTime).toHaveBeenCalledTimes(2);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
|
|
@ -139,7 +139,9 @@ export class PointInTimeFinder<T = unknown, A = unknown>
|
|||
|
||||
private async open() {
|
||||
try {
|
||||
const { id } = await this.#client.openPointInTimeForType(this.#findOptions.type);
|
||||
const { id } = await this.#client.openPointInTimeForType(this.#findOptions.type, {
|
||||
namespaces: this.#findOptions.namespaces,
|
||||
});
|
||||
this.#pitId = id;
|
||||
this.#open = true;
|
||||
} catch (e) {
|
||||
|
|
|
@ -334,7 +334,7 @@ export interface SavedObjectsResolveResponse<T = unknown> {
|
|||
/**
|
||||
* @public
|
||||
*/
|
||||
export interface SavedObjectsOpenPointInTimeOptions extends SavedObjectsBaseOptions {
|
||||
export interface SavedObjectsOpenPointInTimeOptions {
|
||||
/**
|
||||
* Optionally specify how long ES should keep the PIT alive until the next request. Defaults to `5m`.
|
||||
*/
|
||||
|
@ -343,6 +343,15 @@ export interface SavedObjectsOpenPointInTimeOptions extends SavedObjectsBaseOpti
|
|||
* An optional ES preference value to be used for the query.
|
||||
*/
|
||||
preference?: string;
|
||||
/**
|
||||
* An optional list of namespaces to be used when opening the PIT.
|
||||
*
|
||||
* When the spaces plugin is enabled:
|
||||
* - this will default to the user's current space (as determined by the URL)
|
||||
* - if specified, the user's current space will be ignored
|
||||
* - `['*']` will search across all available spaces
|
||||
*/
|
||||
namespaces?: string[];
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -2460,8 +2460,9 @@ export interface SavedObjectsMigrationVersion {
|
|||
export type SavedObjectsNamespaceType = 'single' | 'multiple' | 'multiple-isolated' | 'agnostic';
|
||||
|
||||
// @public (undocumented)
|
||||
export interface SavedObjectsOpenPointInTimeOptions extends SavedObjectsBaseOptions {
|
||||
export interface SavedObjectsOpenPointInTimeOptions {
|
||||
keepAlive?: string;
|
||||
namespaces?: string[];
|
||||
preference?: string;
|
||||
}
|
||||
|
||||
|
|
|
@ -779,17 +779,6 @@ describe('#find', () => {
|
|||
);
|
||||
});
|
||||
|
||||
test(`throws BadRequestError when searching across namespaces when pit is provided`, async () => {
|
||||
const options = {
|
||||
type: [type1, type2],
|
||||
pit: { id: 'abc123' },
|
||||
namespaces: ['some-ns', 'another-ns'],
|
||||
};
|
||||
await expect(client.find(options)).rejects.toThrowErrorMatchingInlineSnapshot(
|
||||
`"_find across namespaces is not permitted when using the \`pit\` option."`
|
||||
);
|
||||
});
|
||||
|
||||
test(`checks privileges for user, actions, and namespaces`, async () => {
|
||||
const options = { type: [type1, type2], namespaces };
|
||||
await expectPrivilegeCheck(client.find, { options }, namespaces);
|
||||
|
@ -884,7 +873,7 @@ describe('#openPointInTimeForType', () => {
|
|||
const apiCallReturnValue = Symbol();
|
||||
clientOpts.baseClient.openPointInTimeForType.mockReturnValue(apiCallReturnValue as any);
|
||||
|
||||
const options = { namespace };
|
||||
const options = { namespaces: [namespace] };
|
||||
const result = await expectSuccess(client.openPointInTimeForType, { type, options });
|
||||
expect(result).toBe(apiCallReturnValue);
|
||||
});
|
||||
|
@ -892,18 +881,113 @@ describe('#openPointInTimeForType', () => {
|
|||
test(`adds audit event when successful`, async () => {
|
||||
const apiCallReturnValue = Symbol();
|
||||
clientOpts.baseClient.openPointInTimeForType.mockReturnValue(apiCallReturnValue as any);
|
||||
const options = { namespace };
|
||||
const options = { namespaces: [namespace] };
|
||||
await expectSuccess(client.openPointInTimeForType, { type, options });
|
||||
expect(clientOpts.auditLogger.log).toHaveBeenCalledTimes(1);
|
||||
expectAuditEvent('saved_object_open_point_in_time', 'unknown');
|
||||
});
|
||||
|
||||
test(`adds audit event when not successful`, async () => {
|
||||
clientOpts.checkSavedObjectsPrivilegesAsCurrentUser.mockRejectedValue(new Error());
|
||||
await expect(() => client.openPointInTimeForType(type, { namespace })).rejects.toThrow();
|
||||
test(`throws an error when unauthorized`, async () => {
|
||||
clientOpts.checkSavedObjectsPrivilegesAsCurrentUser.mockImplementation(
|
||||
getMockCheckPrivilegesFailure
|
||||
);
|
||||
const options = { namespaces: [namespace] };
|
||||
await expect(() => client.openPointInTimeForType(type, options)).rejects.toThrowError(
|
||||
'unauthorized'
|
||||
);
|
||||
});
|
||||
|
||||
test(`adds audit event when unauthorized`, async () => {
|
||||
clientOpts.checkSavedObjectsPrivilegesAsCurrentUser.mockImplementation(
|
||||
getMockCheckPrivilegesFailure
|
||||
);
|
||||
const options = { namespaces: [namespace] };
|
||||
await expect(() => client.openPointInTimeForType(type, options)).rejects.toThrow();
|
||||
expect(clientOpts.auditLogger.log).toHaveBeenCalledTimes(1);
|
||||
expectAuditEvent('saved_object_open_point_in_time', 'failure');
|
||||
});
|
||||
|
||||
test(`filters types based on authorization`, async () => {
|
||||
clientOpts.checkSavedObjectsPrivilegesAsCurrentUser.mockResolvedValue({
|
||||
hasAllRequested: false,
|
||||
username: USERNAME,
|
||||
privileges: {
|
||||
kibana: [
|
||||
{
|
||||
resource: 'some-ns',
|
||||
privilege: 'mock-saved_object:foo/open_point_in_time',
|
||||
authorized: true,
|
||||
},
|
||||
{
|
||||
resource: 'some-ns',
|
||||
privilege: 'mock-saved_object:bar/open_point_in_time',
|
||||
authorized: true,
|
||||
},
|
||||
{
|
||||
resource: 'some-ns',
|
||||
privilege: 'mock-saved_object:baz/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
{
|
||||
resource: 'some-ns',
|
||||
privilege: 'mock-saved_object:qux/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
{
|
||||
resource: 'another-ns',
|
||||
privilege: 'mock-saved_object:foo/open_point_in_time',
|
||||
authorized: true,
|
||||
},
|
||||
{
|
||||
resource: 'another-ns',
|
||||
privilege: 'mock-saved_object:bar/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
{
|
||||
resource: 'another-ns',
|
||||
privilege: 'mock-saved_object:baz/open_point_in_time',
|
||||
authorized: true,
|
||||
},
|
||||
{
|
||||
resource: 'another-ns',
|
||||
privilege: 'mock-saved_object:qux/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
{
|
||||
resource: 'forbidden-ns',
|
||||
privilege: 'mock-saved_object:foo/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
{
|
||||
resource: 'forbidden-ns',
|
||||
privilege: 'mock-saved_object:bar/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
{
|
||||
resource: 'forbidden-ns',
|
||||
privilege: 'mock-saved_object:baz/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
{
|
||||
resource: 'forbidden-ns',
|
||||
privilege: 'mock-saved_object:qux/open_point_in_time',
|
||||
authorized: false,
|
||||
},
|
||||
],
|
||||
},
|
||||
});
|
||||
|
||||
await client.openPointInTimeForType(['foo', 'bar', 'baz', 'qux'], {
|
||||
namespaces: ['some-ns', 'another-ns', 'forbidden-ns'],
|
||||
});
|
||||
|
||||
expect(clientOpts.baseClient.openPointInTimeForType).toHaveBeenCalledWith(
|
||||
['foo', 'bar', 'baz'],
|
||||
{
|
||||
namespaces: ['some-ns', 'another-ns', 'forbidden-ns'],
|
||||
}
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
describe('#closePointInTime', () => {
|
||||
|
|
|
@ -29,7 +29,7 @@ import type {
|
|||
SavedObjectsUpdateOptions,
|
||||
} from 'src/core/server';
|
||||
|
||||
import { SavedObjectsUtils } from '../../../../../src/core/server';
|
||||
import { SavedObjectsErrorHelpers, SavedObjectsUtils } from '../../../../../src/core/server';
|
||||
import { ALL_SPACES_ID, UNKNOWN_SPACE } from '../../common/constants';
|
||||
import type { AuditLogger, SecurityAuditLogger } from '../audit';
|
||||
import { SavedObjectAction, savedObjectEvent } from '../audit';
|
||||
|
@ -75,10 +75,12 @@ interface LegacyEnsureAuthorizedResult {
|
|||
status: 'fully_authorized' | 'partially_authorized' | 'unauthorized';
|
||||
typeMap: Map<string, LegacyEnsureAuthorizedTypeResult>;
|
||||
}
|
||||
|
||||
interface LegacyEnsureAuthorizedTypeResult {
|
||||
authorizedSpaces: string[];
|
||||
isGloballyAuthorized?: boolean;
|
||||
}
|
||||
|
||||
export class SecureSavedObjectsClientWrapper implements SavedObjectsClientContract {
|
||||
private readonly actions: Actions;
|
||||
private readonly legacyAuditLogger: PublicMethodsOf<SecurityAuditLogger>;
|
||||
|
@ -236,11 +238,6 @@ export class SecureSavedObjectsClientWrapper implements SavedObjectsClientContra
|
|||
`_find across namespaces is not permitted when the Spaces plugin is disabled.`
|
||||
);
|
||||
}
|
||||
if (options.pit && Array.isArray(options.namespaces) && options.namespaces.length > 1) {
|
||||
throw this.errors.createBadRequestError(
|
||||
'_find across namespaces is not permitted when using the `pit` option.'
|
||||
);
|
||||
}
|
||||
|
||||
const args = { options };
|
||||
const { status, typeMap } = await this.legacyEnsureAuthorized(
|
||||
|
@ -508,22 +505,27 @@ export class SecureSavedObjectsClientWrapper implements SavedObjectsClientContra
|
|||
type: string | string[],
|
||||
options: SavedObjectsOpenPointInTimeOptions
|
||||
) {
|
||||
try {
|
||||
const args = { type, options };
|
||||
await this.legacyEnsureAuthorized(type, 'open_point_in_time', options?.namespace, {
|
||||
const args = { type, options };
|
||||
const { status, typeMap } = await this.legacyEnsureAuthorized(
|
||||
type,
|
||||
'open_point_in_time',
|
||||
options?.namespaces,
|
||||
{
|
||||
args,
|
||||
// Partial authorization is acceptable in this case because this method is only designed
|
||||
// to be used with `find`, which already allows for partial authorization.
|
||||
requireFullAuthorization: false,
|
||||
});
|
||||
} catch (error) {
|
||||
}
|
||||
);
|
||||
|
||||
if (status === 'unauthorized') {
|
||||
this.auditLogger.log(
|
||||
savedObjectEvent({
|
||||
action: SavedObjectAction.OPEN_POINT_IN_TIME,
|
||||
error,
|
||||
error: new Error(status),
|
||||
})
|
||||
);
|
||||
throw error;
|
||||
throw SavedObjectsErrorHelpers.decorateForbiddenError(new Error(status));
|
||||
}
|
||||
|
||||
this.auditLogger.log(
|
||||
|
@ -533,7 +535,8 @@ export class SecureSavedObjectsClientWrapper implements SavedObjectsClientContra
|
|||
})
|
||||
);
|
||||
|
||||
return await this.baseClient.openPointInTimeForType(type, options);
|
||||
const allowedTypes = [...typeMap.keys()]; // only allow the user to open a PIT against indices for type(s) they are authorized to access
|
||||
return await this.baseClient.openPointInTimeForType(allowedTypes, options);
|
||||
}
|
||||
|
||||
public async closePointInTime(id: string, options?: SavedObjectsClosePointInTimeOptions) {
|
||||
|
|
|
@ -533,27 +533,94 @@ const ERROR_NAMESPACE_SPECIFIED = 'Spaces currently determines the namespaces';
|
|||
});
|
||||
|
||||
describe('#openPointInTimeForType', () => {
|
||||
test(`throws error if options.namespace is specified`, async () => {
|
||||
const { client } = createSpacesSavedObjectsClient();
|
||||
test(`throws error if if user is unauthorized in this space`, async () => {
|
||||
const { client, baseClient, spacesService } = createSpacesSavedObjectsClient();
|
||||
const spacesClient = spacesClientMock.create();
|
||||
spacesClient.getAll.mockResolvedValue([]);
|
||||
spacesService.createSpacesClient.mockReturnValue(spacesClient);
|
||||
|
||||
await expect(client.openPointInTimeForType('foo', { namespace: 'bar' })).rejects.toThrow(
|
||||
ERROR_NAMESPACE_SPECIFIED
|
||||
);
|
||||
await expect(
|
||||
client.openPointInTimeForType('foo', { namespaces: ['bar'] })
|
||||
).rejects.toThrowError('Bad Request');
|
||||
|
||||
expect(baseClient.openPointInTimeForType).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
test(`supplements options with the current namespace`, async () => {
|
||||
test(`throws error if if user is unauthorized in any space`, async () => {
|
||||
const { client, baseClient, spacesService } = createSpacesSavedObjectsClient();
|
||||
const spacesClient = spacesClientMock.create();
|
||||
spacesClient.getAll.mockRejectedValue(Boom.unauthorized());
|
||||
spacesService.createSpacesClient.mockReturnValue(spacesClient);
|
||||
|
||||
await expect(
|
||||
client.openPointInTimeForType('foo', { namespaces: ['bar'] })
|
||||
).rejects.toThrowError('Bad Request');
|
||||
|
||||
expect(baseClient.openPointInTimeForType).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
test(`filters options.namespaces based on authorization`, async () => {
|
||||
const { client, baseClient, spacesService } = createSpacesSavedObjectsClient();
|
||||
const expectedReturnValue = { id: 'abc123' };
|
||||
baseClient.openPointInTimeForType.mockReturnValue(Promise.resolve(expectedReturnValue));
|
||||
|
||||
const spacesClient = spacesService.createSpacesClient(
|
||||
null as any
|
||||
) as jest.Mocked<SpacesClient>;
|
||||
spacesClient.getAll.mockImplementation(() =>
|
||||
Promise.resolve([
|
||||
{ id: 'ns-1', name: '', disabledFeatures: [] },
|
||||
{ id: 'ns-2', name: '', disabledFeatures: [] },
|
||||
])
|
||||
);
|
||||
|
||||
const options = Object.freeze({ namespaces: ['ns-1', 'ns-3'] });
|
||||
const actualReturnValue = await client.openPointInTimeForType(['foo', 'bar'], options);
|
||||
|
||||
expect(actualReturnValue).toBe(expectedReturnValue);
|
||||
expect(baseClient.openPointInTimeForType).toHaveBeenCalledWith(['foo', 'bar'], {
|
||||
namespaces: ['ns-1'],
|
||||
});
|
||||
expect(spacesClient.getAll).toHaveBeenCalledWith({ purpose: 'findSavedObjects' });
|
||||
});
|
||||
|
||||
test(`translates options.namespaces: ['*']`, async () => {
|
||||
const { client, baseClient, spacesService } = createSpacesSavedObjectsClient();
|
||||
const expectedReturnValue = { id: 'abc123' };
|
||||
baseClient.openPointInTimeForType.mockReturnValue(Promise.resolve(expectedReturnValue));
|
||||
|
||||
const spacesClient = spacesService.createSpacesClient(
|
||||
null as any
|
||||
) as jest.Mocked<SpacesClient>;
|
||||
spacesClient.getAll.mockImplementation(() =>
|
||||
Promise.resolve([
|
||||
{ id: 'ns-1', name: '', disabledFeatures: [] },
|
||||
{ id: 'ns-2', name: '', disabledFeatures: [] },
|
||||
])
|
||||
);
|
||||
|
||||
const options = Object.freeze({ namespaces: ['*'] });
|
||||
const actualReturnValue = await client.openPointInTimeForType(['foo', 'bar'], options);
|
||||
|
||||
expect(actualReturnValue).toBe(expectedReturnValue);
|
||||
expect(baseClient.openPointInTimeForType).toHaveBeenCalledWith(['foo', 'bar'], {
|
||||
namespaces: ['ns-1', 'ns-2'],
|
||||
});
|
||||
expect(spacesClient.getAll).toHaveBeenCalledWith({ purpose: 'findSavedObjects' });
|
||||
});
|
||||
|
||||
test(`supplements options with the current namespace if unspecified`, async () => {
|
||||
const { client, baseClient } = createSpacesSavedObjectsClient();
|
||||
const expectedReturnValue = { id: 'abc123' };
|
||||
baseClient.openPointInTimeForType.mockReturnValue(Promise.resolve(expectedReturnValue));
|
||||
|
||||
const options = Object.freeze({ foo: 'bar' });
|
||||
// @ts-expect-error
|
||||
const options = Object.freeze({ keepAlive: '2m' });
|
||||
const actualReturnValue = await client.openPointInTimeForType('foo', options);
|
||||
|
||||
expect(actualReturnValue).toBe(expectedReturnValue);
|
||||
expect(baseClient.openPointInTimeForType).toHaveBeenCalledWith('foo', {
|
||||
foo: 'bar',
|
||||
namespace: currentSpace.expectedNamespace,
|
||||
keepAlive: '2m',
|
||||
namespaces: [currentSpace.expectedNamespace ?? DEFAULT_SPACE_ID],
|
||||
});
|
||||
});
|
||||
});
|
||||
|
|
|
@ -30,7 +30,7 @@ import type {
|
|||
SavedObjectsUpdateOptions,
|
||||
} from 'src/core/server';
|
||||
|
||||
import { SavedObjectsUtils } from '../../../../../src/core/server';
|
||||
import { SavedObjectsErrorHelpers, SavedObjectsUtils } from '../../../../../src/core/server';
|
||||
import { ALL_SPACES_ID } from '../../common/constants';
|
||||
import { spaceIdToNamespace } from '../lib/utils/namespace';
|
||||
import type { ISpacesClient } from '../spaces_client';
|
||||
|
@ -175,32 +175,19 @@ export class SpacesSavedObjectsClient implements SavedObjectsClientContract {
|
|||
* @returns {promise} - { saved_objects: [{ id, type, version, attributes }], total, per_page, page }
|
||||
*/
|
||||
public async find<T = unknown, A = unknown>(options: SavedObjectsFindOptions) {
|
||||
throwErrorIfNamespaceSpecified(options);
|
||||
|
||||
let namespaces = options.namespaces;
|
||||
if (namespaces) {
|
||||
try {
|
||||
const availableSpaces = await this.spacesClient.getAll({ purpose: 'findSavedObjects' });
|
||||
if (namespaces.includes(ALL_SPACES_ID)) {
|
||||
namespaces = availableSpaces.map((space) => space.id);
|
||||
} else {
|
||||
namespaces = namespaces.filter((namespace) =>
|
||||
availableSpaces.some((space) => space.id === namespace)
|
||||
);
|
||||
}
|
||||
if (namespaces.length === 0) {
|
||||
// return empty response, since the user is unauthorized in this space (or these spaces), but we don't return forbidden errors for `find` operations
|
||||
return SavedObjectsUtils.createEmptyFindResponse<T, A>(options);
|
||||
}
|
||||
} catch (err) {
|
||||
if (Boom.isBoom(err) && err.output.payload.statusCode === 403) {
|
||||
// return empty response, since the user is unauthorized in any space, but we don't return forbidden errors for `find` operations
|
||||
return SavedObjectsUtils.createEmptyFindResponse<T, A>(options);
|
||||
}
|
||||
throw err;
|
||||
let namespaces: string[];
|
||||
try {
|
||||
namespaces = await this.getSearchableSpaces(options.namespaces);
|
||||
} catch (err) {
|
||||
if (Boom.isBoom(err) && err.output.payload.statusCode === 403) {
|
||||
// return empty response, since the user is unauthorized in any space, but we don't return forbidden errors for `find` operations
|
||||
return SavedObjectsUtils.createEmptyFindResponse<T, A>(options);
|
||||
}
|
||||
} else {
|
||||
namespaces = [this.spaceId];
|
||||
throw err;
|
||||
}
|
||||
if (namespaces.length === 0) {
|
||||
// return empty response, since the user is unauthorized in this space (or these spaces), but we don't return forbidden errors for `find` operations
|
||||
return SavedObjectsUtils.createEmptyFindResponse<T, A>(options);
|
||||
}
|
||||
|
||||
return await this.client.find<T, A>({
|
||||
|
@ -396,10 +383,15 @@ export class SpacesSavedObjectsClient implements SavedObjectsClientContract {
|
|||
type: string | string[],
|
||||
options: SavedObjectsOpenPointInTimeOptions = {}
|
||||
) {
|
||||
throwErrorIfNamespaceSpecified(options);
|
||||
const namespaces = await this.getSearchableSpaces(options.namespaces);
|
||||
if (namespaces.length === 0) {
|
||||
// throw bad request if no valid spaces were found.
|
||||
throw SavedObjectsErrorHelpers.createBadRequestError();
|
||||
}
|
||||
|
||||
return await this.client.openPointInTimeForType(type, {
|
||||
...options,
|
||||
namespace: spaceIdToNamespace(this.spaceId),
|
||||
namespaces,
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -446,4 +438,19 @@ export class SpacesSavedObjectsClient implements SavedObjectsClientContract {
|
|||
...dependencies,
|
||||
});
|
||||
}
|
||||
|
||||
private async getSearchableSpaces(namespaces?: string[]): Promise<string[]> {
|
||||
if (namespaces) {
|
||||
const availableSpaces = await this.spacesClient.getAll({ purpose: 'findSavedObjects' });
|
||||
if (namespaces.includes(ALL_SPACES_ID)) {
|
||||
return availableSpaces.map((space) => space.id);
|
||||
} else {
|
||||
return namespaces.filter((namespace) =>
|
||||
availableSpaces.some((space) => space.id === namespace)
|
||||
);
|
||||
}
|
||||
} else {
|
||||
return [this.spaceId];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -155,8 +155,16 @@ export function exportTestSuiteFactory(esArchiver: any, supertest: SuperTest<any
|
|||
if (failure?.reason === 'unauthorized') {
|
||||
// In export only, the API uses "bulkGet" or "find" depending on the parameters it receives.
|
||||
if (failure.statusCode === 403) {
|
||||
// "bulkGet" was unauthorized, which returns a forbidden error
|
||||
await expectSavedObjectForbiddenBulkGet(type)(response);
|
||||
if (id) {
|
||||
// "bulkGet" was unauthorized, which returns a forbidden error
|
||||
await expectSavedObjectForbiddenBulkGet(type)(response);
|
||||
} else {
|
||||
expect(response.body).to.eql({
|
||||
statusCode: 403,
|
||||
error: 'Forbidden',
|
||||
message: `unauthorized`,
|
||||
});
|
||||
}
|
||||
} else if (failure.statusCode === 200) {
|
||||
// "find" was unauthorized, which returns an empty result
|
||||
expect(response.body).not.to.have.property('error');
|
||||
|
|
|
@ -52,7 +52,7 @@ export default function ({ getService }: FtrProviderContext) {
|
|||
return {
|
||||
unauthorized: [
|
||||
createTestDefinitions(exportableObjects, { statusCode: 403, reason: 'unauthorized' }),
|
||||
createTestDefinitions(exportableTypes, { statusCode: 200, reason: 'unauthorized' }), // failure with empty result
|
||||
createTestDefinitions(exportableTypes, { statusCode: 403, reason: 'unauthorized' }),
|
||||
createTestDefinitions(nonExportableObjectsAndTypes, false),
|
||||
].flat(),
|
||||
authorized: createTestDefinitions(allObjectsAndTypes, false),
|
||||
|
|
|
@ -52,7 +52,7 @@ export default function ({ getService }: FtrProviderContext) {
|
|||
return {
|
||||
unauthorized: [
|
||||
createTestDefinitions(exportableObjects, { statusCode: 403, reason: 'unauthorized' }),
|
||||
createTestDefinitions(exportableTypes, { statusCode: 200, reason: 'unauthorized' }), // failure with empty result
|
||||
createTestDefinitions(exportableTypes, { statusCode: 403, reason: 'unauthorized' }), // failure with empty result
|
||||
createTestDefinitions(nonExportableObjectsAndTypes, false),
|
||||
].flat(),
|
||||
authorized: createTestDefinitions(allObjectsAndTypes, false),
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue