mirror of
https://github.com/Unleash/unleash.git
synced 2025-01-25 00:07:47 +01:00
feat: Teams addon for messaging on Microsoft teams (#814)
This commit is contained in:
parent
517f3e2170
commit
6c57aeb232
33
docs/addons/teams.md
Normal file
33
docs/addons/teams.md
Normal file
@ -0,0 +1,33 @@
|
||||
---
|
||||
id: teams
|
||||
title: Microsoft Teams
|
||||
---
|
||||
|
||||
> This feature was introduced in \_Unleash v4.0.x.
|
||||
|
||||
The MicrosoftTeams addon allows Unleash to post Updates when a feature toggle is updated. To set up this addon, you need to set up a webhook connector for your channel. You can follow [Creating an Incoming Webhook for a channel](https://docs.microsoft.com/en-us/microsoftteams/platform/webhooks-and-connectors/how-to/add-incoming-webhook) on how to do that.
|
||||
|
||||
The Microsoft Teams addon will perform a single retry if the HTTP POST against the Microsoft Teams Webhook URL fails (either a 50x or network error). Duplicate events may happen, and you should never assume events always comes in order.
|
||||
|
||||
## Configuration
|
||||
|
||||
#### Events
|
||||
|
||||
You can choose to trigger updates for the following events (we might add more event types in the future):
|
||||
|
||||
- feature-created
|
||||
- feature-updated
|
||||
- feature-archived
|
||||
- feature-revived
|
||||
- feature-stale-on
|
||||
- feature-stale-off
|
||||
|
||||
#### Parameters
|
||||
|
||||
Unleash Microsoft Teams addon takes the following parameters.
|
||||
|
||||
- **Microsoft Teams Webhook URL** - This is the only required property.
|
||||
|
||||
#### Tags
|
||||
|
||||
Microsoft teams's income webhooks are channel specific. You will be able to create multiple addons to support messaging on multiple channels.
|
29
snapshots/src/lib/addons/teams.test.js.md
Normal file
29
snapshots/src/lib/addons/teams.test.js.md
Normal file
@ -0,0 +1,29 @@
|
||||
# Snapshot report for `src/lib/addons/teams.test.js`
|
||||
|
||||
The actual snapshot is saved in `teams.test.js.snap`.
|
||||
|
||||
Generated by [AVA](https://avajs.dev).
|
||||
|
||||
## Should call slack webhook
|
||||
|
||||
> Snapshot 1
|
||||
|
||||
'{"themeColor":"0076D7","summary":"Message","sections":[{"activityTitle":"Feature toggle some-toggle | *Type*: undefined | *Project*: undefined <br /> *Activation strategies*: \\n- name: default\\n","activitySubtitle":"Unleash notification update","facts":[{"name":"User","value":"some@user.com"},{"name":"Action","value":"Create"},{"name":"Enabled","value":"*no*"}]}],"potentialAction":[{"@type":"OpenUri","name":"Go to feature","targets":[{"os":"default","uri":"http://some-url.com/#/features/strategies/some-toggle"}]}]}'
|
||||
|
||||
## Should call slack webhook for archived toggle
|
||||
|
||||
> Snapshot 1
|
||||
|
||||
'{"themeColor":"0076D7","summary":"Message","sections":[{"activityTitle":"The feature toggle *some-toggle* was *archived*","activitySubtitle":"Unleash notification update","facts":[{"name":"User","value":"some@user.com"},{"name":"Action","value":"feature-archived"},{"name":"Enabled","value":"*no*"}]}],"potentialAction":[{"@type":"OpenUri","name":"Go to feature","targets":[{"os":"default","uri":"http://some-url.com/#/archive/strategies/some-toggle"}]}]}'
|
||||
|
||||
## Should call teams webhook
|
||||
|
||||
> Snapshot 1
|
||||
|
||||
'{"themeColor":"0076D7","summary":"Message","sections":[{"activityTitle":"Feature toggle some-toggle | *Type*: undefined | *Project*: undefined <br /> *Activation strategies*: \\n- name: default\\n","activitySubtitle":"Unleash notification update","facts":[{"name":"User","value":"some@user.com"},{"name":"Action","value":"Create"},{"name":"Enabled","value":"*no*"}]}],"potentialAction":[{"@type":"OpenUri","name":"Go to feature","targets":[{"os":"default","uri":"http://some-url.com/#/features/strategies/some-toggle"}]}]}'
|
||||
|
||||
## Should call teams webhook for archived toggle
|
||||
|
||||
> Snapshot 1
|
||||
|
||||
'{"themeColor":"0076D7","summary":"Message","sections":[{"activityTitle":"The feature toggle *some-toggle* was *archived*","activitySubtitle":"Unleash notification update","facts":[{"name":"User","value":"some@user.com"},{"name":"Action","value":"feature-archived"},{"name":"Enabled","value":"*no*"}]}],"potentialAction":[{"@type":"OpenUri","name":"Go to feature","targets":[{"os":"default","uri":"http://some-url.com/#/archive/strategies/some-toggle"}]}]}'
|
BIN
snapshots/src/lib/addons/teams.test.js.snap
Normal file
BIN
snapshots/src/lib/addons/teams.test.js.snap
Normal file
Binary file not shown.
@ -1,6 +1,7 @@
|
||||
const webhook = require('./webhook');
|
||||
const slackAddon = require('./slack');
|
||||
const teamsAddon = require('./teams');
|
||||
|
||||
const addons = [webhook, slackAddon];
|
||||
const addons = [webhook, slackAddon, teamsAddon];
|
||||
|
||||
module.exports = addons;
|
||||
|
34
src/lib/addons/teams-definition.js
Normal file
34
src/lib/addons/teams-definition.js
Normal file
@ -0,0 +1,34 @@
|
||||
'use strict';
|
||||
|
||||
const {
|
||||
FEATURE_CREATED,
|
||||
FEATURE_UPDATED,
|
||||
FEATURE_ARCHIVED,
|
||||
FEATURE_REVIVED,
|
||||
FEATURE_STALE_ON,
|
||||
FEATURE_STALE_OFF,
|
||||
} = require('../event-type');
|
||||
|
||||
module.exports = {
|
||||
name: 'teams',
|
||||
displayName: 'Microsoft Teams',
|
||||
description: 'Allows Unleash to post updates to Microsoft Teams.',
|
||||
documentationUrl: 'https://docs.getunleash.io/docs/addons/teams',
|
||||
parameters: [
|
||||
{
|
||||
name: 'url',
|
||||
displayName: 'Microsoft Teams webhook URL',
|
||||
type: 'url',
|
||||
required: true,
|
||||
sensitive: true,
|
||||
},
|
||||
],
|
||||
events: [
|
||||
FEATURE_CREATED,
|
||||
FEATURE_UPDATED,
|
||||
FEATURE_ARCHIVED,
|
||||
FEATURE_REVIVED,
|
||||
FEATURE_STALE_ON,
|
||||
FEATURE_STALE_OFF,
|
||||
],
|
||||
};
|
126
src/lib/addons/teams.js
Normal file
126
src/lib/addons/teams.js
Normal file
@ -0,0 +1,126 @@
|
||||
'use strict';
|
||||
|
||||
const YAML = require('js-yaml');
|
||||
const Addon = require('./addon');
|
||||
|
||||
const {
|
||||
FEATURE_CREATED,
|
||||
FEATURE_UPDATED,
|
||||
FEATURE_ARCHIVED,
|
||||
FEATURE_REVIVED,
|
||||
FEATURE_STALE_ON,
|
||||
FEATURE_STALE_OFF,
|
||||
} = require('../event-type');
|
||||
|
||||
const definition = require('./teams-definition');
|
||||
|
||||
class TeamsAddon extends Addon {
|
||||
constructor(args) {
|
||||
super(definition, args);
|
||||
this.unleashUrl = args.unleashUrl;
|
||||
}
|
||||
|
||||
async handleEvent(event, parameters) {
|
||||
const { url } = parameters;
|
||||
const { createdBy, data, type } = event;
|
||||
let text = '';
|
||||
if ([FEATURE_ARCHIVED, FEATURE_REVIVED].includes(event.type)) {
|
||||
text = this.generateArchivedText(event);
|
||||
} else if ([FEATURE_STALE_ON, FEATURE_STALE_OFF].includes(event.type)) {
|
||||
text = this.generateStaleText(event);
|
||||
} else {
|
||||
text = this.generateText(event);
|
||||
}
|
||||
|
||||
const enabled = `*${data.enabled ? 'yes' : 'no'}*`;
|
||||
const stale = data.stale ? '("stale")' : '';
|
||||
const body = {
|
||||
themeColor: '0076D7',
|
||||
summary: 'Message',
|
||||
sections: [
|
||||
{
|
||||
activityTitle: text,
|
||||
activitySubtitle: 'Unleash notification update',
|
||||
facts: [
|
||||
{
|
||||
name: 'User',
|
||||
value: createdBy,
|
||||
},
|
||||
{
|
||||
name: 'Action',
|
||||
value: this.getAction(type),
|
||||
},
|
||||
{
|
||||
name: 'Enabled',
|
||||
value: `${enabled}${stale}`,
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
potentialAction: [
|
||||
{
|
||||
'@type': 'OpenUri',
|
||||
name: 'Go to feature',
|
||||
targets: [
|
||||
{
|
||||
os: 'default',
|
||||
uri: this.featureLink(event),
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
const requestOpts = {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify(body),
|
||||
};
|
||||
const result = await this.fetchRetry(url, requestOpts);
|
||||
this.logger.info(`Handled event ${event.type}. Status codes=${result}`);
|
||||
}
|
||||
|
||||
featureLink(event) {
|
||||
const path = event.type === FEATURE_ARCHIVED ? 'archive' : 'features';
|
||||
return `${this.unleashUrl}/#/${path}/strategies/${event.data.name}`;
|
||||
}
|
||||
|
||||
generateStaleText(event) {
|
||||
const { data, type } = event;
|
||||
const isStale = type === FEATURE_STALE_ON;
|
||||
if (isStale) {
|
||||
return `The feature toggle *${data.name}* is now *ready to be removed* from the code.`;
|
||||
}
|
||||
return `The feature toggle *${data.name}* was *unmarked* as stale`;
|
||||
}
|
||||
|
||||
generateArchivedText(event) {
|
||||
const { data, type } = event;
|
||||
const action = type === FEATURE_ARCHIVED ? 'archived' : 'revived';
|
||||
return `The feature toggle *${data.name}* was *${action}*`;
|
||||
}
|
||||
|
||||
generateText(event) {
|
||||
const { data } = event;
|
||||
const typeStr = `*Type*: ${data.type}`;
|
||||
const project = `*Project*: ${data.project}`;
|
||||
const strategies = `*Activation strategies*: \n${YAML.safeDump(
|
||||
data.strategies,
|
||||
{ skipInvalid: true },
|
||||
)}`;
|
||||
return `Feature toggle ${data.name} | ${typeStr} | ${project} <br /> ${strategies}`;
|
||||
}
|
||||
|
||||
getAction(type) {
|
||||
switch (type) {
|
||||
case FEATURE_CREATED:
|
||||
return 'Create';
|
||||
case FEATURE_UPDATED:
|
||||
return 'Update';
|
||||
default:
|
||||
return type;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = TeamsAddon;
|
67
src/lib/addons/teams.test.js
Normal file
67
src/lib/addons/teams.test.js
Normal file
@ -0,0 +1,67 @@
|
||||
const test = require('ava');
|
||||
const proxyquire = require('proxyquire').noCallThru();
|
||||
const { FEATURE_CREATED, FEATURE_ARCHIVED } = require('../event-type');
|
||||
|
||||
const TeamsAddon = proxyquire.load('./teams', {
|
||||
'./addon': class Addon {
|
||||
constructor(definition, { getLogger }) {
|
||||
this.logger = getLogger('addon/test');
|
||||
this.fetchRetryCalls = [];
|
||||
}
|
||||
|
||||
async fetchRetry(url, options, retries, backoff) {
|
||||
this.fetchRetryCalls.push({ url, options, retries, backoff });
|
||||
return Promise.resolve({ status: 200 });
|
||||
}
|
||||
},
|
||||
});
|
||||
|
||||
const noLogger = require('../../test/fixtures/no-logger');
|
||||
|
||||
test('Should call teams webhook', async t => {
|
||||
const addon = new TeamsAddon({
|
||||
getLogger: noLogger,
|
||||
unleashUrl: 'http://some-url.com',
|
||||
});
|
||||
const event = {
|
||||
type: FEATURE_CREATED,
|
||||
createdBy: 'some@user.com',
|
||||
data: {
|
||||
name: 'some-toggle',
|
||||
enabled: false,
|
||||
strategies: [{ name: 'default' }],
|
||||
},
|
||||
};
|
||||
|
||||
const parameters = {
|
||||
url: 'http://hooks.office.com',
|
||||
};
|
||||
|
||||
await addon.handleEvent(event, parameters);
|
||||
t.is(addon.fetchRetryCalls.length, 1);
|
||||
t.is(addon.fetchRetryCalls[0].url, parameters.url);
|
||||
t.snapshot(addon.fetchRetryCalls[0].options.body);
|
||||
});
|
||||
|
||||
test('Should call teams webhook for archived toggle', async t => {
|
||||
const addon = new TeamsAddon({
|
||||
getLogger: noLogger,
|
||||
unleashUrl: 'http://some-url.com',
|
||||
});
|
||||
const event = {
|
||||
type: FEATURE_ARCHIVED,
|
||||
createdBy: 'some@user.com',
|
||||
data: {
|
||||
name: 'some-toggle',
|
||||
},
|
||||
};
|
||||
|
||||
const parameters = {
|
||||
url: 'http://hooks.office.com',
|
||||
};
|
||||
|
||||
await addon.handleEvent(event, parameters);
|
||||
t.is(addon.fetchRetryCalls.length, 1);
|
||||
t.is(addon.fetchRetryCalls[0].url, parameters.url);
|
||||
t.snapshot(addon.fetchRetryCalls[0].options.body);
|
||||
});
|
@ -29,7 +29,7 @@ test.serial('gets all addons', async t => {
|
||||
.expect(200)
|
||||
.expect(res => {
|
||||
t.is(res.body.addons.length, 0, 'expected 0 configured addons');
|
||||
t.is(res.body.providers.length, 2, 'expected 2 addon providers');
|
||||
t.is(res.body.providers.length, 3, 'expected 3 addon providers');
|
||||
t.is(res.body.providers[0].name, 'webhook');
|
||||
});
|
||||
});
|
||||
|
Loading…
Reference in New Issue
Block a user