2023-07-20 14:37:06 +02:00
|
|
|
import {
|
|
|
|
WebClient,
|
|
|
|
ConversationsListResponse,
|
|
|
|
ErrorCode,
|
|
|
|
WebClientEvent,
|
|
|
|
CodedError,
|
|
|
|
WebAPIPlatformError,
|
|
|
|
WebAPIRequestError,
|
|
|
|
WebAPIRateLimitedError,
|
|
|
|
WebAPIHTTPError,
|
|
|
|
} from '@slack/web-api';
|
2023-07-14 10:49:34 +02:00
|
|
|
import Addon from './addon';
|
|
|
|
|
|
|
|
import slackAppDefinition from './slack-app-definition';
|
|
|
|
import { IAddonConfig } from '../types/model';
|
|
|
|
|
|
|
|
import {
|
|
|
|
FeatureEventFormatter,
|
|
|
|
FeatureEventFormatterMd,
|
|
|
|
LinkStyle,
|
|
|
|
} from './feature-event-formatter-md';
|
|
|
|
import { IEvent } from '../types/events';
|
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
const CACHE_SECONDS = 30;
|
|
|
|
|
2023-07-14 10:49:34 +02:00
|
|
|
interface ISlackAppAddonParameters {
|
|
|
|
accessToken: string;
|
2023-07-21 15:15:43 +02:00
|
|
|
defaultChannels: string;
|
2023-07-14 10:49:34 +02:00
|
|
|
}
|
2023-07-20 14:37:06 +02:00
|
|
|
|
2023-07-14 10:49:34 +02:00
|
|
|
export default class SlackAppAddon extends Addon {
|
|
|
|
private msgFormatter: FeatureEventFormatter;
|
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
private accessToken?: string;
|
|
|
|
|
2023-07-14 10:49:34 +02:00
|
|
|
private slackClient?: WebClient;
|
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
private slackChannels?: ConversationsListResponse['channels'];
|
|
|
|
|
|
|
|
private slackChannelsCacheTimeout?: NodeJS.Timeout;
|
|
|
|
|
2023-07-14 10:49:34 +02:00
|
|
|
constructor(args: IAddonConfig) {
|
|
|
|
super(slackAppDefinition, args);
|
|
|
|
this.msgFormatter = new FeatureEventFormatterMd(
|
|
|
|
args.unleashUrl,
|
|
|
|
LinkStyle.SLACK,
|
|
|
|
);
|
2023-07-20 14:37:06 +02:00
|
|
|
this.startCacheInvalidation();
|
2023-07-14 10:49:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async handleEvent(
|
|
|
|
event: IEvent,
|
|
|
|
parameters: ISlackAppAddonParameters,
|
|
|
|
): Promise<void> {
|
2023-07-20 14:37:06 +02:00
|
|
|
try {
|
2023-07-21 15:15:43 +02:00
|
|
|
const { accessToken, defaultChannels } = parameters;
|
2023-07-20 14:37:06 +02:00
|
|
|
if (!accessToken) {
|
|
|
|
this.logger.warn('No access token provided.');
|
|
|
|
return;
|
|
|
|
}
|
2023-07-14 10:49:34 +02:00
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
const taggedChannels = this.findTaggedChannels(event);
|
2023-07-21 15:15:43 +02:00
|
|
|
const eventChannels = taggedChannels.length
|
|
|
|
? taggedChannels
|
|
|
|
: this.getDefaultChannels(defaultChannels);
|
|
|
|
|
|
|
|
if (!eventChannels.length) {
|
2023-07-20 14:37:06 +02:00
|
|
|
this.logger.debug(
|
2023-07-21 15:15:43 +02:00
|
|
|
`No Slack channels found for event ${event.type}.`,
|
2023-07-20 14:37:06 +02:00
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
2023-08-02 15:03:51 +02:00
|
|
|
this.logger.debug(`Found candidate channels: ${eventChannels}.`);
|
2023-07-14 10:49:34 +02:00
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
if (!this.slackClient || this.accessToken !== accessToken) {
|
|
|
|
const client = new WebClient(accessToken);
|
|
|
|
client.on(WebClientEvent.RATE_LIMITED, (numSeconds) => {
|
|
|
|
this.logger.debug(
|
|
|
|
`Rate limit reached for event ${event.type}. Retry scheduled after ${numSeconds} seconds`,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
this.slackClient = client;
|
|
|
|
this.accessToken = accessToken;
|
|
|
|
}
|
2023-07-14 10:49:34 +02:00
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
if (!this.slackChannels) {
|
|
|
|
const slackConversationsList =
|
|
|
|
await this.slackClient.conversations.list({
|
|
|
|
types: 'public_channel,private_channel',
|
2023-08-08 12:54:32 +02:00
|
|
|
exclude_archived: true,
|
|
|
|
limit: 200,
|
2023-07-20 14:37:06 +02:00
|
|
|
});
|
|
|
|
this.slackChannels = slackConversationsList.channels || [];
|
2023-08-08 12:54:32 +02:00
|
|
|
let nextCursor =
|
|
|
|
slackConversationsList.response_metadata?.next_cursor;
|
|
|
|
while (nextCursor !== undefined && nextCursor !== '') {
|
|
|
|
this.logger.debug('Fetching next page of channels');
|
|
|
|
const moreChannels =
|
|
|
|
await this.slackClient.conversations.list({
|
|
|
|
cursor: nextCursor,
|
|
|
|
types: 'public_channel,private_channel',
|
|
|
|
exclude_archived: true,
|
|
|
|
limit: 200,
|
|
|
|
});
|
|
|
|
const channels = moreChannels.channels;
|
|
|
|
if (channels === undefined) {
|
|
|
|
this.logger.debug(
|
|
|
|
'Channels list was empty, breaking pagination',
|
|
|
|
);
|
|
|
|
nextCursor = undefined;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
nextCursor = moreChannels.response_metadata?.next_cursor;
|
|
|
|
this.logger.debug(
|
|
|
|
`This page had ${channels.length} channels`,
|
|
|
|
);
|
|
|
|
|
|
|
|
channels.forEach((channel) =>
|
|
|
|
this.slackChannels?.push(channel),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
this.logger.debug(
|
2023-08-02 15:03:51 +02:00
|
|
|
`Fetched ${
|
|
|
|
this.slackChannels.length
|
|
|
|
} available Slack channels: ${this.slackChannels.map(
|
|
|
|
({ name }) => name,
|
|
|
|
)}`,
|
2023-07-20 14:37:06 +02:00
|
|
|
);
|
|
|
|
}
|
2023-07-14 10:49:34 +02:00
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
const currentSlackChannels = [...this.slackChannels];
|
|
|
|
if (!currentSlackChannels.length) {
|
2023-08-02 15:03:51 +02:00
|
|
|
this.logger.warn('No available Slack channels found.');
|
2023-07-20 14:37:06 +02:00
|
|
|
return;
|
|
|
|
}
|
2023-07-14 10:49:34 +02:00
|
|
|
|
|
|
|
const text = this.msgFormatter.format(event);
|
2023-07-20 14:37:06 +02:00
|
|
|
const url = this.msgFormatter.featureLink(event);
|
|
|
|
|
|
|
|
const slackChannelsToPostTo = currentSlackChannels.filter(
|
2023-07-21 15:15:43 +02:00
|
|
|
({ id, name }) => id && name && eventChannels.includes(name),
|
2023-07-20 14:37:06 +02:00
|
|
|
);
|
2023-07-14 10:49:34 +02:00
|
|
|
|
2023-08-02 15:03:51 +02:00
|
|
|
if (!slackChannelsToPostTo.length) {
|
|
|
|
this.logger.info('No eligible Slack channel found.');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.logger.debug(
|
|
|
|
`Posting event to ${slackChannelsToPostTo.map(
|
|
|
|
({ name }) => name,
|
|
|
|
)}.`,
|
|
|
|
);
|
|
|
|
|
2023-07-14 10:49:34 +02:00
|
|
|
const requests = slackChannelsToPostTo.map(({ id }) =>
|
|
|
|
this.slackClient!.chat.postMessage({
|
|
|
|
channel: id!,
|
|
|
|
text,
|
|
|
|
attachments: [
|
|
|
|
{
|
|
|
|
actions: [
|
|
|
|
{
|
|
|
|
name: 'featureToggle',
|
|
|
|
text: 'Open in Unleash',
|
|
|
|
type: 'button',
|
|
|
|
value: 'featureToggle',
|
|
|
|
style: 'primary',
|
2023-07-20 14:37:06 +02:00
|
|
|
url,
|
2023-07-14 10:49:34 +02:00
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
const results = await Promise.allSettled(requests);
|
|
|
|
|
|
|
|
results
|
|
|
|
.filter(({ status }) => status === 'rejected')
|
|
|
|
.map(({ reason }: PromiseRejectedResult) =>
|
|
|
|
this.logError(event, reason),
|
|
|
|
);
|
|
|
|
|
|
|
|
this.logger.info(
|
|
|
|
`Handled event ${event.type} dispatching ${
|
|
|
|
results.filter(({ status }) => status === 'fulfilled')
|
|
|
|
.length
|
|
|
|
} out of ${requests.length} messages successfully.`,
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
this.logError(event, error);
|
2023-07-14 10:49:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
findTaggedChannels({ tags }: Pick<IEvent, 'tags'>): string[] {
|
|
|
|
if (tags) {
|
|
|
|
return tags
|
|
|
|
.filter((tag) => tag.type === 'slack')
|
|
|
|
.map((t) => t.value);
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
2023-07-20 14:37:06 +02:00
|
|
|
|
2023-07-21 15:15:43 +02:00
|
|
|
getDefaultChannels(defaultChannels?: string): string[] {
|
|
|
|
if (defaultChannels) {
|
|
|
|
return defaultChannels.split(',').map((c) => c.trim());
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2023-07-20 14:37:06 +02:00
|
|
|
startCacheInvalidation(): void {
|
|
|
|
this.slackChannelsCacheTimeout = setInterval(() => {
|
|
|
|
this.slackChannels = undefined;
|
|
|
|
}, CACHE_SECONDS * 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
logError(event: IEvent, error: Error | CodedError): void {
|
|
|
|
if (!('code' in error)) {
|
|
|
|
this.logger.warn(`Error handling event ${event.type}.`, error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (error.code === ErrorCode.PlatformError) {
|
|
|
|
const { data } = error as WebAPIPlatformError;
|
|
|
|
this.logger.warn(
|
|
|
|
`Error handling event ${event.type}. A platform error occurred: ${data}`,
|
|
|
|
error,
|
|
|
|
);
|
|
|
|
} else if (error.code === ErrorCode.RequestError) {
|
|
|
|
const { original } = error as WebAPIRequestError;
|
|
|
|
this.logger.warn(
|
|
|
|
`Error handling event ${event.type}. A request error occurred: ${original}`,
|
|
|
|
error,
|
|
|
|
);
|
|
|
|
} else if (error.code === ErrorCode.RateLimitedError) {
|
|
|
|
const { retryAfter } = error as WebAPIRateLimitedError;
|
|
|
|
this.logger.warn(
|
|
|
|
`Error handling event ${event.type}. A rate limit error occurred: retry after ${retryAfter} seconds`,
|
|
|
|
error,
|
|
|
|
);
|
|
|
|
} else if (error.code === ErrorCode.HTTPError) {
|
|
|
|
const { statusCode } = error as WebAPIHTTPError;
|
|
|
|
this.logger.warn(
|
|
|
|
`Error handling event ${event.type}. An HTTP error occurred: status code ${statusCode}`,
|
|
|
|
error,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
this.logger.warn(`Error handling event ${event.type}.`, error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
destroy(): void {
|
|
|
|
if (this.slackChannelsCacheTimeout) {
|
|
|
|
clearInterval(this.slackChannelsCacheTimeout);
|
|
|
|
this.slackChannelsCacheTimeout = undefined;
|
|
|
|
}
|
|
|
|
}
|
2023-07-14 10:49:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = SlackAppAddon;
|