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

[share] hideEmbed serverless configuration to hide 'Embed code' share option in serverless #162354

Closed
wants to merge 6 commits into from
Closed
Show file tree
Hide file tree
Changes from 5 commits
Commits
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
24 changes: 24 additions & 0 deletions src/plugins/share/config.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
/*
* 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 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/

import { schema, TypeOf } from '@kbn/config-schema';

export const configSchema = schema.object({
hideEmbed: schema.conditional(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

note: Up to you, but I'm wondering if it'd be easier\cleaner to rely on env.buildFlavor provided by the Core (both on the server and client side) instead of creating a new, technically immutable, config flag.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It doesn't have to be a config value. You can expose an API on the start contract and call it from the relevant serverless plugin.

Flexing based on is(serverless) means you lose context on related serverless decisions. It will be hard to parse out what code applies to a particular functional decision if everything is keyed off of that binary proposition.

Copy link
Member

@azasypkin azasypkin Jul 26, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It will be hard to parse out what code applies to a particular functional decision if everything is keyed off of that binary proposition.

Well, it might sound good in theory, but I'm not sure it's the case in practice. We have discussed this a lot internally for the past few weeks, so sorry for lots of text 🙈

The things like share.hideEmbed just add an unnecessary level of indirection and complexity around something that is only relevant to the Serverless offering. As far as I am aware, there is no valid use case for "hiding embed" controls, no such "feature" is on our roadmap, and might never be. We're just trying to come up with a workaround for the limitations we have in one specific offering, pretending that it's not just an "isServerless" flag. Moreover, we might lift some of these limitations eventually, making many of these "fake-future" flags unnecessary.

When I read the code and see share.hideEmbed and friends, unlike isServerless, the following questions pop up in my mind:

  • What use case is this for?
  • What consumers outside of this plugin rely on it?
  • Are there any consumers at all?
  • Can I safely change or remove it if I have to?
  • Since it's exposed through a contract, can I rely on it in my other plugin? If not, why?

No matter how you expose it, via config or via plugin contracts, these are all public APIs that we're supposed to support backward compatibility for. It's just that we usually ignore breaking changes in our programmatic API contracts since we don't have many external plugins that could be affected by these changes.

In summary, what I want to say is that we have at least three different use cases when it comes to Serverless, and trying to solve them with one tool might not be the best approach:

  1. Kibana functionality that should not be available or work differently in Serverless but represents a well-isolated "domain" that makes sense on its own, and there is a high chance that this functionality might be disabled/re-configured in other contexts/offerings in the future. This is the perfect use case for feature config flags and additional APIs on the setup/start contracts.

  2. Kibana functionality that mostly depends on counterpart Elasticsearch functionality that should not be available or work differently in Serverless. In this case, Kibana shouldn't care about Serverless directly and ideally should rely on available Elasticsearch capabilities exposed in one way or another. If Elasticsearch doesn't yet expose a way to advertise certain capabilities, relying on isServerless in Kibana as an interim solution sounds like the best\fairest approach. Alternatively, we can leverage ES capabilities API exposed by the Core's Elasticsearch client Introducing the concept of ES capabilities #164850.

  3. Kibana functionality which customizations only make sense in the Serverless context. Unless we really can tick all the boxes from the option 1, I don't think we should pretend that it's a "feature" and just use the isServerless flag until we feel it's indeed a feature so that we can drop isServerless and expose a proper feature flag or programmatic feature management API.

Having said that, it's not my call here, just wanted to share my perspective!

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Kibana functionality which customizations only make sense in the Serverless context. Unless we really can tick all the boxes from the option 1, I don't think we should pretend that it's a "feature" and just use the isServerless flag until we feel it's indeed a feature so that we can drop isServerless and expose a proper feature flag or programmatic feature management API.

Is there an example that uses isServerless flag that you can point me too? I have no strong opinions about implementation path.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The env.buildFlavor, provided by the Core (both on the server and client side), is a relatively new addition, but we're planning to use it like that on the server side https://github.com/elastic/kibana/pull/162087/files#diff-4fbb178a08b40c270265f939de2e735ba79ea1181ea5c1ebca18dd032fcfcf2eR98, and I see Maps is using this flag on the client side as well:

if (buildFlavor === 'traditional' && kbnSemVer) {
landingPageUrl = `${landingPageUrl}/v${kbnSemVer.major}.${kbnSemVer.minor}`;
}

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the examples.

As another data point, #161528, mentioned in AppEx updates 2023-08-02, uses a yaml configuration to disable functionality in serverless. I don't think there is a consistent approach across Kibana

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would +1 to the yaml config option as it allows us to flip this switch regardless of serverless. I.e. we might want to hide this functionality in all offerings in the future.

Copy link
Member

@azasypkin azasypkin Aug 2, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think there is a consistent approach across Kibana

Yes, that's true, unfortunately. The timeline for this looks like this:

timeline
    title The history of "Serverless" flag in Kibana
    Beginning : We agreed to use configuration feature flags to toggle/adjust features in Serverless
    Next : We realized that configuration flags are essentially public APIs and we decided to gate them with "serverless" configuration context reference
    Next : People started doing crazy things with this config context reference just to emulate "isServerless" config flag, because they didn't want to invent "fake" features just to have flags for them
    Next : In parallel, other people started doing other ...interesting... things with optional dependencies to the "serverless"-only plugins just, again, to emulate "isServerless" config flag
    Next : We realized that there are "serverless"-only things that depend on Elasticsearch capabilities for which config feature flags don't make sense
    Next : The Core team introduced the `env.buildFlavour` to stop that madness (opinion is my own) with the three previous points. 
    We're here : Now we have multiple different solutions, but no clear guidance for when to use which. All in all, it's much more straightforward to migrate from `env.buildFlavour` to any other solution in the future then vice versa.
Loading

schema.contextRef('serverless'),
true,
schema.maybe(schema.boolean({ defaultValue: false })),
schema.never()
),
});

export type ShareConfig = TypeOf<typeof configSchema>;

export interface SharePublicConfig {
hideEmbed?: boolean;
}
3 changes: 2 additions & 1 deletion src/plugins/share/public/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
*/

import type { PluginInitializerContext } from '@kbn/core/public';
import { SharePublicConfig } from '../config';

export { CSV_QUOTE_VALUES_SETTING, CSV_SEPARATOR_SETTING } from '../common/constants';

Expand All @@ -31,6 +32,6 @@ import { SharePlugin } from './plugin';
export { downloadMultipleAs, downloadFileAs } from './lib/download_as';
export type { DownloadableContent } from './lib/download_as';

export function plugin(ctx: PluginInitializerContext) {
export function plugin(ctx: PluginInitializerContext<SharePublicConfig>) {
return new SharePlugin(ctx);
}
26 changes: 24 additions & 2 deletions src/plugins/share/public/plugin.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,8 @@ describe('SharePlugin', () => {
expect.objectContaining({
getShareMenuItems: expect.any(Function),
}),
undefined
undefined,
undefined // hideEmbed - undefined because its not provided in configuration
);
expect(start.toggleShareContextMenu).toBeDefined();
});
Expand All @@ -73,7 +74,28 @@ describe('SharePlugin', () => {
expect.objectContaining({
getShareMenuItems: expect.any(Function),
}),
anonymousAccessServiceProvider
anonymousAccessServiceProvider,
undefined // hideEmbed - undefined because its not provided in configuration
);
expect(start.toggleShareContextMenu).toBeDefined();
});

test('passes hideEmbed config to share menu manager', async () => {
const coreSetup = coreMock.createSetup();
const service = new SharePlugin(
coreMock.createPluginInitializerContext({ hideEmbed: true })
);
await service.setup(coreSetup);
const start = await service.start({} as CoreStart);
expect(registryMock.start).toHaveBeenCalled();
expect(managerMock.start).toHaveBeenCalledWith(
expect.anything(),
expect.anything(),
expect.objectContaining({
getShareMenuItems: expect.any(Function),
}),
undefined,
true // hideEmbed - true because its set in configuration as 'true'
);
expect(start.toggleShareContextMenu).toBeDefined();
});
Expand Down
7 changes: 5 additions & 2 deletions src/plugins/share/public/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
import './index.scss';

import type { CoreSetup, CoreStart, Plugin, PluginInitializerContext } from '@kbn/core/public';
import { SharePublicConfig } from '../config';
import { ShareMenuManager, ShareMenuManagerStart } from './services';
import { ShareMenuRegistry, ShareMenuRegistrySetup } from './services';
import { UrlService } from '../common/url_service';
Expand Down Expand Up @@ -65,7 +66,7 @@ export class SharePlugin implements Plugin<SharePluginSetup, SharePluginStart> {
private url?: BrowserUrlService;
private anonymousAccessServiceProvider?: () => AnonymousAccessServiceContract;

constructor(private readonly initializerContext: PluginInitializerContext) {}
constructor(private readonly initializerContext: PluginInitializerContext<SharePublicConfig>) {}

public setup(core: CoreSetup): SharePluginSetup {
const { http } = core;
Expand Down Expand Up @@ -120,11 +121,13 @@ export class SharePlugin implements Plugin<SharePluginSetup, SharePluginStart> {
}

public start(core: CoreStart): SharePluginStart {
const { hideEmbed } = this.initializerContext.config.get<SharePublicConfig>();
const sharingContextMenuStart = this.shareContextMenu.start(
core,
this.url!,
this.shareMenuRegistry.start(),
this.anonymousAccessServiceProvider
this.anonymousAccessServiceProvider,
hideEmbed
);

return {
Expand Down
4 changes: 3 additions & 1 deletion src/plugins/share/public/services/share_menu_manager.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,8 @@ export class ShareMenuManager {
core: CoreStart,
urlService: BrowserUrlService,
shareRegistry: ShareMenuRegistryStart,
anonymousAccessServiceProvider?: () => AnonymousAccessServiceContract
anonymousAccessServiceProvider?: () => AnonymousAccessServiceContract,
hideEmbed?: boolean
) {
return {
/**
Expand All @@ -45,6 +46,7 @@ export class ShareMenuManager {
const anonymousAccess = anonymousAccessServiceProvider?.();
this.toggleShareContextMenu({
...options,
allowEmbed: hideEmbed ? false : options.allowEmbed,
onClose,
menuItems,
urlService,
Expand Down
10 changes: 9 additions & 1 deletion src/plugins/share/server/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,21 @@
* Side Public License, v 1.
*/

import { PluginInitializerContext } from '@kbn/core/server';
import { PluginInitializerContext, PluginConfigDescriptor } from '@kbn/core/server';
import { configSchema, ShareConfig } from '../config';
import { SharePlugin } from './plugin';

export type { SharePluginSetup, SharePluginStart } from './plugin';

export { CSV_QUOTE_VALUES_SETTING, CSV_SEPARATOR_SETTING } from '../common/constants';

export const config: PluginConfigDescriptor<ShareConfig> = {
exposeToBrowser: {
hideEmbed: true,
},
schema: configSchema,
};

export function plugin(initializerContext: PluginInitializerContext) {
return new SharePlugin(initializerContext);
}
2 changes: 1 addition & 1 deletion src/plugins/share/tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
"compilerOptions": {
"outDir": "target/types",
},
"include": ["common/**/*", "public/**/*", "server/**/*"],
"include": ["common/**/*", "public/**/*", "server/**/*", "*.ts"],
"kbn_references": [
"@kbn/core",
"@kbn/kibana-react-plugin",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -143,6 +143,7 @@ export default function ({ getService }: PluginFunctionalProviderContext) {
'newsfeed.mainInterval (duration)',
'newsfeed.service.pathTemplate (string)',
'newsfeed.service.urlRoot (string)',
'share.hideEmbed (any)',
'telemetry.allowChangingOptInStatus (boolean)',
'telemetry.appendServerlessChannelsSuffix (any)', // It's a boolean (any because schema.conditional)
'telemetry.banner (boolean)',
Expand Down