Frontend Sandbox: Allow enabling sandbox before installing plugins (#100365)

* Frontend Sandbox: Allow enabling sandbox before installing plugins
This commit is contained in:
Esteban Beltran 2025-02-11 10:52:21 +01:00 committed by GitHub
parent e5154ce799
commit 9f77c86b21
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 153 additions and 34 deletions

View File

@ -37,6 +37,8 @@ export async function getPluginDetails(id: string): Promise<CatalogPluginDetails
iam: remote?.json?.iam,
lastCommitDate: remote?.lastCommitDate,
changelog: remote?.changelog || localChangelog,
signatureType: local?.signatureType || (remote?.signatureType !== '' ? remote?.signatureType : undefined),
signature: local?.signature,
};
}

View File

@ -79,6 +79,8 @@ export interface CatalogPluginDetails {
iam?: IdentityAccessManagement;
changelog?: string;
lastCommitDate?: string;
signatureType?: PluginSignatureType;
signature?: PluginSignatureStatus;
}
export interface CatalogPluginInfo {

View File

@ -1,6 +1,8 @@
import { PluginMeta, PluginSignatureType } from '@grafana/data';
import { PluginMeta, PluginSignatureStatus, PluginSignatureType } from '@grafana/data';
import { config } from '@grafana/runtime';
import { getPluginDetails } from '../admin/api';
import { CatalogPluginDetails } from '../admin/types';
import { getPluginSettings } from '../pluginSettings';
import {
@ -22,18 +24,31 @@ jest.mock('../pluginSettings', () => ({
getPluginSettings: jest.fn(),
}));
const getPluginSettingsMock = getPluginSettings as jest.MockedFunction<typeof getPluginSettings>;
jest.mock('../admin/api', () => ({
getPluginDetails: jest.fn(),
}));
const fakePlugin: PluginMeta = {
const getPluginSettingsMock = jest.mocked(getPluginSettings);
const getPluginDetailsMock = jest.mocked(getPluginDetails);
const fakePluginSettings: PluginMeta = {
id: 'test-plugin',
name: 'Test Plugin',
} as PluginMeta;
const fakePluginDetails: CatalogPluginDetails = {} as CatalogPluginDetails;
describe('Sandbox eligibility checks', () => {
const originalNodeEnv = process.env.NODE_ENV;
beforeEach(() => {
jest.clearAllMocks();
getPluginDetailsMock.mockReset();
getPluginSettingsMock.mockReset();
// restore default check
setSandboxEnabledCheck(isPluginFrontendSandboxEnabled);
config.enableFrontendSandboxForPlugins = [];
config.featureToggles.pluginsFrontendSandbox = true;
process.env.NODE_ENV = 'development';
@ -54,26 +69,8 @@ describe('Sandbox eligibility checks', () => {
expect(result).toBe(false);
});
test('shouldLoadPluginInFrontendSandbox returns false for Grafana-signed plugins', async () => {
getPluginSettingsMock.mockResolvedValue({ ...fakePlugin, signatureType: PluginSignatureType.grafana });
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
test('shouldLoadPluginInFrontendSandbox returns true for eligible plugins in the list', async () => {
getPluginSettingsMock.mockResolvedValue({ ...fakePlugin, signatureType: PluginSignatureType.community });
config.enableFrontendSandboxForPlugins = ['test-plugin'];
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(true);
});
test('isPluginFrontendSandboxEnabled returns false when plugin is not in the enabled list', async () => {
config.enableFrontendSandboxForPlugins = ['other-plugin'];
const result = await isPluginFrontendSandboxEnabled({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
test('setSandboxEnabledCheck sets custom check function', async () => {
getPluginDetailsMock.mockResolvedValue(fakePluginDetails);
const customCheck = jest.fn().mockResolvedValue(true);
setSandboxEnabledCheck(customCheck);
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
@ -82,6 +79,7 @@ describe('Sandbox eligibility checks', () => {
});
test('setSandboxEnabledCheck has precedence over default', async () => {
getPluginDetailsMock.mockResolvedValue(fakePluginDetails);
const customCheck = jest.fn().mockResolvedValue(false);
setSandboxEnabledCheck(customCheck);
// this should be ignored by the custom check
@ -91,10 +89,123 @@ describe('Sandbox eligibility checks', () => {
expect(result).toBe(false);
});
test('isPluginFrontendSandboxEligible returns false for plugins with internal signature', async () => {
//@ts-expect-error We don't publicly export the internal signature
getPluginSettingsMock.mockResolvedValue({ ...fakePlugin, signature: 'internal' });
const result = await isPluginFrontendSandboxEligible({ pluginId: 'test-plugin' });
expect(result).toBe(false);
describe('with getPluginDetails', () => {
test('shouldLoadPluginInFrontendSandbox returns false for Grafana-signed plugins', async () => {
getPluginSettingsMock.mockRejectedValueOnce(new Error('not found'));
getPluginDetailsMock.mockResolvedValue({ ...fakePluginDetails, signatureType: PluginSignatureType.grafana });
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
test('shouldLoadPluginInFrontendSandbox returns true for community plugins', async () => {
getPluginSettingsMock.mockRejectedValueOnce(new Error('not found'));
getPluginDetailsMock.mockResolvedValue({ ...fakePluginDetails, signatureType: PluginSignatureType.community });
config.enableFrontendSandboxForPlugins = ['test-plugin'];
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(true);
});
test('isPluginFrontendSandboxEnabled returns false when plugin is not in the enabled list', async () => {
getPluginSettingsMock.mockRejectedValueOnce(new Error('not found'));
getPluginDetailsMock.mockResolvedValue({ ...fakePluginDetails, signatureType: PluginSignatureType.community });
config.enableFrontendSandboxForPlugins = ['other-plugin'];
const result = await isPluginFrontendSandboxEnabled({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
test('shouldLoadPluginInFrontendSandbox returns true for commercial plugins in the enabled list', async () => {
getPluginSettingsMock.mockRejectedValueOnce(new Error('not found'));
getPluginDetailsMock.mockResolvedValue({ ...fakePluginDetails, signatureType: PluginSignatureType.commercial });
config.enableFrontendSandboxForPlugins = ['test-plugin'];
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(true);
});
test('shouldLoadPluginInFrontendSandbox returns true for private plugins in the enabled list', async () => {
getPluginSettingsMock.mockRejectedValueOnce(new Error('not found'));
getPluginDetailsMock.mockResolvedValue({ ...fakePluginDetails, signatureType: PluginSignatureType.private });
config.enableFrontendSandboxForPlugins = ['test-plugin'];
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(true);
});
test('isPluginFrontendSandboxEligible returns false for plugins with internal signature', async () => {
getPluginSettingsMock.mockRejectedValueOnce(new Error('not found'));
getPluginDetailsMock.mockResolvedValue({
...fakePluginDetails,
signatureType: PluginSignatureType.community,
signature: PluginSignatureStatus.internal,
});
const result = await isPluginFrontendSandboxEligible({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
});
describe('with getPluginSettings', () => {
test('shouldLoadPluginInFrontendSandbox returns false for Grafana-signed plugins', async () => {
// if getPluginDetails fails it fallsback to getPluginSettings
getPluginDetailsMock.mockRejectedValueOnce(new Error('not found'));
getPluginSettingsMock.mockResolvedValue({ ...fakePluginSettings, signatureType: PluginSignatureType.grafana });
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
test('shouldLoadPluginInFrontendSandbox returns true for eligible plugins in the list', async () => {
// if getPluginDetails fails it fallsback to getPluginSettings
getPluginDetailsMock.mockRejectedValueOnce(new Error('not found'));
getPluginSettingsMock.mockResolvedValue({ ...fakePluginSettings, signatureType: PluginSignatureType.community });
config.enableFrontendSandboxForPlugins = ['test-plugin'];
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(true);
});
test('isPluginFrontendSandboxEnabled returns false when plugin is not in the enabled list', async () => {
// if getPluginDetails fails it fallsback to getPluginSettings
getPluginDetailsMock.mockRejectedValueOnce(new Error('not found'));
config.enableFrontendSandboxForPlugins = ['other-plugin'];
const result = await isPluginFrontendSandboxEnabled({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
test('shouldLoadPluginInFrontendSandbox returns true for commercial plugins in the enabled list', async () => {
// if getPluginDetails fails it fallsback to getPluginSettings
getPluginDetailsMock.mockRejectedValueOnce(new Error('not found'));
getPluginSettingsMock.mockResolvedValue({ ...fakePluginSettings, signatureType: PluginSignatureType.commercial });
config.enableFrontendSandboxForPlugins = ['test-plugin'];
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(true);
});
test('shouldLoadPluginInFrontendSandbox returns true for private plugins in the enabled list', async () => {
// if getPluginDetails fails it fallsback to getPluginSettings
getPluginDetailsMock.mockRejectedValueOnce(new Error('not found'));
getPluginSettingsMock.mockResolvedValue({ ...fakePluginSettings, signatureType: PluginSignatureType.private });
config.enableFrontendSandboxForPlugins = ['test-plugin'];
const result = await shouldLoadPluginInFrontendSandbox({ pluginId: 'test-plugin' });
expect(result).toBe(true);
});
test('isPluginFrontendSandboxEligible returns false for plugins with internal signature', async () => {
// if getPluginDetails fails it fallsback to getPluginSettings
getPluginDetailsMock.mockRejectedValueOnce(new Error('not found'));
getPluginSettingsMock.mockResolvedValue({ ...fakePluginSettings, signature: PluginSignatureStatus.internal });
const result = await isPluginFrontendSandboxEligible({ pluginId: 'test-plugin' });
expect(result).toBe(false);
});
});
});

View File

@ -1,6 +1,7 @@
import { PluginSignatureType } from '@grafana/data';
import { config } from '@grafana/runtime';
import { getPluginDetails } from '../admin/api';
import { getPluginSettings } from '../pluginSettings';
type SandboxEligibilityCheckParams = {
@ -61,17 +62,20 @@ export async function isPluginFrontendSandboxEligible({
return false;
}
// don't run grafana-signed plugins in sandbox
try {
// don't run grafana-signed plugins in sandbox
const pluginMeta = await getPluginSettings(pluginId, { showErrorAlert: false });
if (pluginMeta.signatureType === PluginSignatureType.grafana || pluginMeta.signature === 'internal') {
//this can fail if gcom is not accesible
const details = await getPluginDetails(pluginId);
return details.signatureType !== PluginSignatureType.grafana && details.signature !== 'internal';
} catch (e) {
try {
// this can fail if we are trying to fetch settings of a non-installed plugin
const pluginMeta = await getPluginSettings(pluginId, { showErrorAlert: false });
return pluginMeta.signatureType !== PluginSignatureType.grafana && pluginMeta.signature !== 'internal';
} catch (e) {
return false;
}
} catch (e) {
// this can fail if we are trying to fetch settings of a non-installed plugin
return false;
}
return true;
}