feat(notifications): Webhook Notifications (#632)
parent
1aa0005b42
commit
a7cc7c5975
@ -0,0 +1,139 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import { get } from 'lodash';
|
||||||
|
import { hasNotificationType, Notification } from '..';
|
||||||
|
import { MediaStatus } from '../../../constants/media';
|
||||||
|
import logger from '../../../logger';
|
||||||
|
import { getSettings, NotificationAgentWebhook } from '../../settings';
|
||||||
|
import { BaseAgent, NotificationAgent, NotificationPayload } from './agent';
|
||||||
|
|
||||||
|
type KeyMapFunction = (
|
||||||
|
payload: NotificationPayload,
|
||||||
|
type: Notification
|
||||||
|
) => string;
|
||||||
|
|
||||||
|
const KeyMap: Record<string, string | KeyMapFunction> = {
|
||||||
|
notification_type: (_payload, type) => Notification[type],
|
||||||
|
subject: 'subject',
|
||||||
|
message: 'message',
|
||||||
|
image: 'image',
|
||||||
|
notifyuser_username: 'notifyUser.username',
|
||||||
|
notifyuser_email: 'notifyUser.email',
|
||||||
|
notifyuser_avatar: 'notifyUser.avatar',
|
||||||
|
media_tmdbid: 'media.tmdbId',
|
||||||
|
media_imdbid: 'media.imdbId',
|
||||||
|
media_tvdbid: 'media.tvdbId',
|
||||||
|
media_type: 'media.mediaType',
|
||||||
|
media_status: (payload) =>
|
||||||
|
payload.media?.status ? MediaStatus[payload.media?.status] : '',
|
||||||
|
media_status4k: (payload) =>
|
||||||
|
payload.media?.status ? MediaStatus[payload.media?.status4k] : '',
|
||||||
|
};
|
||||||
|
|
||||||
|
class WebhookAgent
|
||||||
|
extends BaseAgent<NotificationAgentWebhook>
|
||||||
|
implements NotificationAgent {
|
||||||
|
protected getSettings(): NotificationAgentWebhook {
|
||||||
|
if (this.settings) {
|
||||||
|
return this.settings;
|
||||||
|
}
|
||||||
|
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
return settings.notifications.agents.webhook;
|
||||||
|
}
|
||||||
|
|
||||||
|
private parseKeys(
|
||||||
|
finalPayload: Record<string, unknown>,
|
||||||
|
payload: NotificationPayload,
|
||||||
|
type: Notification
|
||||||
|
): Record<string, unknown> {
|
||||||
|
Object.keys(finalPayload).forEach((key) => {
|
||||||
|
if (key === '{{extra}}') {
|
||||||
|
finalPayload.extra = payload.extra ?? [];
|
||||||
|
delete finalPayload[key];
|
||||||
|
key = 'extra';
|
||||||
|
} else if (key === '{{media}}') {
|
||||||
|
if (payload.media) {
|
||||||
|
finalPayload.media = finalPayload[key];
|
||||||
|
} else {
|
||||||
|
finalPayload.media = null;
|
||||||
|
}
|
||||||
|
delete finalPayload[key];
|
||||||
|
key = 'media';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof finalPayload[key] === 'string') {
|
||||||
|
Object.keys(KeyMap).forEach((keymapKey) => {
|
||||||
|
const keymapValue = KeyMap[keymapKey as keyof typeof KeyMap];
|
||||||
|
finalPayload[key] = (finalPayload[key] as string).replace(
|
||||||
|
`{{${keymapKey}}}`,
|
||||||
|
typeof keymapValue === 'function'
|
||||||
|
? keymapValue(payload, type)
|
||||||
|
: get(payload, keymapValue) ?? ''
|
||||||
|
);
|
||||||
|
});
|
||||||
|
} else if (finalPayload[key] && typeof finalPayload[key] === 'object') {
|
||||||
|
finalPayload[key] = this.parseKeys(
|
||||||
|
finalPayload[key] as Record<string, unknown>,
|
||||||
|
payload,
|
||||||
|
type
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return finalPayload;
|
||||||
|
}
|
||||||
|
|
||||||
|
private buildPayload(type: Notification, payload: NotificationPayload) {
|
||||||
|
const payloadString = Buffer.from(
|
||||||
|
this.getSettings().options.jsonPayload,
|
||||||
|
'base64'
|
||||||
|
).toString('ascii');
|
||||||
|
|
||||||
|
const parsedJSON = JSON.parse(JSON.parse(payloadString));
|
||||||
|
|
||||||
|
return this.parseKeys(parsedJSON, payload, type);
|
||||||
|
}
|
||||||
|
|
||||||
|
public shouldSend(type: Notification): boolean {
|
||||||
|
if (
|
||||||
|
this.getSettings().enabled &&
|
||||||
|
this.getSettings().options.webhookUrl &&
|
||||||
|
hasNotificationType(type, this.getSettings().types)
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async send(
|
||||||
|
type: Notification,
|
||||||
|
payload: NotificationPayload
|
||||||
|
): Promise<boolean> {
|
||||||
|
logger.debug('Sending webhook notification', { label: 'Notifications' });
|
||||||
|
try {
|
||||||
|
const { webhookUrl, authHeader } = this.getSettings().options;
|
||||||
|
|
||||||
|
if (!webhookUrl) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
await axios.post(webhookUrl, this.buildPayload(type, payload), {
|
||||||
|
headers: {
|
||||||
|
Authorization: authHeader,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
logger.error('Error sending Webhook notification', {
|
||||||
|
label: 'Notifications',
|
||||||
|
errorMessage: e.message,
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default WebhookAgent;
|
@ -0,0 +1,265 @@
|
|||||||
|
import { Router } from 'express';
|
||||||
|
import { getSettings } from '../../lib/settings';
|
||||||
|
import { Notification } from '../../lib/notifications';
|
||||||
|
import DiscordAgent from '../../lib/notifications/agents/discord';
|
||||||
|
import EmailAgent from '../../lib/notifications/agents/email';
|
||||||
|
import SlackAgent from '../../lib/notifications/agents/slack';
|
||||||
|
import TelegramAgent from '../../lib/notifications/agents/telegram';
|
||||||
|
import PushoverAgent from '../../lib/notifications/agents/pushover';
|
||||||
|
import WebhookAgent from '../../lib/notifications/agents/webhook';
|
||||||
|
|
||||||
|
const notificationRoutes = Router();
|
||||||
|
|
||||||
|
notificationRoutes.get('/discord', (_req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.discord);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/discord', (req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
settings.notifications.agents.discord = req.body;
|
||||||
|
settings.save();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.discord);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/discord/test', (req, res, next) => {
|
||||||
|
if (!req.user) {
|
||||||
|
return next({
|
||||||
|
status: 500,
|
||||||
|
message: 'User information missing from request',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const discordAgent = new DiscordAgent(req.body);
|
||||||
|
discordAgent.send(Notification.TEST_NOTIFICATION, {
|
||||||
|
notifyUser: req.user,
|
||||||
|
subject: 'Test Notification',
|
||||||
|
message:
|
||||||
|
'This is a test notification! Check check, 1, 2, 3. Are we coming in clear?',
|
||||||
|
});
|
||||||
|
|
||||||
|
return res.status(204).send();
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.get('/slack', (_req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.slack);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/slack', (req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
settings.notifications.agents.slack = req.body;
|
||||||
|
settings.save();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.slack);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/slack/test', (req, res, next) => {
|
||||||
|
if (!req.user) {
|
||||||
|
return next({
|
||||||
|
status: 500,
|
||||||
|
message: 'User information missing from request',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const slackAgent = new SlackAgent(req.body);
|
||||||
|
slackAgent.send(Notification.TEST_NOTIFICATION, {
|
||||||
|
notifyUser: req.user,
|
||||||
|
subject: 'Test Notification',
|
||||||
|
message:
|
||||||
|
'This is a test notification! Check check, 1, 2, 3. Are we coming in clear?',
|
||||||
|
});
|
||||||
|
|
||||||
|
return res.status(204).send();
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.get('/telegram', (_req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.telegram);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/telegram', (req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
settings.notifications.agents.telegram = req.body;
|
||||||
|
settings.save();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.telegram);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/telegram/test', (req, res, next) => {
|
||||||
|
if (!req.user) {
|
||||||
|
return next({
|
||||||
|
status: 500,
|
||||||
|
message: 'User information missing from request',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const telegramAgent = new TelegramAgent(req.body);
|
||||||
|
telegramAgent.send(Notification.TEST_NOTIFICATION, {
|
||||||
|
notifyUser: req.user,
|
||||||
|
subject: 'Test Notification',
|
||||||
|
message:
|
||||||
|
'This is a test notification! Check check, 1, 2, 3. Are we coming in clear?',
|
||||||
|
});
|
||||||
|
|
||||||
|
return res.status(204).send();
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.get('/pushover', (_req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.pushover);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/pushover', (req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
settings.notifications.agents.pushover = req.body;
|
||||||
|
settings.save();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.pushover);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/pushover/test', (req, res, next) => {
|
||||||
|
if (!req.user) {
|
||||||
|
return next({
|
||||||
|
status: 500,
|
||||||
|
message: 'User information missing from request',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const pushoverAgent = new PushoverAgent(req.body);
|
||||||
|
pushoverAgent.send(Notification.TEST_NOTIFICATION, {
|
||||||
|
notifyUser: req.user,
|
||||||
|
subject: 'Test Notification',
|
||||||
|
message:
|
||||||
|
'This is a test notification! Check check, 1, 2, 3. Are we coming in clear?',
|
||||||
|
});
|
||||||
|
|
||||||
|
return res.status(204).send();
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.get('/email', (_req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.email);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/email', (req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
settings.notifications.agents.email = req.body;
|
||||||
|
settings.save();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.email);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/email/test', (req, res, next) => {
|
||||||
|
if (!req.user) {
|
||||||
|
return next({
|
||||||
|
status: 500,
|
||||||
|
message: 'User information missing from request',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const emailAgent = new EmailAgent(req.body);
|
||||||
|
emailAgent.send(Notification.TEST_NOTIFICATION, {
|
||||||
|
notifyUser: req.user,
|
||||||
|
subject: 'Test Notification',
|
||||||
|
message:
|
||||||
|
'This is a test notification! Check check, 1, 2, 3. Are we coming in clear?',
|
||||||
|
});
|
||||||
|
|
||||||
|
return res.status(204).send();
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.get('/webhook', (_req, res) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
|
||||||
|
const webhookSettings = settings.notifications.agents.webhook;
|
||||||
|
|
||||||
|
const response: typeof webhookSettings = {
|
||||||
|
enabled: webhookSettings.enabled,
|
||||||
|
types: webhookSettings.types,
|
||||||
|
options: {
|
||||||
|
...webhookSettings.options,
|
||||||
|
jsonPayload: JSON.parse(
|
||||||
|
Buffer.from(webhookSettings.options.jsonPayload, 'base64').toString(
|
||||||
|
'ascii'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
res.status(200).json(response);
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/webhook', (req, res, next) => {
|
||||||
|
const settings = getSettings();
|
||||||
|
try {
|
||||||
|
JSON.parse(req.body.options.jsonPayload);
|
||||||
|
|
||||||
|
settings.notifications.agents.webhook = {
|
||||||
|
enabled: req.body.enabled,
|
||||||
|
types: req.body.types,
|
||||||
|
options: {
|
||||||
|
jsonPayload: Buffer.from(req.body.options.jsonPayload).toString(
|
||||||
|
'base64'
|
||||||
|
),
|
||||||
|
webhookUrl: req.body.options.webhookUrl,
|
||||||
|
authHeader: req.body.options.authHeader,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
settings.save();
|
||||||
|
|
||||||
|
res.status(200).json(settings.notifications.agents.webhook);
|
||||||
|
} catch (e) {
|
||||||
|
next({ status: 500, message: e.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
notificationRoutes.post('/webhook/test', (req, res, next) => {
|
||||||
|
if (!req.user) {
|
||||||
|
return next({
|
||||||
|
status: 500,
|
||||||
|
message: 'User information missing from request',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
JSON.parse(req.body.options.jsonPayload);
|
||||||
|
|
||||||
|
const testBody = {
|
||||||
|
enabled: req.body.enabled,
|
||||||
|
types: req.body.types,
|
||||||
|
options: {
|
||||||
|
jsonPayload: Buffer.from(req.body.options.jsonPayload).toString(
|
||||||
|
'base64'
|
||||||
|
),
|
||||||
|
webhookUrl: req.body.options.webhookUrl,
|
||||||
|
authHeader: req.body.options.authHeader,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const webhookAgent = new WebhookAgent(testBody);
|
||||||
|
webhookAgent.send(Notification.TEST_NOTIFICATION, {
|
||||||
|
notifyUser: req.user,
|
||||||
|
subject: 'Test Notification',
|
||||||
|
message:
|
||||||
|
'This is a test notification! Check check, 1, 2, 3. Are we coming in clear?',
|
||||||
|
});
|
||||||
|
|
||||||
|
return res.status(204).send();
|
||||||
|
} catch (e) {
|
||||||
|
next({ status: 500, message: e.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
export default notificationRoutes;
|
After Width: | Height: | Size: 257 B |
@ -0,0 +1,35 @@
|
|||||||
|
import React, { HTMLAttributes } from 'react';
|
||||||
|
import AceEditor from 'react-ace';
|
||||||
|
import 'ace-builds/src-noconflict/mode-json';
|
||||||
|
import 'ace-builds/src-noconflict/theme-dracula';
|
||||||
|
|
||||||
|
interface JSONEditorProps extends HTMLAttributes<HTMLDivElement> {
|
||||||
|
name: string;
|
||||||
|
value: string;
|
||||||
|
onUpdate: (value: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const JSONEditor: React.FC<JSONEditorProps> = ({
|
||||||
|
name,
|
||||||
|
value,
|
||||||
|
onUpdate,
|
||||||
|
onBlur,
|
||||||
|
}) => {
|
||||||
|
return (
|
||||||
|
<div className="w-full overflow-hidden rounded-md">
|
||||||
|
<AceEditor
|
||||||
|
mode="json"
|
||||||
|
theme="dracula"
|
||||||
|
onChange={onUpdate}
|
||||||
|
name={name}
|
||||||
|
editorProps={{ $blockScrolling: true }}
|
||||||
|
value={value}
|
||||||
|
onBlur={onBlur}
|
||||||
|
height="300px"
|
||||||
|
width="100%"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default JSONEditor;
|
@ -0,0 +1,315 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Field, Form, Formik } from 'formik';
|
||||||
|
import dynamic from 'next/dynamic';
|
||||||
|
import useSWR from 'swr';
|
||||||
|
import LoadingSpinner from '../../../Common/LoadingSpinner';
|
||||||
|
import Button from '../../../Common/Button';
|
||||||
|
import { defineMessages, useIntl } from 'react-intl';
|
||||||
|
import axios from 'axios';
|
||||||
|
import * as Yup from 'yup';
|
||||||
|
import { useToasts } from 'react-toast-notifications';
|
||||||
|
import NotificationTypeSelector from '../../../NotificationTypeSelector';
|
||||||
|
|
||||||
|
const JSONEditor = dynamic(() => import('../../../JSONEditor'), { ssr: false });
|
||||||
|
|
||||||
|
const defaultPayload = {
|
||||||
|
notification_type: '{{notification_type}}',
|
||||||
|
subject: '{{subject}}',
|
||||||
|
message: '{{message}}',
|
||||||
|
image: '{{image}}',
|
||||||
|
email: '{{notifyuser_email}}',
|
||||||
|
username: '{{notifyuser_username}}',
|
||||||
|
avatar: '{{notifyuser_avatar}}',
|
||||||
|
'{{media}}': {
|
||||||
|
media_type: '{{media_type}}',
|
||||||
|
tmdbId: '{{media_tmdbid}}',
|
||||||
|
imdbId: '{{media_imdbid}}',
|
||||||
|
tvdbId: '{{media_tvdbid}}',
|
||||||
|
status: '{{media_status}}',
|
||||||
|
status4k: '{{media_status4k}}',
|
||||||
|
},
|
||||||
|
'{{extra}}': [],
|
||||||
|
};
|
||||||
|
|
||||||
|
const messages = defineMessages({
|
||||||
|
save: 'Save Changes',
|
||||||
|
saving: 'Saving...',
|
||||||
|
agentenabled: 'Agent Enabled',
|
||||||
|
webhookUrl: 'Webhook URL',
|
||||||
|
authheader: 'Authorization Header',
|
||||||
|
validationWebhookUrlRequired: 'You must provide a webhook URL',
|
||||||
|
validationJsonPayloadRequired: 'You must provide a JSON Payload',
|
||||||
|
webhookUrlPlaceholder: 'Remote webhook URL',
|
||||||
|
webhooksettingssaved: 'Webhook notification settings saved!',
|
||||||
|
webhooksettingsfailed: 'Webhook notification settings failed to save.',
|
||||||
|
testsent: 'Test notification sent!',
|
||||||
|
test: 'Test',
|
||||||
|
notificationtypes: 'Notification Types',
|
||||||
|
resetPayload: 'Reset to Default JSON Payload',
|
||||||
|
resetPayloadSuccess: 'JSON reset to default payload.',
|
||||||
|
customJson: 'Custom JSON Payload',
|
||||||
|
});
|
||||||
|
|
||||||
|
const NotificationsWebhook: React.FC = () => {
|
||||||
|
const intl = useIntl();
|
||||||
|
const { addToast } = useToasts();
|
||||||
|
const { data, error, revalidate } = useSWR(
|
||||||
|
'/api/v1/settings/notifications/webhook'
|
||||||
|
);
|
||||||
|
|
||||||
|
const NotificationsWebhookSchema = Yup.object().shape({
|
||||||
|
webhookUrl: Yup.string().required(
|
||||||
|
intl.formatMessage(messages.validationWebhookUrlRequired)
|
||||||
|
),
|
||||||
|
jsonPayload: Yup.string()
|
||||||
|
.required(intl.formatMessage(messages.validationJsonPayloadRequired))
|
||||||
|
.test('validate-json', 'Invalid JSON', (value) => {
|
||||||
|
try {
|
||||||
|
JSON.parse(value ?? '');
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!data && !error) {
|
||||||
|
return <LoadingSpinner />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Formik
|
||||||
|
initialValues={{
|
||||||
|
enabled: data.enabled,
|
||||||
|
types: data.types,
|
||||||
|
webhookUrl: data.options.webhookUrl,
|
||||||
|
jsonPayload: data.options.jsonPayload,
|
||||||
|
authHeader: data.options.authHeader,
|
||||||
|
}}
|
||||||
|
validationSchema={NotificationsWebhookSchema}
|
||||||
|
onSubmit={async (values) => {
|
||||||
|
try {
|
||||||
|
await axios.post('/api/v1/settings/notifications/webhook', {
|
||||||
|
enabled: values.enabled,
|
||||||
|
types: values.types,
|
||||||
|
options: {
|
||||||
|
webhookUrl: values.webhookUrl,
|
||||||
|
jsonPayload: JSON.stringify(values.jsonPayload),
|
||||||
|
authHeader: values.authHeader,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
addToast(intl.formatMessage(messages.webhooksettingssaved), {
|
||||||
|
appearance: 'success',
|
||||||
|
autoDismiss: true,
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
addToast(intl.formatMessage(messages.webhooksettingsfailed), {
|
||||||
|
appearance: 'error',
|
||||||
|
autoDismiss: true,
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
revalidate();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{({
|
||||||
|
errors,
|
||||||
|
touched,
|
||||||
|
isSubmitting,
|
||||||
|
values,
|
||||||
|
isValid,
|
||||||
|
setFieldValue,
|
||||||
|
setFieldTouched,
|
||||||
|
}) => {
|
||||||
|
const resetPayload = () => {
|
||||||
|
setFieldValue(
|
||||||
|
'jsonPayload',
|
||||||
|
JSON.stringify(defaultPayload, undefined, ' ')
|
||||||
|
);
|
||||||
|
addToast(intl.formatMessage(messages.resetPayloadSuccess), {
|
||||||
|
appearance: 'info',
|
||||||
|
autoDismiss: true,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const testSettings = async () => {
|
||||||
|
await axios.post('/api/v1/settings/notifications/webhook/test', {
|
||||||
|
enabled: true,
|
||||||
|
types: values.types,
|
||||||
|
options: {
|
||||||
|
webhookUrl: values.webhookUrl,
|
||||||
|
jsonPayload: JSON.stringify(values.jsonPayload),
|
||||||
|
authHeader: values.authHeader,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
addToast(intl.formatMessage(messages.testsent), {
|
||||||
|
appearance: 'info',
|
||||||
|
autoDismiss: true,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Form>
|
||||||
|
<div className="sm:grid sm:grid-cols-3 sm:gap-4 sm:items-start sm:border-t sm:border-gray-200">
|
||||||
|
<label
|
||||||
|
htmlFor="enabled"
|
||||||
|
className="block text-sm font-medium leading-5 text-gray-400 sm:mt-px"
|
||||||
|
>
|
||||||
|
{intl.formatMessage(messages.agentenabled)}
|
||||||
|
</label>
|
||||||
|
<div className="mt-1 sm:mt-0 sm:col-span-2">
|
||||||
|
<Field
|
||||||
|
type="checkbox"
|
||||||
|
id="enabled"
|
||||||
|
name="enabled"
|
||||||
|
className="w-6 h-6 text-indigo-600 transition duration-150 ease-in-out rounded-md form-checkbox"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="mt-6 sm:mt-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:items-start sm:border-t sm:border-gray-800">
|
||||||
|
<label
|
||||||
|
htmlFor="name"
|
||||||
|
className="block text-sm font-medium leading-5 text-gray-400 sm:mt-px"
|
||||||
|
>
|
||||||
|
{intl.formatMessage(messages.webhookUrl)}
|
||||||
|
</label>
|
||||||
|
<div className="mt-1 sm:mt-0 sm:col-span-2">
|
||||||
|
<div className="flex max-w-lg rounded-md shadow-sm">
|
||||||
|
<Field
|
||||||
|
id="webhookUrl"
|
||||||
|
name="webhookUrl"
|
||||||
|
type="text"
|
||||||
|
placeholder={intl.formatMessage(
|
||||||
|
messages.webhookUrlPlaceholder
|
||||||
|
)}
|
||||||
|
className="flex-1 block w-full min-w-0 transition duration-150 ease-in-out bg-gray-700 border border-gray-500 rounded-md form-input sm:text-sm sm:leading-5"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{errors.webhookUrl && touched.webhookUrl && (
|
||||||
|
<div className="mt-2 text-red-500">{errors.webhookUrl}</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="mt-6 sm:mt-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:items-start sm:border-t sm:border-gray-800">
|
||||||
|
<label
|
||||||
|
htmlFor="name"
|
||||||
|
className="block text-sm font-medium leading-5 text-gray-400 sm:mt-px"
|
||||||
|
>
|
||||||
|
{intl.formatMessage(messages.authheader)}
|
||||||
|
</label>
|
||||||
|
<div className="mt-1 sm:mt-0 sm:col-span-2">
|
||||||
|
<div className="flex max-w-lg rounded-md shadow-sm">
|
||||||
|
<Field
|
||||||
|
id="authHeader"
|
||||||
|
name="authHeader"
|
||||||
|
type="text"
|
||||||
|
className="flex-1 block w-full min-w-0 transition duration-150 ease-in-out bg-gray-700 border border-gray-500 rounded-md form-input sm:text-sm sm:leading-5"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="mt-6 sm:mt-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:items-start sm:border-t sm:border-gray-800">
|
||||||
|
<label
|
||||||
|
htmlFor="name"
|
||||||
|
className="block text-sm font-medium leading-5 text-gray-400 sm:mt-px"
|
||||||
|
>
|
||||||
|
{intl.formatMessage(messages.customJson)}
|
||||||
|
</label>
|
||||||
|
<div className="mt-1 sm:mt-0 sm:col-span-2">
|
||||||
|
<div className="flex max-w-lg rounded-md shadow-sm">
|
||||||
|
<JSONEditor
|
||||||
|
name="webhook-json-payload"
|
||||||
|
onUpdate={(value) => setFieldValue('jsonPayload', value)}
|
||||||
|
value={values.jsonPayload}
|
||||||
|
onBlur={() => setFieldTouched('jsonPayload')}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{errors.jsonPayload && touched.jsonPayload && (
|
||||||
|
<div className="mt-2 text-red-500">{errors.jsonPayload}</div>
|
||||||
|
)}
|
||||||
|
<div className="mt-2">
|
||||||
|
<Button
|
||||||
|
buttonSize="sm"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
resetPayload();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
className="w-5 h-5 mr-1"
|
||||||
|
fill="currentColor"
|
||||||
|
viewBox="0 0 20 20"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
fillRule="evenodd"
|
||||||
|
d="M4 2a1 1 0 011 1v2.101a7.002 7.002 0 0111.601 2.566 1 1 0 11-1.885.666A5.002 5.002 0 005.999 7H9a1 1 0 010 2H4a1 1 0 01-1-1V3a1 1 0 011-1zm.008 9.057a1 1 0 011.276.61A5.002 5.002 0 0014.001 13H11a1 1 0 110-2h5a1 1 0 011 1v5a1 1 0 11-2 0v-2.101a7.002 7.002 0 01-11.601-2.566 1 1 0 01.61-1.276z"
|
||||||
|
clipRule="evenodd"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
{intl.formatMessage(messages.resetPayload)}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="mt-6">
|
||||||
|
<div role="group" aria-labelledby="label-permissions">
|
||||||
|
<div className="sm:grid sm:grid-cols-3 sm:gap-4 sm:items-baseline">
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
className="text-base font-medium leading-6 text-gray-400 sm:text-sm sm:leading-5"
|
||||||
|
id="label-types"
|
||||||
|
>
|
||||||
|
{intl.formatMessage(messages.notificationtypes)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="mt-4 sm:mt-0 sm:col-span-2">
|
||||||
|
<div className="max-w-lg">
|
||||||
|
<NotificationTypeSelector
|
||||||
|
currentTypes={values.types}
|
||||||
|
onUpdate={(newTypes) =>
|
||||||
|
setFieldValue('types', newTypes)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="pt-5 mt-8 border-t border-gray-700">
|
||||||
|
<div className="flex justify-end">
|
||||||
|
<span className="inline-flex ml-3 rounded-md shadow-sm">
|
||||||
|
<Button
|
||||||
|
buttonType="warning"
|
||||||
|
disabled={isSubmitting || !isValid}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
testSettings();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{intl.formatMessage(messages.test)}
|
||||||
|
</Button>
|
||||||
|
</span>
|
||||||
|
<span className="inline-flex ml-3 rounded-md shadow-sm">
|
||||||
|
<Button
|
||||||
|
buttonType="primary"
|
||||||
|
type="submit"
|
||||||
|
disabled={isSubmitting || !isValid}
|
||||||
|
>
|
||||||
|
{isSubmitting
|
||||||
|
? intl.formatMessage(messages.saving)
|
||||||
|
: intl.formatMessage(messages.save)}
|
||||||
|
</Button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Form>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
</Formik>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default NotificationsWebhook;
|
@ -0,0 +1,17 @@
|
|||||||
|
import { NextPage } from 'next';
|
||||||
|
import React from 'react';
|
||||||
|
import NotificationsWebhook from '../../../components/Settings/Notifications/NotificationsWebhook';
|
||||||
|
import SettingsLayout from '../../../components/Settings/SettingsLayout';
|
||||||
|
import SettingsNotifications from '../../../components/Settings/SettingsNotifications';
|
||||||
|
|
||||||
|
const NotificationsPage: NextPage = () => {
|
||||||
|
return (
|
||||||
|
<SettingsLayout>
|
||||||
|
<SettingsNotifications>
|
||||||
|
<NotificationsWebhook />
|
||||||
|
</SettingsNotifications>
|
||||||
|
</SettingsLayout>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default NotificationsPage;
|
Loading…
Reference in new issue