kibana/x-pack/plugins/spaces/server/plugin.ts
elena-shostak 3d09eaa6fc
[Spaces] Passing default solution from cloud onboarding process (#185926)
## Summary

Passing default solution from cloud onboarding process.

1. Renaming. Solution changes are not released yet, would be shipped
with `8.15`, so it's fine to do it.
   - `search` -> `es`
   - `observability` -> `oblt`
   - Adjusted telemetry accordingly
2. Added `cloud` as optional dependency to `spaces` plugin to use
`onboarding.defaultSolution` passed through setup contract.

### How to test
1. Set `xpack.cloud.onboarding.default_solution` to `es | oblt |
security`
2. Check that default space was created with provided solution `GET
kbn:/api/spaces/space/default`

### Checklist

- [x]
[Documentation](https://www.elastic.co/guide/en/kibana/master/development-documentation.html)
was added for features that require explanation or tutorials
- [x] [Unit or functional
tests](https://www.elastic.co/guide/en/kibana/master/development-tests.html)
were updated or added to match the most common scenarios


### For maintainers

- [x] This was checked for breaking API changes and was [labeled
appropriately](https://www.elastic.co/guide/en/kibana/master/contributing.html#kibana-release-notes-process)

__Fixes: https://github.com/elastic/kibana/issues/184999__

---------

Co-authored-by: kibanamachine <42973632+kibanamachine@users.noreply.github.com>
2024-06-18 11:10:52 +02:00

243 lines
7.6 KiB
TypeScript

/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/
import type { Observable } from 'rxjs';
import { map } from 'rxjs';
import type { CloudSetup } from '@kbn/cloud-plugin/server';
import type {
CoreSetup,
CoreStart,
Logger,
Plugin,
PluginInitializerContext,
} from '@kbn/core/server';
import type {
PluginSetupContract as FeaturesPluginSetup,
PluginStartContract as FeaturesPluginStart,
} from '@kbn/features-plugin/server';
import type { HomeServerPluginSetup } from '@kbn/home-plugin/server';
import type { LicensingPluginSetup } from '@kbn/licensing-plugin/server';
import type { UsageCollectionSetup } from '@kbn/usage-collection-plugin/server';
import { setupCapabilities } from './capabilities';
import type { ConfigType } from './config';
import { DefaultSpaceService } from './default_space';
import { initSpacesRequestInterceptors } from './lib/request_interceptors';
import { createSpacesTutorialContextFactory } from './lib/spaces_tutorial_context_factory';
import { initExternalSpacesApi } from './routes/api/external';
import { initInternalSpacesApi } from './routes/api/internal';
import { initSpacesViewsRoutes } from './routes/views';
import { SpacesSavedObjectsService } from './saved_objects';
import type { SpacesClientRepositoryFactory, SpacesClientWrapper } from './spaces_client';
import { SpacesClientService } from './spaces_client';
import type { SpacesServiceSetup, SpacesServiceStart } from './spaces_service';
import { SpacesService } from './spaces_service';
import type { SpacesRequestHandlerContext } from './types';
import { registerSpacesUsageCollector } from './usage_collection';
import { UsageStatsService } from './usage_stats';
import { SpacesLicenseService } from '../common/licensing';
export interface PluginsSetup {
features: FeaturesPluginSetup;
licensing: LicensingPluginSetup;
usageCollection?: UsageCollectionSetup;
home?: HomeServerPluginSetup;
cloud?: CloudSetup;
}
export interface PluginsStart {
features: FeaturesPluginStart;
}
/**
* Setup contract for the Spaces plugin.
*/
export interface SpacesPluginSetup {
/**
* Service for interacting with spaces.
*/
spacesService: SpacesServiceSetup;
/**
* Registries exposed for the security plugin to transparently provide authorization and audit logging.
* @private
*/
spacesClient: {
/**
* Sets the client repository factory.
* @private
*/
setClientRepositoryFactory: (factory: SpacesClientRepositoryFactory) => void;
/**
* Registers a client wrapper.
* @private
*/
registerClientWrapper: (wrapper: SpacesClientWrapper) => void;
};
/**
* Determines whether Kibana supports multiple spaces or only the default space.
*
* When `xpack.spaces.maxSpaces` is set to 1 Kibana only supports the default space and any spaces related UI can safely be hidden.
*/
hasOnlyDefaultSpace$: Observable<boolean>;
}
/**
* Start contract for the Spaces plugin.
*/
export interface SpacesPluginStart {
/** Service for interacting with spaces. */
spacesService: SpacesServiceStart;
/**
* Determines whether Kibana supports multiple spaces or only the default space.
*
* When `xpack.spaces.maxSpaces` is set to 1 Kibana only supports the default space and any spaces related UI can safely be hidden.
*/
hasOnlyDefaultSpace$: Observable<boolean>;
}
export class SpacesPlugin
implements Plugin<SpacesPluginSetup, SpacesPluginStart, PluginsSetup, PluginsStart>
{
private readonly config$: Observable<ConfigType>;
private readonly log: Logger;
private readonly spacesLicenseService = new SpacesLicenseService();
private readonly spacesClientService: SpacesClientService;
private readonly spacesService: SpacesService;
private readonly hasOnlyDefaultSpace$: Observable<boolean>;
private spacesServiceStart?: SpacesServiceStart;
private defaultSpaceService?: DefaultSpaceService;
constructor(private readonly initializerContext: PluginInitializerContext) {
this.config$ = initializerContext.config.create<ConfigType>();
this.hasOnlyDefaultSpace$ = this.config$.pipe(map(({ maxSpaces }) => maxSpaces === 1));
this.log = initializerContext.logger.get();
this.spacesService = new SpacesService();
this.spacesClientService = new SpacesClientService(
(message) => this.log.debug(message),
initializerContext.env.packageInfo.buildFlavor
);
}
public setup(core: CoreSetup<PluginsStart>, plugins: PluginsSetup): SpacesPluginSetup {
const spacesClientSetup = this.spacesClientService.setup({ config$: this.config$ });
const spacesServiceSetup = this.spacesService.setup({
basePath: core.http.basePath,
});
const getSpacesService = () => {
if (!this.spacesServiceStart) {
throw new Error('spaces service has not been initialized!');
}
return this.spacesServiceStart;
};
const usageStatsServicePromise = new UsageStatsService(this.log).setup({
getStartServices: core.getStartServices,
});
const savedObjectsService = new SpacesSavedObjectsService();
savedObjectsService.setup({ core, getSpacesService });
const { license } = this.spacesLicenseService.setup({ license$: plugins.licensing.license$ });
this.defaultSpaceService = new DefaultSpaceService();
this.defaultSpaceService.setup({
coreStatus: core.status,
getSavedObjects: async () => (await core.getStartServices())[0].savedObjects,
license$: plugins.licensing.license$,
spacesLicense: license,
logger: this.log,
solution: plugins.cloud?.onboarding?.defaultSolution,
});
initSpacesViewsRoutes({
httpResources: core.http.resources,
basePath: core.http.basePath,
logger: this.log,
});
const router = core.http.createRouter<SpacesRequestHandlerContext>();
initExternalSpacesApi({
router,
log: this.log,
getStartServices: core.getStartServices,
getSpacesService,
usageStatsServicePromise,
isServerless: this.initializerContext.env.packageInfo.buildFlavor === 'serverless',
});
initInternalSpacesApi({
router,
getSpacesService,
});
initSpacesRequestInterceptors({
http: core.http,
log: this.log,
getSpacesService,
features: plugins.features,
});
setupCapabilities(core, getSpacesService, this.log);
if (plugins.usageCollection) {
const getIndexForType = (type: string) =>
core.getStartServices().then(([coreStart]) => coreStart.savedObjects.getIndexForType(type));
registerSpacesUsageCollector(plugins.usageCollection, {
getIndexForType,
features: plugins.features,
licensing: plugins.licensing,
usageStatsServicePromise,
});
}
if (plugins.home) {
plugins.home.tutorials.addScopedTutorialContextFactory(
createSpacesTutorialContextFactory(getSpacesService)
);
}
return {
spacesClient: spacesClientSetup,
spacesService: spacesServiceSetup,
hasOnlyDefaultSpace$: this.hasOnlyDefaultSpace$,
};
}
public start(core: CoreStart) {
const spacesClientStart = this.spacesClientService.start(core);
this.spacesServiceStart = this.spacesService.start({
basePath: core.http.basePath,
spacesClientService: spacesClientStart,
});
return {
spacesService: this.spacesServiceStart,
hasOnlyDefaultSpace$: this.hasOnlyDefaultSpace$,
};
}
public stop() {
if (this.defaultSpaceService) {
this.defaultSpaceService.stop();
}
}
}