Compare commits
21 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
c762f0562f | ||
![]() |
98274c3d71 | ||
![]() |
8c936a91be | ||
![]() |
24451892ff | ||
![]() |
6bba2c82fe | ||
![]() |
3320dc6bc4 | ||
![]() |
9d42fd9293 | ||
![]() |
e6b806616f | ||
![]() |
6ec5872391 | ||
![]() |
a26cf932a1 | ||
![]() |
38a3e3ab9f | ||
![]() |
32b17c1418 | ||
![]() |
44aa6a1579 | ||
![]() |
2369aacd2a | ||
![]() |
7dafc6364b | ||
![]() |
3d25fa0aeb | ||
![]() |
0297b0f296 | ||
![]() |
4c7d09c3d8 | ||
![]() |
48a74826e8 | ||
![]() |
ef34068ac4 | ||
![]() |
3987a8db77 |
@@ -8,7 +8,7 @@
|
|||||||
"version": "latest"
|
"version": "latest"
|
||||||
},
|
},
|
||||||
"ghcr.io/devcontainers/features/node:1": {
|
"ghcr.io/devcontainers/features/node:1": {
|
||||||
"version": 16
|
"version": 20
|
||||||
},
|
},
|
||||||
"ghcr.io/devcontainers/features/common-utils:1": {
|
"ghcr.io/devcontainers/features/common-utils:1": {
|
||||||
"username": "vscode",
|
"username": "vscode",
|
||||||
|
1
packages/backend/src/apps/airbrake/assets/favicon.svg
Normal file
1
packages/backend/src/apps/airbrake/assets/favicon.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg height="255" preserveAspectRatio="xMidYMid" viewBox="0 0 256 255" width="256" xmlns="http://www.w3.org/2000/svg"><path d="m128.636514 155.746615v-155.23361889h-3.522242v.06873152l-124.60824865 64.03287157v60.8642488h.00597665v3.234366h-.00597665v60.868233l124.60824865 64.747082h3.842989v-98.581914z" fill="#ff8e4a"/><path d="m129.941416 254.328529 125.568498-64.747082v-124.9668478l-125.887253-64.10160309h-2.243237v253.81055289h2.243237" fill="#f48746"/><path d="m109.097837 87.2551595h36.19561v59.2077195h-36.19561z" fill="#ff8e4a"/><path d="m66.1735097 188.397074h14.8639378c9.4102412 0 12.6087471-2.238257 15.6189883-9.988981l8.2796572-21.353587h45.159596l8.280653 21.353587c3.011238 7.750724 6.396016 9.988981 15.805261 9.988981h14.677665v-19.114335h-3.011237c-3.19751 0-4.704622-.689307-5.831222-3.790194l-39.516638-99.3658524h-25.779299l-39.703907 99.3658524c-1.1285915 3.100887-2.632716 3.790194-5.833214 3.790194h-3.0102413zm44.4075333-49.939922 11.478163-30.655253c2.445448-6.714771 5.269417-18.2556889 5.269417-18.2556889h.375533s2.822972 11.5409179 5.269416 18.2556889l11.478163 30.655253z" fill="#fff"/><path d="m231.204856 150.082739v-51.8086223c.235082 4.5233303 2.970397 16.8432063 24.305058 27.8512063v11.653479zm0-53.1623343v1.353712c-.029883-.5926848-.01793-1.0479066 0-1.353712zm.041837-.4392841s-.022911.1534008-.041837.4392841v-.4392841z" fill="#d4763c"/><path d="m231.155051 94.3016342c-.013946.9931207.05877 1.8945993.049805 2.0460078-.01793.2480312-2.220327 16.094132 24.305058 29.777681v-60.863253c-23.325883 12.0349884-24.449494 25.7414475-24.354863 29.0395642" fill="#ff8e4a"/></svg>
|
After Width: | Height: | Size: 1.6 KiB |
@@ -16,14 +16,25 @@ export default {
|
|||||||
clickToCopy: false,
|
clickToCopy: false,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
key: 'apiKey',
|
key: 'instanceUrl',
|
||||||
label: 'API Key',
|
label: 'Instance URL',
|
||||||
type: 'string',
|
type: 'string',
|
||||||
required: true,
|
required: true,
|
||||||
readOnly: false,
|
readOnly: false,
|
||||||
value: null,
|
value: null,
|
||||||
placeholder: null,
|
placeholder: null,
|
||||||
description: 'Better Stack API key of your account.',
|
description: 'Your subdomain as https://{yoursubdomain}.airbrake.io',
|
||||||
|
clickToCopy: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'authToken',
|
||||||
|
label: 'Auth Token',
|
||||||
|
type: 'string',
|
||||||
|
required: true,
|
||||||
|
readOnly: false,
|
||||||
|
value: null,
|
||||||
|
placeholder: null,
|
||||||
|
description: 'Airbrake Auth Token of your account.',
|
||||||
clickToCopy: false,
|
clickToCopy: false,
|
||||||
},
|
},
|
||||||
],
|
],
|
@@ -0,0 +1,14 @@
|
|||||||
|
const verifyCredentials = async ($) => {
|
||||||
|
await $.http.get(`/api/v4/projects?key=${$.auth.data.authToken}`, {
|
||||||
|
additionalProperties: {
|
||||||
|
skipAddingAuthToken: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await $.auth.set({
|
||||||
|
screenName: $.auth.data.screenName,
|
||||||
|
authToken: $.auth.data.authToken,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export default verifyCredentials;
|
10
packages/backend/src/apps/airbrake/common/add-auth-token.js
Normal file
10
packages/backend/src/apps/airbrake/common/add-auth-token.js
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
const addAuthToken = ($, requestConfig) => {
|
||||||
|
if (requestConfig.additionalProperties?.skipAddingAuthToken)
|
||||||
|
return requestConfig;
|
||||||
|
|
||||||
|
requestConfig.url = requestConfig.url + `?key=${$.auth.data.authToken}`;
|
||||||
|
|
||||||
|
return requestConfig;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default addAuthToken;
|
11
packages/backend/src/apps/airbrake/common/set-base-url.js
Normal file
11
packages/backend/src/apps/airbrake/common/set-base-url.js
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
const setBaseUrl = ($, requestConfig) => {
|
||||||
|
const subdomain = $.auth.data.instanceUrl;
|
||||||
|
|
||||||
|
if (subdomain) {
|
||||||
|
requestConfig.baseURL = `https://${subdomain}.airbrake.io`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return requestConfig;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default setBaseUrl;
|
3
packages/backend/src/apps/airbrake/dynamic-data/index.js
Normal file
3
packages/backend/src/apps/airbrake/dynamic-data/index.js
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
import listProjects from './list-projects/index.js';
|
||||||
|
|
||||||
|
export default [listProjects];
|
@@ -0,0 +1,23 @@
|
|||||||
|
export default {
|
||||||
|
name: 'List projects',
|
||||||
|
key: 'listProjects',
|
||||||
|
|
||||||
|
async run($) {
|
||||||
|
const projects = {
|
||||||
|
data: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
const { data } = await $.http.get('/api/v4/projects');
|
||||||
|
|
||||||
|
if (data.projects.length) {
|
||||||
|
for (const project of data.projects) {
|
||||||
|
projects.data.push({
|
||||||
|
value: project.id,
|
||||||
|
name: project.name,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return projects;
|
||||||
|
},
|
||||||
|
};
|
21
packages/backend/src/apps/airbrake/index.js
Normal file
21
packages/backend/src/apps/airbrake/index.js
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
import defineApp from '../../helpers/define-app.js';
|
||||||
|
import setBaseUrl from './common/set-base-url.js';
|
||||||
|
import auth from './auth/index.js';
|
||||||
|
import addAuthToken from './common/add-auth-token.js';
|
||||||
|
import triggers from './triggers/index.js';
|
||||||
|
import dynamicData from './dynamic-data/index.js';
|
||||||
|
|
||||||
|
export default defineApp({
|
||||||
|
name: 'Airbrake',
|
||||||
|
key: 'airbrake',
|
||||||
|
iconUrl: '{BASE_URL}/apps/airbrake/assets/favicon.svg',
|
||||||
|
authDocUrl: 'https://automatisch.io/docs/apps/airbrake/connection',
|
||||||
|
supportsConnections: true,
|
||||||
|
baseUrl: 'https://www.airbrake.io',
|
||||||
|
apiBaseUrl: '',
|
||||||
|
primaryColor: 'f58c54',
|
||||||
|
beforeRequest: [setBaseUrl, addAuthToken],
|
||||||
|
auth,
|
||||||
|
triggers,
|
||||||
|
dynamicData,
|
||||||
|
});
|
3
packages/backend/src/apps/airbrake/triggers/index.js
Normal file
3
packages/backend/src/apps/airbrake/triggers/index.js
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
import newErrors from './new-errors/index.js';
|
||||||
|
|
||||||
|
export default [newErrors];
|
@@ -0,0 +1,66 @@
|
|||||||
|
//import { URLSearchParams } from 'node:url';
|
||||||
|
import defineTrigger from '../../../../helpers/define-trigger.js';
|
||||||
|
|
||||||
|
export default defineTrigger({
|
||||||
|
name: 'New errors',
|
||||||
|
key: 'newErrors',
|
||||||
|
pollInterval: 15,
|
||||||
|
description: 'Triggers when a new error occurs.',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
label: 'Project',
|
||||||
|
key: 'projectId',
|
||||||
|
type: 'dropdown',
|
||||||
|
required: true,
|
||||||
|
description: '',
|
||||||
|
variables: true,
|
||||||
|
source: {
|
||||||
|
type: 'query',
|
||||||
|
name: 'getDynamicData',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'key',
|
||||||
|
value: 'listProjects',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
|
||||||
|
async run($) {
|
||||||
|
const projectId = $.step.parameters.projectId;
|
||||||
|
|
||||||
|
const params = {
|
||||||
|
limit: 100,
|
||||||
|
page: 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
let next = false;
|
||||||
|
do {
|
||||||
|
const { data } = await $.http.get(
|
||||||
|
`/api/v4/projects/${projectId}/groups`,
|
||||||
|
{ params }
|
||||||
|
);
|
||||||
|
|
||||||
|
if (data.count > params.limit) {
|
||||||
|
params.page = params.page + 1;
|
||||||
|
next = true;
|
||||||
|
} else {
|
||||||
|
next = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data?.groups?.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const group of data.groups) {
|
||||||
|
$.pushTriggerItem({
|
||||||
|
raw: group,
|
||||||
|
meta: {
|
||||||
|
internalId: group.id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} while (next);
|
||||||
|
},
|
||||||
|
});
|
@@ -1,43 +0,0 @@
|
|||||||
import defineAction from '../../../../helpers/define-action.js';
|
|
||||||
|
|
||||||
export default defineAction({
|
|
||||||
name: 'Acknowledge incident',
|
|
||||||
key: 'acknowledgeIncident',
|
|
||||||
description: 'Acknowledges an incident.',
|
|
||||||
arguments: [
|
|
||||||
{
|
|
||||||
label: 'Incident ID',
|
|
||||||
key: 'incidentId',
|
|
||||||
type: 'string',
|
|
||||||
required: true,
|
|
||||||
variables: true,
|
|
||||||
description:
|
|
||||||
'This serves as the incident ID that requires your acknowledgment.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Acknowledged by',
|
|
||||||
key: 'acknowledgedBy',
|
|
||||||
type: 'string',
|
|
||||||
required: false,
|
|
||||||
variables: true,
|
|
||||||
description:
|
|
||||||
"This refers to the individual's name, email, or another form of identification that the person who acknowledged the incident has provided.",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
async run($) {
|
|
||||||
const acknowledgedBy = $.step.parameters.acknowledgedBy;
|
|
||||||
const incidentId = $.step.parameters.incidentId;
|
|
||||||
|
|
||||||
const body = {
|
|
||||||
acknowledged_by: acknowledgedBy,
|
|
||||||
};
|
|
||||||
|
|
||||||
const response = await $.http.post(
|
|
||||||
`/v2/incidents/${incidentId}/acknowledge`,
|
|
||||||
body
|
|
||||||
);
|
|
||||||
|
|
||||||
$.setActionItem({ raw: response.data.data });
|
|
||||||
},
|
|
||||||
});
|
|
@@ -1,120 +0,0 @@
|
|||||||
import defineAction from '../../../../helpers/define-action.js';
|
|
||||||
|
|
||||||
export default defineAction({
|
|
||||||
name: 'Create incident',
|
|
||||||
key: 'createIncident',
|
|
||||||
description: 'Creates an incident that informs the team.',
|
|
||||||
arguments: [
|
|
||||||
{
|
|
||||||
label: 'Brief Summary',
|
|
||||||
key: 'briefSummary',
|
|
||||||
type: 'string',
|
|
||||||
required: true,
|
|
||||||
variables: true,
|
|
||||||
description: 'A short description outlining the issue.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Description',
|
|
||||||
key: 'description',
|
|
||||||
type: 'string',
|
|
||||||
required: false,
|
|
||||||
variables: true,
|
|
||||||
description:
|
|
||||||
'An elaborate description of the situation, offering insights into what is occurring, along with instructions to reproduce the problem.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Requester Email',
|
|
||||||
key: 'requesterEmail',
|
|
||||||
type: 'string',
|
|
||||||
required: true,
|
|
||||||
variables: true,
|
|
||||||
description:
|
|
||||||
'This represents the email address of the individual who initiated the incident request.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Alert Settings - Call',
|
|
||||||
key: 'alertSettingsCall',
|
|
||||||
type: 'dropdown',
|
|
||||||
required: true,
|
|
||||||
description: 'Should we call the on-call person?',
|
|
||||||
variables: true,
|
|
||||||
options: [
|
|
||||||
{ label: 'Yes', value: 'true' },
|
|
||||||
{ label: 'No', value: 'false' },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Alert Settings - Text',
|
|
||||||
key: 'alertSettingsText',
|
|
||||||
type: 'dropdown',
|
|
||||||
required: true,
|
|
||||||
description: 'Should we text the on-call person?',
|
|
||||||
variables: true,
|
|
||||||
options: [
|
|
||||||
{ label: 'Yes', value: 'true' },
|
|
||||||
{ label: 'No', value: 'false' },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Alert Settings - Email',
|
|
||||||
key: 'alertSettingsEmail',
|
|
||||||
type: 'dropdown',
|
|
||||||
required: true,
|
|
||||||
description: 'Should we email the on-call person?',
|
|
||||||
variables: true,
|
|
||||||
options: [
|
|
||||||
{ label: 'Yes', value: 'true' },
|
|
||||||
{ label: 'No', value: 'false' },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Alert Settings - Push Notification',
|
|
||||||
key: 'alertSettingsPushNotification',
|
|
||||||
type: 'dropdown',
|
|
||||||
required: true,
|
|
||||||
description: 'Should we send a push notification to the on-call person?',
|
|
||||||
variables: true,
|
|
||||||
options: [
|
|
||||||
{ label: 'Yes', value: 'true' },
|
|
||||||
{ label: 'No', value: 'false' },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Team Alert Wait Time',
|
|
||||||
key: 'teamAlertWaitTime',
|
|
||||||
type: 'string',
|
|
||||||
required: true,
|
|
||||||
variables: true,
|
|
||||||
description:
|
|
||||||
"What is the time threshold for acknowledgment before escalating to the entire team? (Specify in seconds) - Use a negative value to indicate no team alert if the on-call person doesn't respond, and use 0 for an immediate alert to the entire team.",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
async run($) {
|
|
||||||
const {
|
|
||||||
briefSummary,
|
|
||||||
description,
|
|
||||||
requesterEmail,
|
|
||||||
alertSettingsCall,
|
|
||||||
alertSettingsText,
|
|
||||||
alertSettingsEmail,
|
|
||||||
alertSettingsPushNotification,
|
|
||||||
teamAlertWaitTime,
|
|
||||||
} = $.step.parameters;
|
|
||||||
|
|
||||||
const body = {
|
|
||||||
summary: briefSummary,
|
|
||||||
description,
|
|
||||||
requester_email: requesterEmail,
|
|
||||||
call: alertSettingsCall,
|
|
||||||
sms: alertSettingsText,
|
|
||||||
email: alertSettingsEmail,
|
|
||||||
push: alertSettingsPushNotification,
|
|
||||||
team_wait: teamAlertWaitTime,
|
|
||||||
};
|
|
||||||
|
|
||||||
const response = await $.http.post('/v2/incidents', body);
|
|
||||||
|
|
||||||
$.setActionItem({ raw: response.data.data });
|
|
||||||
},
|
|
||||||
});
|
|
@@ -1,25 +0,0 @@
|
|||||||
import defineAction from '../../../../helpers/define-action.js';
|
|
||||||
|
|
||||||
export default defineAction({
|
|
||||||
name: 'Find incident',
|
|
||||||
key: 'findIncident',
|
|
||||||
description: 'finds an incident.',
|
|
||||||
arguments: [
|
|
||||||
{
|
|
||||||
label: 'Incident ID',
|
|
||||||
key: 'incidentId',
|
|
||||||
type: 'string',
|
|
||||||
required: true,
|
|
||||||
variables: true,
|
|
||||||
description: 'ID for querying incidents.',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
async run($) {
|
|
||||||
const incidentId = $.step.parameters.incidentId;
|
|
||||||
|
|
||||||
const response = await $.http.get(`/v2/incidents/${incidentId}`);
|
|
||||||
|
|
||||||
$.setActionItem({ raw: response.data.data });
|
|
||||||
},
|
|
||||||
});
|
|
@@ -1,11 +0,0 @@
|
|||||||
import acknowledgeIncident from './acknowledge-incident/index.js';
|
|
||||||
import createIncident from './create-incident/index.js';
|
|
||||||
import findIncident from './find-incident/index.js';
|
|
||||||
import resolveIncident from './resolve-incident/index.js';
|
|
||||||
|
|
||||||
export default [
|
|
||||||
acknowledgeIncident,
|
|
||||||
createIncident,
|
|
||||||
findIncident,
|
|
||||||
resolveIncident,
|
|
||||||
];
|
|
@@ -1,43 +0,0 @@
|
|||||||
import defineAction from '../../../../helpers/define-action.js';
|
|
||||||
|
|
||||||
export default defineAction({
|
|
||||||
name: 'Resolve incident',
|
|
||||||
key: 'resolveIncident',
|
|
||||||
description: 'Resolves an incident.',
|
|
||||||
arguments: [
|
|
||||||
{
|
|
||||||
label: 'Incident ID',
|
|
||||||
key: 'incidentId',
|
|
||||||
type: 'string',
|
|
||||||
required: true,
|
|
||||||
variables: true,
|
|
||||||
description:
|
|
||||||
'This represents the identification for an incident that requires resolution.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Resolved by',
|
|
||||||
key: 'resolvedBy',
|
|
||||||
type: 'string',
|
|
||||||
required: false,
|
|
||||||
variables: true,
|
|
||||||
description:
|
|
||||||
"This refers to the individual's name, email, or another form of identification that the person who resolved the incident has provided.",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
async run($) {
|
|
||||||
const resolvedBy = $.step.parameters.resolvedBy;
|
|
||||||
const incidentId = $.step.parameters.incidentId;
|
|
||||||
|
|
||||||
const body = {
|
|
||||||
resolved_by: resolvedBy,
|
|
||||||
};
|
|
||||||
|
|
||||||
const response = await $.http.post(
|
|
||||||
`/v2/incidents/${incidentId}/resolve`,
|
|
||||||
body
|
|
||||||
);
|
|
||||||
|
|
||||||
$.setActionItem({ raw: response.data.data });
|
|
||||||
},
|
|
||||||
});
|
|
@@ -1,21 +0,0 @@
|
|||||||
<?xml version="1.0" standalone="no"?>
|
|
||||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN"
|
|
||||||
"http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd">
|
|
||||||
<svg version="1.0" xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="200.000000pt" height="200.000000pt" viewBox="0 0 200.000000 200.000000"
|
|
||||||
preserveAspectRatio="xMidYMid meet">
|
|
||||||
|
|
||||||
<g transform="translate(0.000000,200.000000) scale(0.100000,-0.100000)"
|
|
||||||
fill="#000" stroke="none">
|
|
||||||
<path d="M0 1000 l0 -1000 1000 0 1000 0 0 1000 0 1000 -1000 0 -1000 0 0
|
|
||||||
-1000z m1162 460 c14 -11 113 -184 232 -408 228 -429 231 -439 175 -486 -35
|
|
||||||
-30 -30 -29 -140 -15 -89 12 -123 25 -152 56 -9 11 -72 147 -140 304 -113 263
|
|
||||||
-124 284 -149 287 -14 2 -29 10 -32 17 -8 21 67 214 94 242 28 29 78 30 112 3z
|
|
||||||
m-340 -148 c10 -10 72 -175 139 -367 114 -325 121 -351 108 -374 -8 -14 -27
|
|
||||||
-32 -41 -41 -25 -13 -34 -12 -126 18 -55 18 -111 43 -125 56 -19 17 -40 67
|
|
||||||
-76 182 -36 112 -58 164 -73 176 l-22 16 27 99 c63 224 66 232 95 248 31 17
|
|
||||||
69 12 94 -13z m-314 -219 c16 -15 26 -59 56 -243 42 -262 43 -285 17 -300 -11
|
|
||||||
-5 -24 -10 -30 -10 -19 0 -140 114 -150 141 -7 20 -4 76 10 191 10 90 19 171
|
|
||||||
19 181 0 18 33 57 49 57 5 0 18 -8 29 -17z"/>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 1.1 KiB |
@@ -1,10 +0,0 @@
|
|||||||
const verifyCredentials = async ($) => {
|
|
||||||
await $.http.get('/v2/metadata');
|
|
||||||
|
|
||||||
await $.auth.set({
|
|
||||||
screenName: $.auth.data.screenName,
|
|
||||||
apiKey: $.auth.data.apiKey,
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
export default verifyCredentials;
|
|
@@ -1,9 +0,0 @@
|
|||||||
const addAuthHeader = ($, requestConfig) => {
|
|
||||||
if ($.auth.data?.apiKey) {
|
|
||||||
requestConfig.headers.Authorization = `Bearer ${$.auth.data.apiKey}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
return requestConfig;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default addAuthHeader;
|
|
@@ -1,18 +0,0 @@
|
|||||||
import defineApp from '../../helpers/define-app.js';
|
|
||||||
import addAuthHeader from './common/add-auth-header.js';
|
|
||||||
import auth from './auth/index.js';
|
|
||||||
import actions from './actions/index.js';
|
|
||||||
|
|
||||||
export default defineApp({
|
|
||||||
name: 'Better Stack',
|
|
||||||
key: 'better-stack',
|
|
||||||
iconUrl: '{BASE_URL}/apps/better-stack/assets/favicon.svg',
|
|
||||||
authDocUrl: 'https://automatisch.io/docs/apps/better-stack/connection',
|
|
||||||
supportsConnections: true,
|
|
||||||
baseUrl: 'https://betterstack.com',
|
|
||||||
apiBaseUrl: 'https://uptime.betterstack.com/api',
|
|
||||||
primaryColor: '000000',
|
|
||||||
beforeRequest: [addAuthHeader],
|
|
||||||
auth,
|
|
||||||
actions,
|
|
||||||
});
|
|
@@ -1,5 +1,6 @@
|
|||||||
import defineAction from '../../../../helpers/define-action.js';
|
import defineAction from '../../../../helpers/define-action.js';
|
||||||
|
|
||||||
|
import base64ToString from './transformers/base64-to-string.js';
|
||||||
import capitalize from './transformers/capitalize.js';
|
import capitalize from './transformers/capitalize.js';
|
||||||
import extractEmailAddress from './transformers/extract-email-address.js';
|
import extractEmailAddress from './transformers/extract-email-address.js';
|
||||||
import extractNumber from './transformers/extract-number.js';
|
import extractNumber from './transformers/extract-number.js';
|
||||||
@@ -8,10 +9,12 @@ import lowercase from './transformers/lowercase.js';
|
|||||||
import markdownToHtml from './transformers/markdown-to-html.js';
|
import markdownToHtml from './transformers/markdown-to-html.js';
|
||||||
import pluralize from './transformers/pluralize.js';
|
import pluralize from './transformers/pluralize.js';
|
||||||
import replace from './transformers/replace.js';
|
import replace from './transformers/replace.js';
|
||||||
|
import stringToBase64 from './transformers/string-to-base64.js';
|
||||||
import trimWhitespace from './transformers/trim-whitespace.js';
|
import trimWhitespace from './transformers/trim-whitespace.js';
|
||||||
import useDefaultValue from './transformers/use-default-value.js';
|
import useDefaultValue from './transformers/use-default-value.js';
|
||||||
|
|
||||||
const transformers = {
|
const transformers = {
|
||||||
|
base64ToString,
|
||||||
capitalize,
|
capitalize,
|
||||||
extractEmailAddress,
|
extractEmailAddress,
|
||||||
extractNumber,
|
extractNumber,
|
||||||
@@ -20,6 +23,7 @@ const transformers = {
|
|||||||
markdownToHtml,
|
markdownToHtml,
|
||||||
pluralize,
|
pluralize,
|
||||||
replace,
|
replace,
|
||||||
|
stringToBase64,
|
||||||
trimWhitespace,
|
trimWhitespace,
|
||||||
useDefaultValue,
|
useDefaultValue,
|
||||||
};
|
};
|
||||||
@@ -37,6 +41,7 @@ export default defineAction({
|
|||||||
required: true,
|
required: true,
|
||||||
variables: true,
|
variables: true,
|
||||||
options: [
|
options: [
|
||||||
|
{ label: 'Base64 to String', value: 'base64ToString' },
|
||||||
{ label: 'Capitalize', value: 'capitalize' },
|
{ label: 'Capitalize', value: 'capitalize' },
|
||||||
{ label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' },
|
{ label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' },
|
||||||
{ label: 'Convert Markdown to HTML', value: 'markdownToHtml' },
|
{ label: 'Convert Markdown to HTML', value: 'markdownToHtml' },
|
||||||
@@ -45,6 +50,7 @@ export default defineAction({
|
|||||||
{ label: 'Lowercase', value: 'lowercase' },
|
{ label: 'Lowercase', value: 'lowercase' },
|
||||||
{ label: 'Pluralize', value: 'pluralize' },
|
{ label: 'Pluralize', value: 'pluralize' },
|
||||||
{ label: 'Replace', value: 'replace' },
|
{ label: 'Replace', value: 'replace' },
|
||||||
|
{ label: 'String to Base64', value: 'stringToBase64' },
|
||||||
{ label: 'Trim Whitespace', value: 'trimWhitespace' },
|
{ label: 'Trim Whitespace', value: 'trimWhitespace' },
|
||||||
{ label: 'Use Default Value', value: 'useDefaultValue' },
|
{ label: 'Use Default Value', value: 'useDefaultValue' },
|
||||||
],
|
],
|
||||||
|
@@ -0,0 +1,8 @@
|
|||||||
|
const base64ToString = ($) => {
|
||||||
|
const input = $.step.parameters.input;
|
||||||
|
const decodedString = Buffer.from(input, 'base64').toString('utf8');
|
||||||
|
|
||||||
|
return decodedString;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default base64ToString;
|
@@ -0,0 +1,8 @@
|
|||||||
|
const stringtoBase64 = ($) => {
|
||||||
|
const input = $.step.parameters.input;
|
||||||
|
const base64String = Buffer.from(input).toString('base64');
|
||||||
|
|
||||||
|
return base64String;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default stringtoBase64;
|
@@ -1,3 +1,4 @@
|
|||||||
|
import base64ToString from './text/base64-to-string.js';
|
||||||
import capitalize from './text/capitalize.js';
|
import capitalize from './text/capitalize.js';
|
||||||
import extractEmailAddress from './text/extract-email-address.js';
|
import extractEmailAddress from './text/extract-email-address.js';
|
||||||
import extractNumber from './text/extract-number.js';
|
import extractNumber from './text/extract-number.js';
|
||||||
@@ -6,6 +7,7 @@ import lowercase from './text/lowercase.js';
|
|||||||
import markdownToHtml from './text/markdown-to-html.js';
|
import markdownToHtml from './text/markdown-to-html.js';
|
||||||
import pluralize from './text/pluralize.js';
|
import pluralize from './text/pluralize.js';
|
||||||
import replace from './text/replace.js';
|
import replace from './text/replace.js';
|
||||||
|
import stringToBase64 from './text/string-to-base64.js';
|
||||||
import trimWhitespace from './text/trim-whitespace.js';
|
import trimWhitespace from './text/trim-whitespace.js';
|
||||||
import useDefaultValue from './text/use-default-value.js';
|
import useDefaultValue from './text/use-default-value.js';
|
||||||
import performMathOperation from './numbers/perform-math-operation.js';
|
import performMathOperation from './numbers/perform-math-operation.js';
|
||||||
@@ -15,6 +17,7 @@ import formatPhoneNumber from './numbers/format-phone-number.js';
|
|||||||
import formatDateTime from './date-time/format-date-time.js';
|
import formatDateTime from './date-time/format-date-time.js';
|
||||||
|
|
||||||
const options = {
|
const options = {
|
||||||
|
base64ToString,
|
||||||
capitalize,
|
capitalize,
|
||||||
extractEmailAddress,
|
extractEmailAddress,
|
||||||
extractNumber,
|
extractNumber,
|
||||||
@@ -23,6 +26,7 @@ const options = {
|
|||||||
markdownToHtml,
|
markdownToHtml,
|
||||||
pluralize,
|
pluralize,
|
||||||
replace,
|
replace,
|
||||||
|
stringToBase64,
|
||||||
trimWhitespace,
|
trimWhitespace,
|
||||||
useDefaultValue,
|
useDefaultValue,
|
||||||
performMathOperation,
|
performMathOperation,
|
||||||
|
@@ -0,0 +1,12 @@
|
|||||||
|
const base64ToString = [
|
||||||
|
{
|
||||||
|
label: 'Input',
|
||||||
|
key: 'input',
|
||||||
|
type: 'string',
|
||||||
|
required: true,
|
||||||
|
description: 'Text that will be converted from Base64 to string.',
|
||||||
|
variables: true,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export default base64ToString;
|
@@ -0,0 +1,12 @@
|
|||||||
|
const stringToBase64 = [
|
||||||
|
{
|
||||||
|
label: 'Input',
|
||||||
|
key: 'input',
|
||||||
|
type: 'string',
|
||||||
|
required: true,
|
||||||
|
description: 'Text that will be converted to Base64.',
|
||||||
|
variables: true,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export default stringToBase64;
|
@@ -18,7 +18,9 @@ const port = process.env.PORT || '3000';
|
|||||||
const serveWebAppSeparately =
|
const serveWebAppSeparately =
|
||||||
process.env.SERVE_WEB_APP_SEPARATELY === 'true' ? true : false;
|
process.env.SERVE_WEB_APP_SEPARATELY === 'true' ? true : false;
|
||||||
|
|
||||||
let apiUrl = new URL(`${protocol}://${host}:${port}`).toString();
|
let apiUrl = new URL(
|
||||||
|
process.env.API_URL || `${protocol}://${host}:${port}`
|
||||||
|
).toString();
|
||||||
apiUrl = apiUrl.substring(0, apiUrl.length - 1);
|
apiUrl = apiUrl.substring(0, apiUrl.length - 1);
|
||||||
|
|
||||||
// use apiUrl by default, which has less priority over the following cases
|
// use apiUrl by default, which has less priority over the following cases
|
||||||
@@ -88,6 +90,10 @@ const appConfig = {
|
|||||||
licenseKey: process.env.LICENSE_KEY,
|
licenseKey: process.env.LICENSE_KEY,
|
||||||
sentryDsn: process.env.SENTRY_DSN,
|
sentryDsn: process.env.SENTRY_DSN,
|
||||||
CI: process.env.CI === 'true',
|
CI: process.env.CI === 'true',
|
||||||
|
disableNotificationsPage: process.env.DISABLE_NOTIFICATIONS_PAGE === 'true',
|
||||||
|
disableFavicon: process.env.DISABLE_FAVICON === 'true',
|
||||||
|
additionalDrawerLink: process.env.ADDITIONAL_DRAWER_LINK,
|
||||||
|
additionalDrawerLinkText: process.env.ADDITIONAL_DRAWER_LINK_TEXT,
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!appConfig.encryptionKey) {
|
if (!appConfig.encryptionKey) {
|
||||||
|
@@ -1,7 +1,10 @@
|
|||||||
|
import appConfig from '../../config/app.js';
|
||||||
import User from '../../models/user.js';
|
import User from '../../models/user.js';
|
||||||
import Role from '../../models/role.js';
|
import Role from '../../models/role.js';
|
||||||
|
|
||||||
const registerUser = async (_parent, params) => {
|
const registerUser = async (_parent, params) => {
|
||||||
|
if (!appConfig.isCloud) return;
|
||||||
|
|
||||||
const { fullName, email, password } = params.input;
|
const { fullName, email, password } = params.input;
|
||||||
|
|
||||||
const existingUser = await User.query().findOne({
|
const existingUser = await User.query().findOne({
|
||||||
|
@@ -1,9 +1,17 @@
|
|||||||
|
import appConfig from '../../config/app.js';
|
||||||
import { hasValidLicense } from '../../helpers/license.ee.js';
|
import { hasValidLicense } from '../../helpers/license.ee.js';
|
||||||
import Config from '../../models/config.js';
|
import Config from '../../models/config.js';
|
||||||
|
|
||||||
const getConfig = async (_parent, params) => {
|
const getConfig = async (_parent, params) => {
|
||||||
if (!(await hasValidLicense())) return {};
|
if (!(await hasValidLicense())) return {};
|
||||||
|
|
||||||
|
const defaultConfig = {
|
||||||
|
disableNotificationsPage: appConfig.disableNotificationsPage,
|
||||||
|
disableFavicon: appConfig.disableFavicon,
|
||||||
|
additionalDrawerLink: appConfig.additionalDrawerLink,
|
||||||
|
additionalDrawerLinkText: appConfig.additionalDrawerLinkText,
|
||||||
|
};
|
||||||
|
|
||||||
const configQuery = Config.query();
|
const configQuery = Config.query();
|
||||||
|
|
||||||
if (Array.isArray(params.keys)) {
|
if (Array.isArray(params.keys)) {
|
||||||
@@ -18,7 +26,7 @@ const getConfig = async (_parent, params) => {
|
|||||||
computedConfig[key] = value?.data;
|
computedConfig[key] = value?.data;
|
||||||
|
|
||||||
return computedConfig;
|
return computedConfig;
|
||||||
}, {});
|
}, defaultConfig);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default getConfig;
|
export default getConfig;
|
||||||
|
@@ -2,6 +2,7 @@ import { vi, describe, it, expect, beforeEach } from 'vitest';
|
|||||||
import request from 'supertest';
|
import request from 'supertest';
|
||||||
import app from '../../app';
|
import app from '../../app';
|
||||||
import { createConfig } from '../../../test/factories/config';
|
import { createConfig } from '../../../test/factories/config';
|
||||||
|
import appConfig from '../../config/app';
|
||||||
import * as license from '../../helpers/license.ee';
|
import * as license from '../../helpers/license.ee';
|
||||||
|
|
||||||
describe('graphQL getConfig query', () => {
|
describe('graphQL getConfig query', () => {
|
||||||
@@ -56,6 +57,10 @@ describe('graphQL getConfig query', () => {
|
|||||||
[configOne.key]: configOne.value.data,
|
[configOne.key]: configOne.value.data,
|
||||||
[configTwo.key]: configTwo.value.data,
|
[configTwo.key]: configTwo.value.data,
|
||||||
[configThree.key]: configThree.value.data,
|
[configThree.key]: configThree.value.data,
|
||||||
|
disableNotificationsPage: false,
|
||||||
|
disableFavicon: false,
|
||||||
|
additionalDrawerLink: undefined,
|
||||||
|
additionalDrawerLinkText: undefined,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@@ -82,6 +87,48 @@ describe('graphQL getConfig query', () => {
|
|||||||
getConfig: {
|
getConfig: {
|
||||||
[configOne.key]: configOne.value.data,
|
[configOne.key]: configOne.value.data,
|
||||||
[configTwo.key]: configTwo.value.data,
|
[configTwo.key]: configTwo.value.data,
|
||||||
|
disableNotificationsPage: false,
|
||||||
|
disableFavicon: false,
|
||||||
|
additionalDrawerLink: undefined,
|
||||||
|
additionalDrawerLinkText: undefined,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
expect(response.body).toEqual(expectedResponsePayload);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('and with different defaults', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
vi.spyOn(appConfig, 'disableNotificationsPage', 'get').mockReturnValue(
|
||||||
|
true
|
||||||
|
);
|
||||||
|
vi.spyOn(appConfig, 'disableFavicon', 'get').mockReturnValue(true);
|
||||||
|
vi.spyOn(appConfig, 'additionalDrawerLink', 'get').mockReturnValue(
|
||||||
|
'https://automatisch.io'
|
||||||
|
);
|
||||||
|
vi.spyOn(appConfig, 'additionalDrawerLinkText', 'get').mockReturnValue(
|
||||||
|
'Automatisch'
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return custom config', async () => {
|
||||||
|
const response = await request(app)
|
||||||
|
.post('/graphql')
|
||||||
|
.send({ query })
|
||||||
|
.expect(200);
|
||||||
|
|
||||||
|
const expectedResponsePayload = {
|
||||||
|
data: {
|
||||||
|
getConfig: {
|
||||||
|
[configOne.key]: configOne.value.data,
|
||||||
|
[configTwo.key]: configTwo.value.data,
|
||||||
|
[configThree.key]: configThree.value.data,
|
||||||
|
disableNotificationsPage: true,
|
||||||
|
disableFavicon: true,
|
||||||
|
additionalDrawerLink: 'https://automatisch.io',
|
||||||
|
additionalDrawerLinkText: 'Automatisch',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@@ -1,6 +1,9 @@
|
|||||||
import * as path from 'path';
|
import path from 'path';
|
||||||
import * as fs from 'fs';
|
import fs from 'fs';
|
||||||
import * as handlebars from 'handlebars';
|
import handlebars from 'handlebars';
|
||||||
|
import { fileURLToPath } from 'url';
|
||||||
|
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
|
||||||
const compileEmail = (emailPath, replacements = {}) => {
|
const compileEmail = (emailPath, replacements = {}) => {
|
||||||
const filePath = path.join(__dirname, `../views/emails/${emailPath}.ee.hbs`);
|
const filePath = path.join(__dirname, `../views/emails/${emailPath}.ee.hbs`);
|
||||||
|
@@ -15,7 +15,7 @@ const webUIHandler = async (app) => {
|
|||||||
app.use(express.static(webBuildPath));
|
app.use(express.static(webBuildPath));
|
||||||
|
|
||||||
app.get('*', (_req, res) => {
|
app.get('*', (_req, res) => {
|
||||||
res.set('Content-Security-Policy', 'frame-ancestors: none;');
|
res.set('Content-Security-Policy', 'frame-ancestors \'none\';');
|
||||||
res.set('X-Frame-Options', 'DENY');
|
res.set('X-Frame-Options', 'DENY');
|
||||||
|
|
||||||
res.sendFile(indexHtml);
|
res.sendFile(indexHtml);
|
||||||
|
@@ -33,12 +33,12 @@ export default defineConfig({
|
|||||||
sidebar: {
|
sidebar: {
|
||||||
'/apps/': [
|
'/apps/': [
|
||||||
{
|
{
|
||||||
text: 'Better Stack',
|
text: 'Airbrake',
|
||||||
collapsible: true,
|
collapsible: true,
|
||||||
collapsed: true,
|
collapsed: true,
|
||||||
items: [
|
items: [
|
||||||
{ text: 'Actions', link: '/apps/better-stack/actions' },
|
{ text: 'Triggers', link: '/apps/airbrake/triggers' },
|
||||||
{ text: 'Connection', link: '/apps/better-stack/connection' },
|
{ text: 'Connection', link: '/apps/airbrake/connection' },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@@ -15,7 +15,7 @@ Please be careful with the `ENCRYPTION_KEY` and `WEBHOOK_SECRET_KEY` environment
|
|||||||
:::
|
:::
|
||||||
|
|
||||||
| Variable Name | Type | Default Value | Description |
|
| Variable Name | Type | Default Value | Description |
|
||||||
| --------------------------- | ------- | ------------------ | ---------------------------------------------------------------------------------------------------- |
|
| ---------------------------- | ------- | ------------------ | ----------------------------------------------------------------------------------- |
|
||||||
| `HOST` | string | `localhost` | HTTP Host |
|
| `HOST` | string | `localhost` | HTTP Host |
|
||||||
| `PROTOCOL` | string | `http` | HTTP Protocol |
|
| `PROTOCOL` | string | `http` | HTTP Protocol |
|
||||||
| `PORT` | string | `3000` | HTTP Port |
|
| `PORT` | string | `3000` | HTTP Port |
|
||||||
@@ -42,3 +42,5 @@ Please be careful with the `ENCRYPTION_KEY` and `WEBHOOK_SECRET_KEY` environment
|
|||||||
| `ENABLE_BULLMQ_DASHBOARD` | boolean | `false` | Enable BullMQ Dashboard |
|
| `ENABLE_BULLMQ_DASHBOARD` | boolean | `false` | Enable BullMQ Dashboard |
|
||||||
| `BULLMQ_DASHBOARD_USERNAME` | string | | Username to login BullMQ Dashboard |
|
| `BULLMQ_DASHBOARD_USERNAME` | string | | Username to login BullMQ Dashboard |
|
||||||
| `BULLMQ_DASHBOARD_PASSWORD` | string | | Password to login BullMQ Dashboard |
|
| `BULLMQ_DASHBOARD_PASSWORD` | string | | Password to login BullMQ Dashboard |
|
||||||
|
| `DISABLE_NOTIFICATIONS_PAGE` | boolean | `false` | Enable/Disable notifications page |
|
||||||
|
| `DISABLE_FAVICON` | boolean | `false` | Enable/Disable favicon |
|
||||||
|
13
packages/docs/pages/apps/airbrake/connection.md
Normal file
13
packages/docs/pages/apps/airbrake/connection.md
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# Airbrake
|
||||||
|
|
||||||
|
:::info
|
||||||
|
This page explains the steps you need to follow to set up the Airbrake
|
||||||
|
connection in Automatisch. If any of the steps are outdated, please let us know!
|
||||||
|
:::
|
||||||
|
|
||||||
|
1. Login to your Airbrake account: [https://www.airbrake.io/](https://www.airbrake.io/).
|
||||||
|
2. Go to your profile & notifications page.
|
||||||
|
3. Copy `Auth Token` from the page to the `Auth Token` field on Automatisch.
|
||||||
|
4. Fill the instance URL field with your subdomain. (https://{yoursubdomain}.airbrake.io)
|
||||||
|
5. Write any screen name to be displayed in Automatisch.
|
||||||
|
6. Now, you can start using the Airbrake connection with Automatisch.
|
12
packages/docs/pages/apps/airbrake/triggers.md
Normal file
12
packages/docs/pages/apps/airbrake/triggers.md
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
---
|
||||||
|
favicon: /favicons/airbrake.svg
|
||||||
|
items:
|
||||||
|
- name: New errors
|
||||||
|
desc: Triggers when a new error occurs.
|
||||||
|
---
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import CustomListing from '../../components/CustomListing.vue'
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<CustomListing />
|
@@ -1,18 +0,0 @@
|
|||||||
---
|
|
||||||
favicon: /favicons/better-stack.svg
|
|
||||||
items:
|
|
||||||
- name: Acknowledge incident
|
|
||||||
desc: Acknowledges an incident.
|
|
||||||
- name: Create incident
|
|
||||||
desc: Creates an incident that informs the team.
|
|
||||||
- name: Find incident
|
|
||||||
desc: Finds an incident.
|
|
||||||
- name: Resolve incident
|
|
||||||
desc: Resolves an incident.
|
|
||||||
---
|
|
||||||
|
|
||||||
<script setup>
|
|
||||||
import CustomListing from '../../components/CustomListing.vue'
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<CustomListing />
|
|
@@ -1,14 +0,0 @@
|
|||||||
# Better Stack
|
|
||||||
|
|
||||||
:::info
|
|
||||||
This page explains the steps you need to follow to set up the Better Stack
|
|
||||||
connection in Automatisch. If any of the steps are outdated, please let us know!
|
|
||||||
:::
|
|
||||||
|
|
||||||
1. Login to your Better Stack account: [https://betterstack.com/](https://betterstack.com/).
|
|
||||||
2. Click on the team name bottom left and select **Manage Teams** option.
|
|
||||||
3. Click on the three dots icon of your team and select **manage** option.
|
|
||||||
4. Click on the **API tokens** tab.
|
|
||||||
5. Copy the token next to **Direct API tokens** to the `API Key` field on Automatisch.
|
|
||||||
6. Fill the screen name on Automatisch.
|
|
||||||
7. Now, you can start using the Better Stack connection with Automatisch.
|
|
@@ -2,8 +2,8 @@
|
|||||||
|
|
||||||
The following integrations are currently supported by Automatisch.
|
The following integrations are currently supported by Automatisch.
|
||||||
|
|
||||||
- [Better Stack](/apps/better-stack/actions)
|
|
||||||
- [Carbone](/apps/carbone/actions)
|
- [Carbone](/apps/carbone/actions)
|
||||||
|
- [Airbrake](/apps/airbrake/triggers)
|
||||||
- [DeepL](/apps/deepl/actions)
|
- [DeepL](/apps/deepl/actions)
|
||||||
- [Delay](/apps/delay/actions)
|
- [Delay](/apps/delay/actions)
|
||||||
- [Discord](/apps/discord/actions)
|
- [Discord](/apps/discord/actions)
|
||||||
|
1
packages/docs/pages/public/favicons/airbrake.svg
Normal file
1
packages/docs/pages/public/favicons/airbrake.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg height="255" preserveAspectRatio="xMidYMid" viewBox="0 0 256 255" width="256" xmlns="http://www.w3.org/2000/svg"><path d="m128.636514 155.746615v-155.23361889h-3.522242v.06873152l-124.60824865 64.03287157v60.8642488h.00597665v3.234366h-.00597665v60.868233l124.60824865 64.747082h3.842989v-98.581914z" fill="#ff8e4a"/><path d="m129.941416 254.328529 125.568498-64.747082v-124.9668478l-125.887253-64.10160309h-2.243237v253.81055289h2.243237" fill="#f48746"/><path d="m109.097837 87.2551595h36.19561v59.2077195h-36.19561z" fill="#ff8e4a"/><path d="m66.1735097 188.397074h14.8639378c9.4102412 0 12.6087471-2.238257 15.6189883-9.988981l8.2796572-21.353587h45.159596l8.280653 21.353587c3.011238 7.750724 6.396016 9.988981 15.805261 9.988981h14.677665v-19.114335h-3.011237c-3.19751 0-4.704622-.689307-5.831222-3.790194l-39.516638-99.3658524h-25.779299l-39.703907 99.3658524c-1.1285915 3.100887-2.632716 3.790194-5.833214 3.790194h-3.0102413zm44.4075333-49.939922 11.478163-30.655253c2.445448-6.714771 5.269417-18.2556889 5.269417-18.2556889h.375533s2.822972 11.5409179 5.269416 18.2556889l11.478163 30.655253z" fill="#fff"/><path d="m231.204856 150.082739v-51.8086223c.235082 4.5233303 2.970397 16.8432063 24.305058 27.8512063v11.653479zm0-53.1623343v1.353712c-.029883-.5926848-.01793-1.0479066 0-1.353712zm.041837-.4392841s-.022911.1534008-.041837.4392841v-.4392841z" fill="#d4763c"/><path d="m231.155051 94.3016342c-.013946.9931207.05877 1.8945993.049805 2.0460078-.01793.2480312-2.220327 16.094132 24.305058 29.777681v-60.863253c-23.325883 12.0349884-24.449494 25.7414475-24.354863 29.0395642" fill="#ff8e4a"/></svg>
|
After Width: | Height: | Size: 1.6 KiB |
@@ -1,21 +0,0 @@
|
|||||||
<?xml version="1.0" standalone="no"?>
|
|
||||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN"
|
|
||||||
"http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd">
|
|
||||||
<svg version="1.0" xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="200.000000pt" height="200.000000pt" viewBox="0 0 200.000000 200.000000"
|
|
||||||
preserveAspectRatio="xMidYMid meet">
|
|
||||||
|
|
||||||
<g transform="translate(0.000000,200.000000) scale(0.100000,-0.100000)"
|
|
||||||
fill="#000000" stroke="none">
|
|
||||||
<path d="M0 1000 l0 -1000 1000 0 1000 0 0 1000 0 1000 -1000 0 -1000 0 0
|
|
||||||
-1000z m1162 460 c14 -11 113 -184 232 -408 228 -429 231 -439 175 -486 -35
|
|
||||||
-30 -30 -29 -140 -15 -89 12 -123 25 -152 56 -9 11 -72 147 -140 304 -113 263
|
|
||||||
-124 284 -149 287 -14 2 -29 10 -32 17 -8 21 67 214 94 242 28 29 78 30 112 3z
|
|
||||||
m-340 -148 c10 -10 72 -175 139 -367 114 -325 121 -351 108 -374 -8 -14 -27
|
|
||||||
-32 -41 -41 -25 -13 -34 -12 -126 18 -55 18 -111 43 -125 56 -19 17 -40 67
|
|
||||||
-76 182 -36 112 -58 164 -73 176 l-22 16 27 99 c63 224 66 232 95 248 31 17
|
|
||||||
69 12 94 -13z m-314 -219 c16 -15 26 -59 56 -243 42 -262 43 -285 17 -300 -11
|
|
||||||
-5 -24 -10 -30 -10 -19 0 -140 114 -150 141 -7 20 -4 76 10 191 10 90 19 171
|
|
||||||
19 181 0 18 33 57 49 57 5 0 18 -8 29 -17z"/>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 1.1 KiB |
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB |
@@ -2,7 +2,6 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
<link rel="icon" href="%PUBLIC_URL%/favicon.ico" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<meta name="theme-color" content="#0059F7" />
|
<meta name="theme-color" content="#0059F7" />
|
||||||
<meta
|
<meta
|
||||||
|
@@ -2,13 +2,6 @@
|
|||||||
"short_name": "automatisch",
|
"short_name": "automatisch",
|
||||||
"name": "automatisch",
|
"name": "automatisch",
|
||||||
"description": "Build workflow automation without spending time and money. No code is required.",
|
"description": "Build workflow automation without spending time and money. No code is required.",
|
||||||
"icons": [
|
|
||||||
{
|
|
||||||
"src": "favicon.ico",
|
|
||||||
"sizes": "64x64 32x32 24x24 16x16",
|
|
||||||
"type": "image/x-icon"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"start_url": ".",
|
"start_url": ".",
|
||||||
"display": "standalone",
|
"display": "standalone",
|
||||||
"theme_color": "#000000",
|
"theme_color": "#000000",
|
||||||
|
@@ -15,6 +15,7 @@ import { SvgIconComponent } from '@mui/icons-material';
|
|||||||
import AppBar from 'components/AppBar';
|
import AppBar from 'components/AppBar';
|
||||||
import Drawer from 'components/Drawer';
|
import Drawer from 'components/Drawer';
|
||||||
import * as URLS from 'config/urls';
|
import * as URLS from 'config/urls';
|
||||||
|
import useFormatMessage from 'hooks/useFormatMessage';
|
||||||
import useCurrentUserAbility from 'hooks/useCurrentUserAbility';
|
import useCurrentUserAbility from 'hooks/useCurrentUserAbility';
|
||||||
|
|
||||||
type SettingsLayoutProps = {
|
type SettingsLayoutProps = {
|
||||||
@@ -86,19 +87,11 @@ function createDrawerLinks({
|
|||||||
return items;
|
return items;
|
||||||
}
|
}
|
||||||
|
|
||||||
const drawerBottomLinks = [
|
|
||||||
{
|
|
||||||
Icon: ArrowBackIosNewIcon,
|
|
||||||
primary: 'adminSettingsDrawer.goBack',
|
|
||||||
to: '/',
|
|
||||||
dataTest: 'go-back-drawer-link',
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
export default function SettingsLayout({
|
export default function SettingsLayout({
|
||||||
children,
|
children,
|
||||||
}: SettingsLayoutProps): React.ReactElement {
|
}: SettingsLayoutProps): React.ReactElement {
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
const formatMessage = useFormatMessage();
|
||||||
const currentUserAbility = useCurrentUserAbility();
|
const currentUserAbility = useCurrentUserAbility();
|
||||||
const matchSmallScreens = useMediaQuery(theme.breakpoints.down('lg'));
|
const matchSmallScreens = useMediaQuery(theme.breakpoints.down('lg'));
|
||||||
const [isDrawerOpen, setDrawerOpen] = React.useState(!matchSmallScreens);
|
const [isDrawerOpen, setDrawerOpen] = React.useState(!matchSmallScreens);
|
||||||
@@ -116,6 +109,15 @@ export default function SettingsLayout({
|
|||||||
canUpdateApp: currentUserAbility.can('update', 'App'),
|
canUpdateApp: currentUserAbility.can('update', 'App'),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const drawerBottomLinks = [
|
||||||
|
{
|
||||||
|
Icon: ArrowBackIosNewIcon,
|
||||||
|
primary: formatMessage('adminSettingsDrawer.goBack'),
|
||||||
|
to: '/',
|
||||||
|
dataTest: 'go-back-drawer-link',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<AppBar
|
<AppBar
|
||||||
|
@@ -19,6 +19,7 @@ type DrawerLink = {
|
|||||||
Icon: React.ElementType;
|
Icon: React.ElementType;
|
||||||
primary: string;
|
primary: string;
|
||||||
to: string;
|
to: string;
|
||||||
|
target?: '_blank';
|
||||||
badgeContent?: React.ReactNode;
|
badgeContent?: React.ReactNode;
|
||||||
dataTest?: string;
|
dataTest?: string;
|
||||||
};
|
};
|
||||||
@@ -69,7 +70,7 @@ export default function Drawer(props: DrawerProps): React.ReactElement {
|
|||||||
|
|
||||||
<List sx={{ py: 0, mt: 3 }}>
|
<List sx={{ py: 0, mt: 3 }}>
|
||||||
{bottomLinks.map(
|
{bottomLinks.map(
|
||||||
({ Icon, badgeContent, primary, to, dataTest }, index) => (
|
({ Icon, badgeContent, primary, to, dataTest, target }, index) => (
|
||||||
<ListItemLink
|
<ListItemLink
|
||||||
key={`${to}-${index}`}
|
key={`${to}-${index}`}
|
||||||
icon={
|
icon={
|
||||||
@@ -77,9 +78,10 @@ export default function Drawer(props: DrawerProps): React.ReactElement {
|
|||||||
<Icon htmlColor={theme.palette.primary.main} />
|
<Icon htmlColor={theme.palette.primary.main} />
|
||||||
</Badge>
|
</Badge>
|
||||||
}
|
}
|
||||||
primary={formatMessage(primary)}
|
primary={primary}
|
||||||
to={to}
|
to={to}
|
||||||
onClick={closeOnClick}
|
onClick={closeOnClick}
|
||||||
|
target={target}
|
||||||
data-test={dataTest}
|
data-test={dataTest}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
|
@@ -7,12 +7,14 @@ import AppsIcon from '@mui/icons-material/Apps';
|
|||||||
import SwapCallsIcon from '@mui/icons-material/SwapCalls';
|
import SwapCallsIcon from '@mui/icons-material/SwapCalls';
|
||||||
import HistoryIcon from '@mui/icons-material/History';
|
import HistoryIcon from '@mui/icons-material/History';
|
||||||
import NotificationsIcon from '@mui/icons-material/Notifications';
|
import NotificationsIcon from '@mui/icons-material/Notifications';
|
||||||
|
import ArrowBackIosNew from '@mui/icons-material/ArrowBackIosNew';
|
||||||
|
|
||||||
import * as URLS from 'config/urls';
|
import * as URLS from 'config/urls';
|
||||||
|
import useFormatMessage from 'hooks/useFormatMessage';
|
||||||
import useVersion from 'hooks/useVersion';
|
import useVersion from 'hooks/useVersion';
|
||||||
import AppBar from 'components/AppBar';
|
import AppBar from 'components/AppBar';
|
||||||
import Drawer from 'components/Drawer';
|
import Drawer from 'components/Drawer';
|
||||||
import useAutomatischInfo from 'hooks/useAutomatischInfo';
|
import useConfig from 'hooks/useConfig';
|
||||||
|
|
||||||
type PublicLayoutProps = {
|
type PublicLayoutProps = {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
@@ -40,48 +42,95 @@ const drawerLinks = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
type GenerateDrawerBottomLinksOptions = {
|
type GenerateDrawerBottomLinksOptions = {
|
||||||
isMation: boolean;
|
disableNotificationsPage: boolean;
|
||||||
loading: boolean;
|
|
||||||
notificationBadgeContent: number;
|
notificationBadgeContent: number;
|
||||||
|
additionalDrawerLink?: string;
|
||||||
|
additionalDrawerLinkText?: string;
|
||||||
|
additionalDrawerLinkIcon?: string;
|
||||||
|
formatMessage: ReturnType<typeof useFormatMessage>;
|
||||||
};
|
};
|
||||||
|
|
||||||
const generateDrawerBottomLinks = ({
|
const generateDrawerBottomLinks = async ({
|
||||||
isMation,
|
disableNotificationsPage,
|
||||||
loading,
|
|
||||||
notificationBadgeContent = 0,
|
notificationBadgeContent = 0,
|
||||||
|
additionalDrawerLink,
|
||||||
|
additionalDrawerLinkText,
|
||||||
|
formatMessage,
|
||||||
}: GenerateDrawerBottomLinksOptions) => {
|
}: GenerateDrawerBottomLinksOptions) => {
|
||||||
if (loading || isMation) {
|
const notificationsPageLinkObject = {
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
Icon: NotificationsIcon,
|
Icon: NotificationsIcon,
|
||||||
primary: 'settingsDrawer.notifications',
|
primary: formatMessage('settingsDrawer.notifications'),
|
||||||
to: URLS.UPDATES,
|
to: URLS.UPDATES,
|
||||||
badgeContent: notificationBadgeContent,
|
badgeContent: notificationBadgeContent,
|
||||||
},
|
};
|
||||||
];
|
|
||||||
|
const hasAdditionalDrawerLink =
|
||||||
|
additionalDrawerLink && additionalDrawerLinkText;
|
||||||
|
|
||||||
|
const additionalDrawerLinkObject = {
|
||||||
|
Icon: ArrowBackIosNew,
|
||||||
|
primary: additionalDrawerLinkText || '',
|
||||||
|
to: additionalDrawerLink || '',
|
||||||
|
target: '_blank' as const,
|
||||||
|
};
|
||||||
|
|
||||||
|
const links = [];
|
||||||
|
|
||||||
|
if (!disableNotificationsPage) {
|
||||||
|
links.push(notificationsPageLinkObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasAdditionalDrawerLink) {
|
||||||
|
links.push(additionalDrawerLinkObject);
|
||||||
|
}
|
||||||
|
|
||||||
|
return links;
|
||||||
|
};
|
||||||
|
|
||||||
|
type Link = {
|
||||||
|
Icon: React.ElementType;
|
||||||
|
primary: string;
|
||||||
|
target?: '_blank';
|
||||||
|
to: string;
|
||||||
|
badgeContent?: React.ReactNode;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function PublicLayout({
|
export default function PublicLayout({
|
||||||
children,
|
children,
|
||||||
}: PublicLayoutProps): React.ReactElement {
|
}: PublicLayoutProps): React.ReactElement {
|
||||||
const version = useVersion();
|
const version = useVersion();
|
||||||
const { isMation, loading } = useAutomatischInfo();
|
const { config, loading } = useConfig([
|
||||||
|
'disableNotificationsPage',
|
||||||
|
'additionalDrawerLink',
|
||||||
|
'additionalDrawerLinkText',
|
||||||
|
]);
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
const formatMessage = useFormatMessage();
|
||||||
|
const [bottomLinks, setBottomLinks] = React.useState<Link[]>([]);
|
||||||
const matchSmallScreens = useMediaQuery(theme.breakpoints.down('lg'));
|
const matchSmallScreens = useMediaQuery(theme.breakpoints.down('lg'));
|
||||||
const [isDrawerOpen, setDrawerOpen] = React.useState(!matchSmallScreens);
|
const [isDrawerOpen, setDrawerOpen] = React.useState(!matchSmallScreens);
|
||||||
|
|
||||||
const openDrawer = () => setDrawerOpen(true);
|
const openDrawer = () => setDrawerOpen(true);
|
||||||
const closeDrawer = () => setDrawerOpen(false);
|
const closeDrawer = () => setDrawerOpen(false);
|
||||||
|
|
||||||
const drawerBottomLinks = generateDrawerBottomLinks({
|
React.useEffect(() => {
|
||||||
|
async function perform() {
|
||||||
|
const newBottomLinks = await generateDrawerBottomLinks({
|
||||||
notificationBadgeContent: version.newVersionCount,
|
notificationBadgeContent: version.newVersionCount,
|
||||||
loading,
|
disableNotificationsPage: config?.disableNotificationsPage as boolean,
|
||||||
isMation,
|
additionalDrawerLink: config?.additionalDrawerLink as string,
|
||||||
|
additionalDrawerLinkText: config?.additionalDrawerLinkText as string,
|
||||||
|
formatMessage,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
setBottomLinks(newBottomLinks);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (loading) return;
|
||||||
|
|
||||||
|
perform();
|
||||||
|
}, [config, loading, version.newVersionCount]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<AppBar
|
<AppBar
|
||||||
@@ -93,7 +142,7 @@ export default function PublicLayout({
|
|||||||
<Box sx={{ display: 'flex' }}>
|
<Box sx={{ display: 'flex' }}>
|
||||||
<Drawer
|
<Drawer
|
||||||
links={drawerLinks}
|
links={drawerLinks}
|
||||||
bottomLinks={drawerBottomLinks}
|
bottomLinks={bottomLinks}
|
||||||
open={isDrawerOpen}
|
open={isDrawerOpen}
|
||||||
onOpen={openDrawer}
|
onOpen={openDrawer}
|
||||||
onClose={closeDrawer}
|
onClose={closeDrawer}
|
||||||
|
@@ -9,6 +9,7 @@ type ListItemLinkProps = {
|
|||||||
icon: React.ReactNode;
|
icon: React.ReactNode;
|
||||||
primary: string;
|
primary: string;
|
||||||
to: string;
|
to: string;
|
||||||
|
target?: '_blank';
|
||||||
onClick?: (event: React.SyntheticEvent) => void;
|
onClick?: (event: React.SyntheticEvent) => void;
|
||||||
'data-test'?: string;
|
'data-test'?: string;
|
||||||
};
|
};
|
||||||
@@ -16,14 +17,29 @@ type ListItemLinkProps = {
|
|||||||
export default function ListItemLink(
|
export default function ListItemLink(
|
||||||
props: ListItemLinkProps
|
props: ListItemLinkProps
|
||||||
): React.ReactElement {
|
): React.ReactElement {
|
||||||
const { icon, primary, to, onClick, 'data-test': dataTest } = props;
|
const { icon, primary, to, onClick, 'data-test': dataTest, target } = props;
|
||||||
const selected = useMatch({ path: to, end: true });
|
const selected = useMatch({ path: to, end: true });
|
||||||
|
|
||||||
const CustomLink = React.useMemo(
|
const CustomLink = React.useMemo(
|
||||||
() =>
|
() =>
|
||||||
React.forwardRef<HTMLAnchorElement, Omit<LinkProps, 'to'>>(
|
React.forwardRef<HTMLAnchorElement, Omit<LinkProps, 'to'>>(
|
||||||
function InLineLink(linkProps, ref) {
|
function InLineLink(linkProps, ref) {
|
||||||
return <Link ref={ref} to={to} {...linkProps} />;
|
try {
|
||||||
|
// challenge the link to check if it's absolute URL
|
||||||
|
new URL(to); // should throw an error if it's not an absolute URL
|
||||||
|
|
||||||
|
return (
|
||||||
|
<a
|
||||||
|
{...linkProps}
|
||||||
|
ref={ref}
|
||||||
|
href={to}
|
||||||
|
target={target}
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} catch {
|
||||||
|
return <Link ref={ref} {...linkProps} to={to} />;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
[to]
|
[to]
|
||||||
@@ -37,6 +53,7 @@ export default function ListItemLink(
|
|||||||
selected={!!selected}
|
selected={!!selected}
|
||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
data-test={dataTest}
|
data-test={dataTest}
|
||||||
|
target={target}
|
||||||
>
|
>
|
||||||
<ListItemIcon sx={{ minWidth: 52 }}>{icon}</ListItemIcon>
|
<ListItemIcon sx={{ minWidth: 52 }}>{icon}</ListItemIcon>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
|
@@ -15,6 +15,27 @@ const MetadataProvider = ({
|
|||||||
document.title = (config?.title as string) || 'Automatisch';
|
document.title = (config?.title as string) || 'Automatisch';
|
||||||
}, [config?.title]);
|
}, [config?.title]);
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
const existingFaviconElement = document.querySelector(
|
||||||
|
"link[rel~='icon']"
|
||||||
|
) as HTMLLinkElement | null;
|
||||||
|
|
||||||
|
if (config?.disableFavicon === true) {
|
||||||
|
existingFaviconElement?.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (config?.disableFavicon === false) {
|
||||||
|
if (existingFaviconElement) {
|
||||||
|
existingFaviconElement.href = '/browser-tab.ico';
|
||||||
|
} else {
|
||||||
|
const newFaviconElement = document.createElement('link');
|
||||||
|
newFaviconElement.rel = 'icon';
|
||||||
|
document.head.appendChild(newFaviconElement);
|
||||||
|
newFaviconElement.href = '/browser-tab.ico';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [config?.disableFavicon]);
|
||||||
|
|
||||||
return <>{children}</>;
|
return <>{children}</>;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -9,6 +9,7 @@ import PaymentIcon from '@mui/icons-material/Payment';
|
|||||||
|
|
||||||
import * as URLS from 'config/urls';
|
import * as URLS from 'config/urls';
|
||||||
import useAutomatischInfo from 'hooks/useAutomatischInfo';
|
import useAutomatischInfo from 'hooks/useAutomatischInfo';
|
||||||
|
import useFormatMessage from 'hooks/useFormatMessage';
|
||||||
import AppBar from 'components/AppBar';
|
import AppBar from 'components/AppBar';
|
||||||
import Drawer from 'components/Drawer';
|
import Drawer from 'components/Drawer';
|
||||||
|
|
||||||
@@ -22,8 +23,8 @@ function createDrawerLinks({ isCloud }: { isCloud: boolean }) {
|
|||||||
Icon: AccountCircleIcon,
|
Icon: AccountCircleIcon,
|
||||||
primary: 'settingsDrawer.myProfile',
|
primary: 'settingsDrawer.myProfile',
|
||||||
to: URLS.SETTINGS_PROFILE,
|
to: URLS.SETTINGS_PROFILE,
|
||||||
}
|
},
|
||||||
]
|
];
|
||||||
|
|
||||||
if (isCloud) {
|
if (isCloud) {
|
||||||
items.push({
|
items.push({
|
||||||
@@ -36,19 +37,12 @@ function createDrawerLinks({ isCloud }: { isCloud: boolean }) {
|
|||||||
return items;
|
return items;
|
||||||
}
|
}
|
||||||
|
|
||||||
const drawerBottomLinks = [
|
|
||||||
{
|
|
||||||
Icon: ArrowBackIosNewIcon,
|
|
||||||
primary: 'settingsDrawer.goBack',
|
|
||||||
to: '/',
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
export default function SettingsLayout({
|
export default function SettingsLayout({
|
||||||
children,
|
children,
|
||||||
}: SettingsLayoutProps): React.ReactElement {
|
}: SettingsLayoutProps): React.ReactElement {
|
||||||
const { isCloud } = useAutomatischInfo();
|
const { isCloud } = useAutomatischInfo();
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
const formatMessage = useFormatMessage();
|
||||||
const matchSmallScreens = useMediaQuery(theme.breakpoints.down('lg'));
|
const matchSmallScreens = useMediaQuery(theme.breakpoints.down('lg'));
|
||||||
const [isDrawerOpen, setDrawerOpen] = React.useState(!matchSmallScreens);
|
const [isDrawerOpen, setDrawerOpen] = React.useState(!matchSmallScreens);
|
||||||
|
|
||||||
@@ -56,6 +50,14 @@ export default function SettingsLayout({
|
|||||||
const closeDrawer = () => setDrawerOpen(false);
|
const closeDrawer = () => setDrawerOpen(false);
|
||||||
const drawerLinks = createDrawerLinks({ isCloud });
|
const drawerLinks = createDrawerLinks({ isCloud });
|
||||||
|
|
||||||
|
const drawerBottomLinks = [
|
||||||
|
{
|
||||||
|
Icon: ArrowBackIosNewIcon,
|
||||||
|
primary: formatMessage('settingsDrawer.goBack'),
|
||||||
|
to: '/',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<AppBar
|
<AppBar
|
||||||
|
12
yarn.lock
12
yarn.lock
@@ -15728,9 +15728,9 @@ vite-node@1.1.3:
|
|||||||
vite "^5.0.0"
|
vite "^5.0.0"
|
||||||
|
|
||||||
vite@^3.1.6:
|
vite@^3.1.6:
|
||||||
version "3.2.7"
|
version "3.2.8"
|
||||||
resolved "https://registry.npmjs.org/vite/-/vite-3.2.7.tgz"
|
resolved "https://registry.yarnpkg.com/vite/-/vite-3.2.8.tgz#0697e13addf99ed44b838b8462a3a922fdd9d37b"
|
||||||
integrity sha512-29pdXjk49xAP0QBr0xXqu2s5jiQIXNvE/xwd0vUizYT2Hzqe4BksNNoWllFVXJf4eLZ+UlVQmXfB4lWrc+t18g==
|
integrity sha512-EtQU16PLIJpAZol2cTLttNP1mX6L0SyI0pgQB1VOoWeQnMSvtiwovV3D6NcjN8CZQWWyESD2v5NGnpz5RvgOZA==
|
||||||
dependencies:
|
dependencies:
|
||||||
esbuild "^0.15.9"
|
esbuild "^0.15.9"
|
||||||
postcss "^8.4.18"
|
postcss "^8.4.18"
|
||||||
@@ -15740,9 +15740,9 @@ vite@^3.1.6:
|
|||||||
fsevents "~2.3.2"
|
fsevents "~2.3.2"
|
||||||
|
|
||||||
vite@^5.0.0:
|
vite@^5.0.0:
|
||||||
version "5.0.11"
|
version "5.0.12"
|
||||||
resolved "https://registry.yarnpkg.com/vite/-/vite-5.0.11.tgz#31562e41e004cb68e1d51f5d2c641ab313b289e4"
|
resolved "https://registry.yarnpkg.com/vite/-/vite-5.0.12.tgz#8a2ffd4da36c132aec4adafe05d7adde38333c47"
|
||||||
integrity sha512-XBMnDjZcNAw/G1gEiskiM1v6yzM4GE5aMGvhWTlHAYYhxb7S3/V1s3m2LDHa8Vh6yIWYYB0iJwsEaS523c4oYA==
|
integrity sha512-4hsnEkG3q0N4Tzf1+t6NdN9dg/L3BM+q8SWgbSPnJvrgH2kgdyzfVJwbR1ic69/4uMJJ/3dqDZZE5/WwqW8U1w==
|
||||||
dependencies:
|
dependencies:
|
||||||
esbuild "^0.19.3"
|
esbuild "^0.19.3"
|
||||||
postcss "^8.4.32"
|
postcss "^8.4.32"
|
||||||
|
Reference in New Issue
Block a user