diff --git a/src/core/server/core_route_handler_context.ts b/src/core/server/core_route_handler_context.ts index 520c5bd3f685b..e037bf5075aa1 100644 --- a/src/core/server/core_route_handler_context.ts +++ b/src/core/server/core_route_handler_context.ts @@ -21,7 +21,11 @@ import { InternalCoreStart } from './internal_types'; import { KibanaRequest } from './http/router'; import { SavedObjectsClientContract } from './saved_objects/types'; -import { InternalSavedObjectsServiceStart, ISavedObjectTypeRegistry } from './saved_objects'; +import { + InternalSavedObjectsServiceStart, + ISavedObjectTypeRegistry, + SavedObjectsClientProviderOptions, +} from './saved_objects'; import { InternalElasticsearchServiceStart, IScopedClusterClient, @@ -78,6 +82,10 @@ class CoreSavedObjectsRouteHandlerContext { } return this.#typeRegistry; } + + public getClient(options?: SavedObjectsClientProviderOptions) { + return this.savedObjectsStart.getScopedClient(this.request, options); + } } class CoreUiSettingsRouteHandlerContext { diff --git a/src/core/server/index.ts b/src/core/server/index.ts index 9e654ea1e2303..ecafc8a831004 100644 --- a/src/core/server/index.ts +++ b/src/core/server/index.ts @@ -56,6 +56,7 @@ import { IUiSettingsClient, UiSettingsServiceSetup, UiSettingsServiceStart } fro import { SavedObjectsClientContract } from './saved_objects/types'; import { ISavedObjectTypeRegistry, + SavedObjectsClientProviderOptions, SavedObjectsServiceSetup, SavedObjectsServiceStart, } from './saved_objects'; @@ -390,6 +391,7 @@ export interface RequestHandlerContext { savedObjects: { client: SavedObjectsClientContract; typeRegistry: ISavedObjectTypeRegistry; + getClient(options?: SavedObjectsClientProviderOptions): SavedObjectsClientContract; }; elasticsearch: { client: IScopedClusterClient; diff --git a/src/core/server/saved_objects/routes/export.ts b/src/core/server/saved_objects/routes/export.ts index 05a91f4aa4c2c..500421483a7ef 100644 --- a/src/core/server/saved_objects/routes/export.ts +++ b/src/core/server/saved_objects/routes/export.ts @@ -59,7 +59,6 @@ export const registerExportRoute = (router: IRouter, config: SavedObjectConfig) }, }, router.handleLegacyErrors(async (context, req, res) => { - const savedObjectsClient = context.core.savedObjects.client; const { type, hasReference, @@ -74,6 +73,13 @@ export const registerExportRoute = (router: IRouter, config: SavedObjectConfig) const supportedTypes = context.core.savedObjects.typeRegistry .getImportableAndExportableTypes() .map((t) => t.name); + + const includedHiddenTypes = supportedTypes.filter((supportedType) => + context.core.savedObjects.typeRegistry.isHidden(supportedType) + ); + + const savedObjectsClient = context.core.savedObjects.getClient({ includedHiddenTypes }); + if (types) { const validationError = validateTypes(types, supportedTypes); if (validationError) { diff --git a/src/core/server/saved_objects/routes/import.ts b/src/core/server/saved_objects/routes/import.ts index 291da5a5f0183..249eff0749bf0 100644 --- a/src/core/server/saved_objects/routes/import.ts +++ b/src/core/server/saved_objects/routes/import.ts @@ -80,8 +80,18 @@ export const registerImportRoute = (router: IRouter, config: SavedObjectConfig) }); } + const supportedTypes = context.core.savedObjects.typeRegistry + .getImportableAndExportableTypes() + .map((t) => t.name); + + const includedHiddenTypes = supportedTypes.filter((supportedType) => + context.core.savedObjects.typeRegistry.isHidden(supportedType) + ); + + const savedObjectsClient = context.core.savedObjects.getClient({ includedHiddenTypes }); + const result = await importSavedObjectsFromStream({ - savedObjectsClient: context.core.savedObjects.client, + savedObjectsClient, typeRegistry: context.core.savedObjects.typeRegistry, readStream, objectLimit: maxImportExportSize, diff --git a/src/core/server/saved_objects/routes/resolve_import_errors.ts b/src/core/server/saved_objects/routes/resolve_import_errors.ts index 03b4322b27cbc..18af0a446c28a 100644 --- a/src/core/server/saved_objects/routes/resolve_import_errors.ts +++ b/src/core/server/saved_objects/routes/resolve_import_errors.ts @@ -87,9 +87,19 @@ export const registerResolveImportErrorsRoute = (router: IRouter, config: SavedO }); } + const supportedTypes = context.core.savedObjects.typeRegistry + .getImportableAndExportableTypes() + .map((t) => t.name); + + const includedHiddenTypes = supportedTypes.filter((supportedType) => + context.core.savedObjects.typeRegistry.isHidden(supportedType) + ); + + const savedObjectsClient = context.core.savedObjects.getClient({ includedHiddenTypes }); + const result = await resolveSavedObjectsImportErrors({ typeRegistry: context.core.savedObjects.typeRegistry, - savedObjectsClient: context.core.savedObjects.client, + savedObjectsClient, readStream, retries: req.body.retries, objectLimit: maxImportExportSize, diff --git a/src/plugins/saved_objects_management/server/plugin.ts b/src/plugins/saved_objects_management/server/plugin.ts index 4e39b08f5c62c..0b1c860ce9113 100644 --- a/src/plugins/saved_objects_management/server/plugin.ts +++ b/src/plugins/saved_objects_management/server/plugin.ts @@ -34,9 +34,11 @@ export class SavedObjectsManagementPlugin this.logger = this.context.logger.get(); } - public async setup({ http, capabilities }: CoreSetup) { + public async setup({ http, capabilities, getStartServices }: CoreSetup) { this.logger.debug('Setting up SavedObjectsManagement plugin'); + const savedObjectsStartPromise = getStartServices().then(([core]) => core.savedObjects); registerRoutes({ + savedObjectsStartPromise, http, managementServicePromise: this.managementService$.pipe(first()).toPromise(), }); diff --git a/src/plugins/saved_objects_management/server/routes/find.ts b/src/plugins/saved_objects_management/server/routes/find.ts index ad9a547910144..a18bf820d3650 100644 --- a/src/plugins/saved_objects_management/server/routes/find.ts +++ b/src/plugins/saved_objects_management/server/routes/find.ts @@ -18,13 +18,14 @@ */ import { schema } from '@kbn/config-schema'; -import { IRouter } from 'src/core/server'; +import { IRouter, SavedObjectsServiceStart } from 'src/core/server'; import { injectMetaAttributes } from '../lib'; import { ISavedObjectsManagement } from '../services'; export const registerFindRoute = ( router: IRouter, - managementServicePromise: Promise + managementServicePromise: Promise, + savedObjectsStartPromise: Promise ) => { const referenceSchema = schema.object({ type: schema.string(), @@ -57,7 +58,7 @@ export const registerFindRoute = ( }, router.handleLegacyErrors(async (context, req, res) => { const managementService = await managementServicePromise; - const { client } = context.core.savedObjects; + const savedObjects = await savedObjectsStartPromise; const searchTypes = Array.isArray(req.query.type) ? req.query.type : [req.query.type]; const includedFields = Array.isArray(req.query.fields) ? req.query.fields @@ -65,6 +66,13 @@ export const registerFindRoute = ( const importAndExportableTypes = searchTypes.filter((type) => managementService.isImportAndExportable(type) ); + const includedHiddenTypes = importAndExportableTypes.filter((type) => + managementService.isHidden(type) + ); + + const client = savedObjects.getScopedClient(req, { + includedHiddenTypes, + }); const searchFields = new Set(); importAndExportableTypes.forEach((type) => { diff --git a/src/plugins/saved_objects_management/server/routes/index.ts b/src/plugins/saved_objects_management/server/routes/index.ts index e074a0d5cbee2..e35a324bcce6a 100644 --- a/src/plugins/saved_objects_management/server/routes/index.ts +++ b/src/plugins/saved_objects_management/server/routes/index.ts @@ -17,7 +17,7 @@ * under the License. */ -import { HttpServiceSetup } from 'src/core/server'; +import { HttpServiceSetup, SavedObjectsServiceStart } from 'src/core/server'; import { ISavedObjectsManagement } from '../services'; import { registerFindRoute } from './find'; import { registerGetRoute } from './get'; @@ -29,11 +29,16 @@ import { registerGetAllowedTypesRoute } from './get_allowed_types'; interface RegisterRouteOptions { http: HttpServiceSetup; managementServicePromise: Promise; + savedObjectsStartPromise: Promise; } -export function registerRoutes({ http, managementServicePromise }: RegisterRouteOptions) { +export function registerRoutes({ + http, + managementServicePromise, + savedObjectsStartPromise, +}: RegisterRouteOptions) { const router = http.createRouter(); - registerFindRoute(router, managementServicePromise); + registerFindRoute(router, managementServicePromise, savedObjectsStartPromise); registerGetRoute(router, managementServicePromise); registerScrollForCountRoute(router); registerScrollForExportRoute(router); diff --git a/src/plugins/saved_objects_management/server/services/management.ts b/src/plugins/saved_objects_management/server/services/management.ts index f24226d798fae..c4d8c123dfa8d 100644 --- a/src/plugins/saved_objects_management/server/services/management.ts +++ b/src/plugins/saved_objects_management/server/services/management.ts @@ -28,6 +28,10 @@ export class SavedObjectsManagement { return this.registry.isImportableAndExportable(type); } + public isHidden(type: string) { + return this.registry.isHidden(type); + } + public getDefaultSearchField(type: string) { return this.registry.getType(type)?.management?.defaultSearchField; } diff --git a/x-pack/plugins/actions/server/saved_objects/index.ts b/x-pack/plugins/actions/server/saved_objects/index.ts index db811e7cd3a4f..488d74222b12d 100644 --- a/x-pack/plugins/actions/server/saved_objects/index.ts +++ b/x-pack/plugins/actions/server/saved_objects/index.ts @@ -23,6 +23,14 @@ export function setupSavedObjects( namespaceType: 'single', mappings: mappings.action, migrations: getMigrations(encryptedSavedObjects), + management: { + importableAndExportable: true, + icon: 'tokenEvent', + defaultSearchField: 'name', + getTitle(obj) { + return obj.attributes.name; + }, + }, }); // Encrypted attributes @@ -33,6 +41,7 @@ export function setupSavedObjects( type: ACTION_SAVED_OBJECT_TYPE, attributesToEncrypt: new Set(['secrets']), attributesToExcludeFromAAD: new Set(['name']), + allowPredefinedID: true, }); savedObjects.registerType({ diff --git a/x-pack/plugins/alerts/server/saved_objects/index.ts b/x-pack/plugins/alerts/server/saved_objects/index.ts index dfe122f56bc48..bf42e376a65bc 100644 --- a/x-pack/plugins/alerts/server/saved_objects/index.ts +++ b/x-pack/plugins/alerts/server/saved_objects/index.ts @@ -42,6 +42,14 @@ export function setupSavedObjects( namespaceType: 'single', migrations: getMigrations(encryptedSavedObjects), mappings: mappings.alert, + management: { + importableAndExportable: true, + icon: 'alert', + defaultSearchField: 'name', + getTitle(obj) { + return obj.attributes.name; + }, + }, }); savedObjects.registerType({ @@ -65,6 +73,7 @@ export function setupSavedObjects( type: 'alert', attributesToEncrypt: new Set(['apiKey']), attributesToExcludeFromAAD: new Set(AlertAttributesExcludedFromAAD), + allowPredefinedID: true, }); // Encrypted attributes diff --git a/x-pack/plugins/features/server/oss_features.ts b/x-pack/plugins/features/server/oss_features.ts index 46dd5fd086b4a..c102bbb19fbcf 100644 --- a/x-pack/plugins/features/server/oss_features.ts +++ b/x-pack/plugins/features/server/oss_features.ts @@ -5,14 +5,17 @@ */ import { i18n } from '@kbn/i18n'; import { KibanaFeatureConfig } from '../common'; -import { DEFAULT_APP_CATEGORIES } from '../../../../src/core/server'; +import { DEFAULT_APP_CATEGORIES, ISavedObjectTypeRegistry } from '../../../../src/core/server'; export interface BuildOSSFeaturesParams { - savedObjectTypes: string[]; + savedObjectTypeRegistry: ISavedObjectTypeRegistry; includeTimelion: boolean; } -export const buildOSSFeatures = ({ savedObjectTypes, includeTimelion }: BuildOSSFeaturesParams) => { +export const buildOSSFeatures = ({ + savedObjectTypeRegistry, + includeTimelion, +}: BuildOSSFeaturesParams) => { return [ { id: 'discover', @@ -346,7 +349,7 @@ export const buildOSSFeatures = ({ savedObjectTypes, includeTimelion }: BuildOSS }, api: ['copySavedObjectsToSpaces'], savedObject: { - all: [...savedObjectTypes], + all: [...savedObjectTypeRegistry.getImportableAndExportableTypes().map((t) => t.name)], read: [], }, ui: ['read', 'edit', 'delete', 'copyIntoSpace', 'shareIntoSpace'], @@ -360,7 +363,7 @@ export const buildOSSFeatures = ({ savedObjectTypes, includeTimelion }: BuildOSS api: ['copySavedObjectsToSpaces'], savedObject: { all: [], - read: [...savedObjectTypes], + read: [...savedObjectTypeRegistry.getImportableAndExportableTypes().map((t) => t.name)], }, ui: ['read'], }, diff --git a/x-pack/plugins/features/server/plugin.ts b/x-pack/plugins/features/server/plugin.ts index 857bba4c606d4..a83b4490210ae 100644 --- a/x-pack/plugins/features/server/plugin.ts +++ b/x-pack/plugins/features/server/plugin.ts @@ -109,16 +109,11 @@ export class Plugin { public stop() {} private registerOssFeatures(savedObjects: SavedObjectsServiceStart) { - const registry = savedObjects.getTypeRegistry(); - const savedObjectTypes = registry.getVisibleTypes().map((t) => t.name); + const savedObjectTypeRegistry = savedObjects.getTypeRegistry(); - this.logger.debug( - `Registering OSS features with SO types: ${savedObjectTypes.join(', ')}. "includeTimelion": ${ - this.isTimelionEnabled - }.` - ); + this.logger.debug(`Registering OSS features. "includeTimelion": ${this.isTimelionEnabled}.`); const features = buildOSSFeatures({ - savedObjectTypes, + savedObjectTypeRegistry, includeTimelion: this.isTimelionEnabled, });