123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263 |
- import capitalize from 'lodash/capitalize';
- import * as qs from 'query-string';
- import {Result} from 'sentry/components/forms/selectAsyncControl';
- import {
- IconAsana,
- IconBitbucket,
- IconGeneric,
- IconGithub,
- IconGitlab,
- IconJira,
- IconVsts,
- } from 'sentry/icons';
- import {t} from 'sentry/locale';
- import HookStore from 'sentry/stores/hookStore';
- import {
- AppOrProviderOrPlugin,
- DocIntegration,
- ExternalActorMapping,
- ExternalActorMappingOrSuggestion,
- Integration,
- IntegrationFeature,
- IntegrationInstallationStatus,
- IntegrationType,
- Organization,
- PluginWithProjectList,
- SentryApp,
- SentryAppInstallation,
- } from 'sentry/types';
- import {Hooks} from 'sentry/types/hooks';
- import {
- integrationEventMap,
- IntegrationEventParameters,
- } from 'sentry/utils/analytics/integrations';
- import makeAnalyticsFunction from 'sentry/utils/analytics/makeAnalyticsFunction';
- const mapIntegrationParams = analyticsParams => {
- // Reload expects integration_status even though it's not relevant for non-sentry apps
- // Passing in a dummy value of published in those cases
- const fullParams = {...analyticsParams};
- if (analyticsParams.integration && analyticsParams.integration_type !== 'sentry_app') {
- fullParams.integration_status = 'published';
- }
- return fullParams;
- };
- export const trackIntegrationAnalytics = makeAnalyticsFunction<
- IntegrationEventParameters,
- {organization: Organization} // org is required
- >(integrationEventMap, {
- mapValuesFn: mapIntegrationParams,
- });
- /**
- * In sentry.io the features list supports rendering plan details. If the hook
- * is not registered for rendering the features list like this simply show the
- * features as a normal list.
- */
- const generateFeaturesList = p => (
- <ul>
- {p.features.map((f, i) => (
- <li key={i}>{f.description}</li>
- ))}
- </ul>
- );
- const generateIntegrationFeatures = p =>
- p.children({
- disabled: false,
- disabledReason: null,
- ungatedFeatures: p.features,
- gatedFeatureGroups: [],
- });
- const defaultFeatureGateComponents: ReturnType<Hooks['integrations:feature-gates']> = {
- IntegrationFeatures: generateIntegrationFeatures,
- FeatureList: generateFeaturesList,
- };
- export const getIntegrationFeatureGate = () => {
- const defaultHook = () => defaultFeatureGateComponents;
- const featureHook = HookStore.get('integrations:feature-gates')[0] || defaultHook;
- return featureHook();
- };
- export const getSentryAppInstallStatus = (install: SentryAppInstallation | undefined) => {
- if (install) {
- return capitalize(install.status) as IntegrationInstallationStatus;
- }
- return 'Not Installed';
- };
- export const getCategories = (features: IntegrationFeature[]): string[] => {
- const transform = features.map(({featureGate}) => {
- const feature = featureGate
- .replace(/integrations/g, '')
- .replace(/-/g, ' ')
- .trim();
- switch (feature) {
- case 'actionable notification':
- return 'notification action';
- case 'issue basic':
- case 'issue link':
- case 'issue sync':
- case 'project management':
- return 'issue tracking';
- case 'commits':
- return 'source code management';
- case 'chat unfurl':
- return 'chat';
- default:
- return feature;
- }
- });
- return [...new Set(transform)];
- };
- export const getCategoriesForIntegration = (
- integration: AppOrProviderOrPlugin
- ): string[] => {
- if (isSentryApp(integration)) {
- return ['internal', 'unpublished'].includes(integration.status)
- ? [integration.status]
- : getCategories(integration.featureData);
- }
- if (isPlugin(integration)) {
- return getCategories(integration.featureDescriptions);
- }
- if (isDocIntegration(integration)) {
- return getCategories(integration.features ?? []);
- }
- return getCategories(integration.metadata.features);
- };
- export function isSentryApp(
- integration: AppOrProviderOrPlugin
- ): integration is SentryApp {
- return !!(integration as SentryApp).uuid;
- }
- export function isPlugin(
- integration: AppOrProviderOrPlugin
- ): integration is PluginWithProjectList {
- return integration.hasOwnProperty('shortName');
- }
- export function isDocIntegration(
- integration: AppOrProviderOrPlugin
- ): integration is DocIntegration {
- return integration.hasOwnProperty('isDraft');
- }
- export function isExternalActorMapping(
- mapping: ExternalActorMappingOrSuggestion
- ): mapping is ExternalActorMapping {
- return mapping.hasOwnProperty('id');
- }
- export const getIntegrationType = (
- integration: AppOrProviderOrPlugin
- ): IntegrationType => {
- if (isSentryApp(integration)) {
- return 'sentry_app';
- }
- if (isPlugin(integration)) {
- return 'plugin';
- }
- if (isDocIntegration(integration)) {
- return 'document';
- }
- return 'first_party';
- };
- export const convertIntegrationTypeToSnakeCase = (
- type: 'plugin' | 'firstParty' | 'sentryApp' | 'docIntegration'
- ) => {
- switch (type) {
- case 'firstParty':
- return 'first_party';
- case 'sentryApp':
- return 'sentry_app';
- case 'docIntegration':
- return 'document';
- default:
- return type;
- }
- };
- export const safeGetQsParam = (param: string) => {
- try {
- const query = qs.parse(window.location.search) || {};
- return query[param];
- } catch {
- return undefined;
- }
- };
- export const getIntegrationIcon = (integrationType?: string, size?: string) => {
- const iconSize = size || 'md';
- switch (integrationType) {
- case 'asana':
- return <IconAsana size={iconSize} />;
- case 'bitbucket':
- return <IconBitbucket size={iconSize} />;
- case 'gitlab':
- return <IconGitlab size={iconSize} />;
- case 'github':
- case 'github_enterprise':
- return <IconGithub size={iconSize} />;
- case 'jira':
- case 'jira_server':
- return <IconJira size={iconSize} />;
- case 'vsts':
- return <IconVsts size={iconSize} />;
- default:
- return <IconGeneric size={iconSize} />;
- }
- };
- // used for project creation and onboarding
- // determines what integration maps to what project platform
- export const platformToIntegrationMap = {
- 'node-awslambda': 'aws_lambda',
- 'python-awslambda': 'aws_lambda',
- };
- export const isSlackIntegrationUpToDate = (integrations: Integration[]): boolean => {
- return integrations.every(
- integration =>
- integration.provider.key !== 'slack' || integration.scopes?.includes('commands')
- );
- };
- export const getAlertText = (integrations?: Integration[]): string | undefined => {
- return isSlackIntegrationUpToDate(integrations || [])
- ? undefined
- : t(
- 'Update to the latest version of our Slack app to get access to personal and team notifications.'
- );
- };
- /**
- * Uses the mapping and baseEndpoint to derive the details for the mappings request.
- * @param baseEndpoint Must have a trailing slash, since the id is appended for PUT requests!
- * @param mapping The mapping or suggestion being sent to the endpoint
- * @returns An object containing the request method (apiMethod), and final endpoint (apiEndpoint)
- */
- export const getExternalActorEndpointDetails = (
- baseEndpoint: string,
- mapping?: ExternalActorMappingOrSuggestion
- ): {apiEndpoint: string; apiMethod: 'POST' | 'PUT'} => {
- const isValidMapping = mapping && isExternalActorMapping(mapping);
- return {
- apiMethod: isValidMapping ? 'PUT' : 'POST',
- apiEndpoint: isValidMapping ? `${baseEndpoint}${mapping.id}/` : baseEndpoint,
- };
- };
- export const sentryNameToOption = ({id, name}): Result => ({
- value: id,
- label: name,
- });
|