import Addon from './addon'; import slackDefinition from './slack-definition'; import { type IAddonConfig, type IFlagResolver, serializeDates, } from '../types'; import { type FeatureEventFormatter, FeatureEventFormatterMd, LinkStyle, } from './feature-event-formatter-md'; import type { IEvent } from '../types/events'; import type { IntegrationEventState } from '../features/integration-events/integration-events-store'; interface ISlackAddonParameters { url: string; username?: string; defaultChannel: string; emojiIcon?: string; customHeaders?: string; } export default class SlackAddon extends Addon { private msgFormatter: FeatureEventFormatter; flagResolver: IFlagResolver; constructor(args: IAddonConfig) { super(slackDefinition, args); this.msgFormatter = new FeatureEventFormatterMd({ unleashUrl: args.unleashUrl, linkStyle: LinkStyle.SLACK, }); this.flagResolver = args.flagResolver; } // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types async handleEvent( event: IEvent, parameters: ISlackAddonParameters, integrationId: number, ): Promise { let state: IntegrationEventState = 'success'; const stateDetails: string[] = []; const { url, defaultChannel, username = 'Unleash', emojiIcon = ':unleash:', customHeaders, } = parameters; const slackChannels = this.findSlackChannels(event); if (slackChannels.length === 0) { slackChannels.push(defaultChannel); } let extraHeaders = {}; if (typeof customHeaders === 'string' && customHeaders.length > 1) { try { extraHeaders = JSON.parse(customHeaders); } catch (e) { state = 'successWithErrors'; const badHeadersMessage = 'Could not parse the JSON in the customHeaders parameter.'; stateDetails.push(badHeadersMessage); this.logger.warn(badHeadersMessage); } } const { text: formattedMessage, url: featureLink } = this.msgFormatter.format(event); const maxLength = 3000; const text = formattedMessage.substring(0, maxLength); const requests = slackChannels.map((channel) => { const body = { username, icon_emoji: emojiIcon, // eslint-disable-line camelcase text, channel: `#${channel}`, attachments: [ { actions: [ { name: 'featureToggle', text: 'Open in Unleash', type: 'button', value: 'featureToggle', style: 'primary', url: featureLink, }, ], }, ], }; const requestOpts = { method: 'POST', headers: { 'Content-Type': 'application/json', ...extraHeaders, }, body: JSON.stringify(body), }; return this.fetchRetry(url, requestOpts); }); const results = await Promise.all(requests); const failedRequests = results.filter((res) => !res.ok); const codes = this.getUniqueArray( results.map((res) => res.status), ).join(', '); this.logger.info(`Handled event ${event.type}.`); if (failedRequests.length === 0) { const successMessage = `All (${results.length}) Slack webhook requests were successful with status codes: ${codes}.`; stateDetails.push(successMessage); this.logger.info(successMessage); } else if (failedRequests.length === results.length) { state = 'failed'; const failedMessage = `All (${results.length}) Slack webhook requests failed with status codes: ${codes}.`; stateDetails.push(failedMessage); this.logger.warn(failedMessage); } else { state = 'successWithErrors'; const successWithErrorsMessage = `Some (${failedRequests.length} of ${results.length}) Slack webhook requests failed. Status codes: ${codes}.`; stateDetails.push(successWithErrorsMessage); this.logger.warn(successWithErrorsMessage); } this.registerEvent({ integrationId, state, stateDetails: stateDetails.join('\n'), event: serializeDates(event), details: { url, channels: slackChannels, username, message: `${formattedMessage}${text.length < formattedMessage.length ? ` (trimmed to ${maxLength} characters)` : ''}`, }, }); } getUniqueArray(arr: T[]): T[] { return [...new Set(arr)]; } findSlackChannels({ tags }: Pick): string[] { if (tags) { return tags .filter((tag) => tag.type === 'slack') .map((t) => t.value); } return []; } } module.exports = SlackAddon;