mirror of
https://github.com/grafana/grafana.git
synced 2025-02-13 00:55:47 -06:00
* Added labels * App page fixes * Switch to switch * wip * Updates * I am stuck * Minor tweak * This props interface could work * removed change * use new page extensions in plugin details page * add link separator, fix action button spacing * some renaming * Move PageInfo into it's own folder + add tests * add support for new props in old page header * remove PluginDetailsHeader as it's no longer used * Fix unit tests * fix some badge alignments * center align actions * badge alignment + only show downloads for community/commercial plugins * better link alignment * conditionally render description * move install control warnings to below subtitle + refactor Co-authored-by: Torkel Ödegaard <torkel@grafana.com>
312 lines
9.3 KiB
TypeScript
312 lines
9.3 KiB
TypeScript
import { PluginSignatureStatus, dateTimeParse, PluginError, PluginType, PluginErrorCode } from '@grafana/data';
|
|
import { config, featureEnabled } from '@grafana/runtime';
|
|
import { Settings } from 'app/core/config';
|
|
import { contextSrv } from 'app/core/core';
|
|
import { getBackendSrv } from 'app/core/services/backend_srv';
|
|
import { AccessControlAction } from 'app/types';
|
|
|
|
import { isGrafanaAdmin } from './permissions';
|
|
import { CatalogPlugin, LocalPlugin, RemotePlugin, Version } from './types';
|
|
|
|
export function mergeLocalsAndRemotes(
|
|
local: LocalPlugin[] = [],
|
|
remote: RemotePlugin[] = [],
|
|
errors?: PluginError[]
|
|
): CatalogPlugin[] {
|
|
const catalogPlugins: CatalogPlugin[] = [];
|
|
const errorByPluginId = groupErrorsByPluginId(errors);
|
|
|
|
// add locals
|
|
local.forEach((l) => {
|
|
const remotePlugin = remote.find((r) => r.slug === l.id);
|
|
const error = errorByPluginId[l.id];
|
|
|
|
if (!remotePlugin) {
|
|
catalogPlugins.push(mergeLocalAndRemote(l, undefined, error));
|
|
}
|
|
});
|
|
|
|
// add remote
|
|
remote.forEach((r) => {
|
|
const localPlugin = local.find((l) => l.id === r.slug);
|
|
const error = errorByPluginId[r.slug];
|
|
|
|
catalogPlugins.push(mergeLocalAndRemote(localPlugin, r, error));
|
|
});
|
|
|
|
return catalogPlugins;
|
|
}
|
|
|
|
export function mergeLocalAndRemote(local?: LocalPlugin, remote?: RemotePlugin, error?: PluginError): CatalogPlugin {
|
|
if (!local && remote) {
|
|
return mapRemoteToCatalog(remote, error);
|
|
}
|
|
|
|
if (local && !remote) {
|
|
return mapLocalToCatalog(local, error);
|
|
}
|
|
|
|
return mapToCatalogPlugin(local, remote, error);
|
|
}
|
|
|
|
export function mapRemoteToCatalog(plugin: RemotePlugin, error?: PluginError): CatalogPlugin {
|
|
const {
|
|
name,
|
|
slug: id,
|
|
description,
|
|
version,
|
|
orgName,
|
|
popularity,
|
|
downloads,
|
|
typeCode,
|
|
updatedAt,
|
|
createdAt: publishedAt,
|
|
status,
|
|
} = plugin;
|
|
|
|
const isDisabled = !!error || isDisabledSecretsPlugin(typeCode);
|
|
return {
|
|
description,
|
|
downloads,
|
|
id,
|
|
info: {
|
|
logos: {
|
|
small: `https://grafana.com/api/plugins/${id}/versions/${version}/logos/small`,
|
|
large: `https://grafana.com/api/plugins/${id}/versions/${version}/logos/large`,
|
|
},
|
|
},
|
|
name,
|
|
orgName,
|
|
popularity,
|
|
publishedAt,
|
|
signature: getPluginSignature({ remote: plugin, error }),
|
|
updatedAt,
|
|
hasUpdate: false,
|
|
isPublished: true,
|
|
isInstalled: isDisabled,
|
|
isDisabled: isDisabled,
|
|
isCore: plugin.internal,
|
|
isDev: false,
|
|
isEnterprise: status === 'enterprise',
|
|
type: typeCode,
|
|
error: error?.errorCode,
|
|
};
|
|
}
|
|
|
|
export function mapLocalToCatalog(plugin: LocalPlugin, error?: PluginError): CatalogPlugin {
|
|
const {
|
|
name,
|
|
info: { description, version, logos, updated, author },
|
|
id,
|
|
dev,
|
|
type,
|
|
signature,
|
|
signatureOrg,
|
|
signatureType,
|
|
hasUpdate,
|
|
accessControl,
|
|
} = plugin;
|
|
|
|
const isDisabled = !!error || isDisabledSecretsPlugin(type);
|
|
return {
|
|
description,
|
|
downloads: 0,
|
|
id,
|
|
info: { logos },
|
|
name,
|
|
orgName: author.name,
|
|
popularity: 0,
|
|
publishedAt: '',
|
|
signature: getPluginSignature({ local: plugin, error }),
|
|
signatureOrg,
|
|
signatureType,
|
|
updatedAt: updated,
|
|
installedVersion: version,
|
|
hasUpdate,
|
|
isInstalled: true,
|
|
isDisabled: isDisabled,
|
|
isCore: signature === 'internal',
|
|
isPublished: false,
|
|
isDev: Boolean(dev),
|
|
isEnterprise: false,
|
|
type,
|
|
error: error?.errorCode,
|
|
accessControl: accessControl,
|
|
};
|
|
}
|
|
|
|
// TODO: change the signature by removing the optionals for local and remote.
|
|
export function mapToCatalogPlugin(local?: LocalPlugin, remote?: RemotePlugin, error?: PluginError): CatalogPlugin {
|
|
const installedVersion = local?.info.version;
|
|
const id = remote?.slug || local?.id || '';
|
|
const type = local?.type || remote?.typeCode;
|
|
const isDisabled = !!error || isDisabledSecretsPlugin(type);
|
|
|
|
let logos = {
|
|
small: `/public/img/icn-${type}.svg`,
|
|
large: `/public/img/icn-${type}.svg`,
|
|
};
|
|
|
|
if (remote) {
|
|
logos = {
|
|
small: `https://grafana.com/api/plugins/${id}/versions/${remote.version}/logos/small`,
|
|
large: `https://grafana.com/api/plugins/${id}/versions/${remote.version}/logos/large`,
|
|
};
|
|
} else if (local && local.info.logos) {
|
|
logos = local.info.logos;
|
|
}
|
|
|
|
return {
|
|
description: local?.info.description || remote?.description || '',
|
|
downloads: remote?.downloads || 0,
|
|
hasUpdate: local?.hasUpdate || false,
|
|
id,
|
|
info: {
|
|
logos,
|
|
},
|
|
isCore: Boolean(remote?.internal || local?.signature === PluginSignatureStatus.internal),
|
|
isDev: Boolean(local?.dev),
|
|
isEnterprise: remote?.status === 'enterprise',
|
|
isInstalled: Boolean(local) || isDisabled,
|
|
isDisabled: isDisabled,
|
|
isPublished: true,
|
|
// TODO<check if we would like to keep preferring the remote version>
|
|
name: remote?.name || local?.name || '',
|
|
// TODO<check if we would like to keep preferring the remote version>
|
|
orgName: remote?.orgName || local?.info.author.name || '',
|
|
popularity: remote?.popularity || 0,
|
|
publishedAt: remote?.createdAt || '',
|
|
type,
|
|
signature: getPluginSignature({ local, remote, error }),
|
|
signatureOrg: local?.signatureOrg || remote?.versionSignedByOrgName,
|
|
signatureType: local?.signatureType || remote?.versionSignatureType || remote?.signatureType || undefined,
|
|
// TODO<check if we would like to keep preferring the remote version>
|
|
updatedAt: remote?.updatedAt || local?.info.updated || '',
|
|
installedVersion,
|
|
error: error?.errorCode,
|
|
// Only local plugins have access control metadata
|
|
accessControl: local?.accessControl,
|
|
};
|
|
}
|
|
|
|
export const getExternalManageLink = (pluginId: string) => `${config.pluginCatalogURL}${pluginId}`;
|
|
|
|
export enum Sorters {
|
|
nameAsc = 'nameAsc',
|
|
nameDesc = 'nameDesc',
|
|
updated = 'updated',
|
|
published = 'published',
|
|
downloads = 'downloads',
|
|
}
|
|
|
|
export const sortPlugins = (plugins: CatalogPlugin[], sortBy: Sorters) => {
|
|
const sorters: { [name: string]: (a: CatalogPlugin, b: CatalogPlugin) => number } = {
|
|
nameAsc: (a: CatalogPlugin, b: CatalogPlugin) => a.name.localeCompare(b.name),
|
|
nameDesc: (a: CatalogPlugin, b: CatalogPlugin) => b.name.localeCompare(a.name),
|
|
updated: (a: CatalogPlugin, b: CatalogPlugin) =>
|
|
dateTimeParse(b.updatedAt).valueOf() - dateTimeParse(a.updatedAt).valueOf(),
|
|
published: (a: CatalogPlugin, b: CatalogPlugin) =>
|
|
dateTimeParse(b.publishedAt).valueOf() - dateTimeParse(a.publishedAt).valueOf(),
|
|
downloads: (a: CatalogPlugin, b: CatalogPlugin) => b.downloads - a.downloads,
|
|
};
|
|
|
|
if (sorters[sortBy]) {
|
|
return plugins.sort(sorters[sortBy]);
|
|
}
|
|
|
|
return plugins;
|
|
};
|
|
|
|
function groupErrorsByPluginId(errors: PluginError[] = []): Record<string, PluginError | undefined> {
|
|
return errors.reduce((byId, error) => {
|
|
byId[error.pluginId] = error;
|
|
return byId;
|
|
}, {} as Record<string, PluginError | undefined>);
|
|
}
|
|
|
|
function getPluginSignature(options: {
|
|
local?: LocalPlugin;
|
|
remote?: RemotePlugin;
|
|
error?: PluginError;
|
|
}): PluginSignatureStatus {
|
|
const { error, local, remote } = options;
|
|
|
|
if (error) {
|
|
switch (error.errorCode) {
|
|
case PluginErrorCode.invalidSignature:
|
|
return PluginSignatureStatus.invalid;
|
|
case PluginErrorCode.missingSignature:
|
|
return PluginSignatureStatus.missing;
|
|
case PluginErrorCode.modifiedSignature:
|
|
return PluginSignatureStatus.modified;
|
|
}
|
|
}
|
|
|
|
if (local?.signature) {
|
|
return local.signature;
|
|
}
|
|
|
|
if (remote?.signatureType || remote?.versionSignatureType) {
|
|
return PluginSignatureStatus.valid;
|
|
}
|
|
|
|
return PluginSignatureStatus.missing;
|
|
}
|
|
|
|
// Updates the core Grafana config to have the correct list available panels
|
|
export const updatePanels = () =>
|
|
getBackendSrv()
|
|
.get('/api/frontend/settings')
|
|
.then((settings: Settings) => {
|
|
config.panels = settings.panels;
|
|
});
|
|
|
|
export function getLatestCompatibleVersion(versions: Version[] | undefined): Version | undefined {
|
|
if (!versions) {
|
|
return;
|
|
}
|
|
const [latest] = versions.filter((v) => Boolean(v.isCompatible));
|
|
|
|
return latest;
|
|
}
|
|
|
|
export const isInstallControlsEnabled = () => config.pluginAdminEnabled;
|
|
|
|
export const hasInstallControlWarning = (
|
|
plugin: CatalogPlugin,
|
|
isRemotePluginsAvailable: boolean,
|
|
latestCompatibleVersion?: Version
|
|
) => {
|
|
const isExternallyManaged = config.pluginAdminExternalManageEnabled;
|
|
const hasPermission = contextSrv.hasAccess(AccessControlAction.PluginsInstall, isGrafanaAdmin());
|
|
const isCompatible = Boolean(latestCompatibleVersion);
|
|
return (
|
|
plugin.type === PluginType.renderer ||
|
|
plugin.type === PluginType.secretsmanager ||
|
|
(plugin.isEnterprise && !featureEnabled('enterprise.plugins')) ||
|
|
plugin.isDev ||
|
|
(!hasPermission && !isExternallyManaged) ||
|
|
!plugin.isPublished ||
|
|
!isCompatible ||
|
|
!isRemotePluginsAvailable
|
|
);
|
|
};
|
|
|
|
export const isLocalPluginVisible = (p: LocalPlugin) => isPluginVisible(p.id);
|
|
|
|
export const isRemotePluginVisible = (p: RemotePlugin) => isPluginVisible(p.slug);
|
|
|
|
function isPluginVisible(id: string) {
|
|
const { pluginCatalogHiddenPlugins }: { pluginCatalogHiddenPlugins: string[] } = config;
|
|
|
|
return !pluginCatalogHiddenPlugins.includes(id);
|
|
}
|
|
|
|
function isDisabledSecretsPlugin(type?: PluginType): boolean {
|
|
return type === PluginType.secretsmanager && !config.secretsManagerPluginEnabled;
|
|
}
|
|
|
|
export function isLocalCorePlugin(local?: LocalPlugin): boolean {
|
|
return Boolean(local?.signature === 'internal');
|
|
}
|