@pagerduty/backstage-plugin-scaffolder-actions 0.2.2 → 0.2.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/config.d.ts +32 -29
- package/dist/actions/custom.cjs.js +83 -0
- package/dist/actions/custom.cjs.js.map +1 -0
- package/dist/apis/pagerduty.cjs.js +451 -0
- package/dist/apis/pagerduty.cjs.js.map +1 -0
- package/dist/auth/auth.cjs.js +260 -0
- package/dist/auth/auth.cjs.js.map +1 -0
- package/dist/index.cjs.js +4 -664
- package/dist/index.cjs.js.map +1 -1
- package/dist/index.d.ts +5 -4
- package/dist/module.cjs.js +30 -0
- package/dist/module.cjs.js.map +1 -0
- package/package.json +24 -19
- package/LICENSE +0 -201
- package/README.md +0 -42
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pagerduty.cjs.js","sources":["../../src/apis/pagerduty.ts"],"sourcesContent":["import fetch from 'node-fetch';\nimport type { RequestInit, Response } from 'node-fetch';\n\nimport { getAuthToken } from '../auth/auth';\nimport { CreateServiceResponse } from '../types';\n\nimport {\n PagerDutyServiceResponse,\n PagerDutyIntegrationResponse,\n PagerDutyAbilitiesResponse,\n PagerDutyAccountConfig,\n PagerDutyEscalationPolicy,\n HttpError,\n PagerDutyEscalationPoliciesResponse,\n} from '@pagerduty/backstage-plugin-common';\nimport {\n LoggerService,\n RootConfigService,\n} from '@backstage/backend-plugin-api';\nimport { Config } from '@backstage/config';\n\ntype JsonValue = boolean | number | string | null | JsonArray | JsonObject;\n\ninterface JsonObject {\n [x: string]: JsonValue;\n}\n\ntype JsonArray = JsonValue[];\n\nexport type LoadEndpointConfigProps = {\n config: RootConfigService | undefined;\n legacyConfig: Config;\n logger: LoggerService;\n};\n\nexport type PagerDutyEndpointConfig = {\n eventsBaseUrl: string;\n apiBaseUrl: string;\n};\n\nconst EndpointConfig: Record<string, PagerDutyEndpointConfig> = {};\nlet fallbackEndpointConfig: PagerDutyEndpointConfig;\nlet isLegacyConfig = false;\n\nlet _config: RootConfigService | undefined;\nlet _legacyConfig: Config;\nlet _logger: LoggerService;\n\nexport function setFallbackEndpointConfig(account: PagerDutyAccountConfig) {\n fallbackEndpointConfig = {\n eventsBaseUrl:\n account.eventsBaseUrl !== undefined\n ? account.eventsBaseUrl\n : 'https://events.pagerduty.com/v2',\n apiBaseUrl:\n account.apiBaseUrl !== undefined\n ? account.apiBaseUrl\n : 'https://api.pagerduty.com',\n };\n}\n\nexport function insertEndpointConfig(account: PagerDutyAccountConfig) {\n EndpointConfig[account.id] = {\n eventsBaseUrl:\n account.eventsBaseUrl !== undefined\n ? account.eventsBaseUrl\n : 'https://events.pagerduty.com/v2',\n apiBaseUrl:\n account.apiBaseUrl !== undefined\n ? account.apiBaseUrl\n : 'https://api.pagerduty.com',\n };\n}\n\nexport function loadPagerDutyEndpointsFromConfig({\n config,\n legacyConfig,\n logger,\n}: LoadEndpointConfigProps) {\n // set config and logger\n _config = config;\n _legacyConfig = legacyConfig;\n _logger = logger;\n\n if (readOptionalObject('pagerDuty.accounts')) {\n _logger.debug(\n `New accounts configuration detected. Loading PagerDuty endpoints from config.`,\n );\n isLegacyConfig = false;\n\n const accounts: PagerDutyAccountConfig[] = JSON.parse(\n JSON.stringify(readOptionalObject('pagerDuty.accounts')),\n );\n\n if (accounts?.length === 1) {\n _logger.debug(\n `Single account configuration detected. Loading PagerDuty endpoints from config to 'default'.`,\n );\n EndpointConfig.default = {\n eventsBaseUrl:\n accounts[0].eventsBaseUrl !== undefined\n ? accounts[0].eventsBaseUrl\n : 'https://events.pagerduty.com/v2',\n apiBaseUrl:\n accounts[0].apiBaseUrl !== undefined\n ? accounts[0].apiBaseUrl\n : 'https://api.pagerduty.com',\n };\n } else {\n _logger.debug(\n `Multiple account configuration detected. Loading PagerDuty endpoints from config.`,\n );\n accounts?.forEach(account => {\n if (account.isDefault) {\n setFallbackEndpointConfig(account);\n }\n\n insertEndpointConfig(account);\n });\n }\n } else {\n _logger.debug(`Loading legacy PagerDuty endpoints from config.`);\n isLegacyConfig = true;\n\n EndpointConfig.default = {\n eventsBaseUrl:\n readOptionalString('pagerDuty.eventsBaseUrl') !== undefined\n ? readString('pagerDuty.eventsBaseUrl')\n : 'https://events.pagerduty.com/v2',\n apiBaseUrl:\n readOptionalString('pagerDuty.apiBaseUrl') !== undefined\n ? readString('pagerDuty.apiBaseUrl')\n : 'https://api.pagerduty.com',\n };\n }\n}\n\nexport function getApiBaseUrl(account?: string): string {\n if (isLegacyConfig === true) {\n return EndpointConfig.default.apiBaseUrl;\n }\n\n if (account) {\n return EndpointConfig[account].apiBaseUrl;\n }\n\n return fallbackEndpointConfig.apiBaseUrl;\n}\n\nexport type CreateServiceProps = {\n name: string;\n description: string;\n escalationPolicyId: string;\n account?: string;\n alertGrouping?: string;\n};\n\n// Supporting custom actions\nexport async function createService({\n name,\n description,\n escalationPolicyId,\n account,\n alertGrouping,\n}: CreateServiceProps): Promise<CreateServiceResponse> {\n let alertGroupingParameters = 'null';\n let response: Response;\n\n const apiBaseUrl = getApiBaseUrl(account);\n const baseUrl = `${apiBaseUrl}/services`;\n\n // Set default body\n let body = JSON.stringify({\n service: {\n type: 'service',\n name: name,\n description: description,\n alert_creation: 'create_alerts_and_incidents',\n auto_pause_notifications_parameters: {\n enabled: true,\n timeout: 300,\n },\n escalation_policy: {\n id: escalationPolicyId,\n type: 'escalation_policy_reference',\n },\n },\n });\n\n // Override body if alert grouping is enabled and passed as parameter\n if (\n (await isEventNoiseReductionEnabled(account)) &&\n alertGrouping !== undefined\n ) {\n alertGroupingParameters = alertGrouping;\n\n switch (alertGroupingParameters) {\n case 'intelligent':\n body = JSON.stringify({\n service: {\n type: 'service',\n name: name,\n description: description,\n escalation_policy: {\n id: escalationPolicyId,\n type: 'escalation_policy_reference',\n },\n alert_creation: 'create_alerts_and_incidents',\n alert_grouping_parameters: {\n type: alertGroupingParameters,\n },\n auto_pause_notifications_parameters: {\n enabled: true,\n timeout: 300,\n },\n },\n });\n break;\n case 'time':\n body = JSON.stringify({\n service: {\n type: 'service',\n name: name,\n description: description,\n escalation_policy: {\n id: escalationPolicyId,\n type: 'escalation_policy_reference',\n },\n alert_creation: 'create_alerts_and_incidents',\n alert_grouping_parameters: {\n type: alertGroupingParameters,\n config: {\n timeout: 0,\n },\n },\n auto_pause_notifications_parameters: {\n enabled: true,\n timeout: 300,\n },\n },\n });\n break;\n case 'content_based':\n body = JSON.stringify({\n service: {\n type: 'service',\n name: name,\n description: description,\n escalation_policy: {\n id: escalationPolicyId,\n type: 'escalation_policy_reference',\n },\n alert_creation: 'create_alerts_and_incidents',\n alert_grouping_parameters: {\n type: alertGroupingParameters,\n config: {\n aggregate: 'all',\n time_window: 0,\n fields: ['source', 'summary'],\n },\n },\n auto_pause_notifications_parameters: {\n enabled: true,\n timeout: 300,\n },\n },\n });\n break;\n default:\n break;\n }\n }\n\n const token = await getAuthToken(account);\n\n const options: RequestInit = {\n method: 'POST',\n body: body,\n headers: {\n Authorization: token,\n Accept: 'application/vnd.pagerduty+json;version=2',\n 'Content-Type': 'application/json',\n },\n };\n\n try {\n response = await fetch(baseUrl, options);\n } catch (error) {\n throw new Error(`Failed to create service: ${error}`);\n }\n\n switch (response.status) {\n case 400:\n throw new Error(\n `Failed to create service. Caller provided invalid arguments.`,\n );\n case 401:\n throw new Error(\n `Failed to create service. Caller did not supply credentials or did not provide the correct credentials.`,\n );\n case 402:\n throw new Error(\n `Failed to create service. Account does not have the abilities to perform the action.`,\n );\n case 403:\n throw new Error(\n `Failed to create service. Caller is not authorized to view the requested resource.`,\n );\n default: // 201\n break;\n }\n\n let result: PagerDutyServiceResponse;\n try {\n result = (await response.json()) as PagerDutyServiceResponse;\n\n const createServiceResult: CreateServiceResponse = {\n url: result.service.html_url,\n id: result.service.id,\n alertGrouping: alertGroupingParameters,\n };\n\n return createServiceResult;\n } catch (error) {\n throw new Error(`Failed to parse service information: ${error}`);\n }\n}\n\nexport type CreateServiceIntegrationProps = {\n serviceId: string;\n vendorId: string;\n account?: string;\n};\n\nexport async function createServiceIntegration({\n serviceId,\n vendorId,\n account,\n}: CreateServiceIntegrationProps): Promise<string> {\n let response: Response;\n\n const apiBaseUrl = getApiBaseUrl(account);\n const baseUrl = `${apiBaseUrl}/services`;\n const token = await getAuthToken(account);\n\n const options: RequestInit = {\n method: 'POST',\n body: JSON.stringify({\n integration: {\n name: 'Backstage',\n service: {\n id: serviceId,\n type: 'service_reference',\n },\n vendor: {\n id: vendorId,\n type: 'vendor_reference',\n },\n },\n }),\n headers: {\n Authorization: token,\n Accept: 'application/vnd.pagerduty+json;version=2',\n 'Content-Type': 'application/json',\n },\n };\n\n try {\n response = await fetch(`${baseUrl}/${serviceId}/integrations`, options);\n } catch (error) {\n throw new Error(`Failed to create service integration: ${error}`);\n }\n\n switch (response.status) {\n case 400:\n throw new Error(\n `Failed to create service integration. Caller provided invalid arguments.`,\n );\n case 401:\n throw new Error(\n `Failed to create service integration. Caller did not supply credentials or did not provide the correct credentials.`,\n );\n case 403:\n throw new Error(\n `Failed to create service integration. Caller is not authorized to view the requested resource.`,\n );\n case 429:\n throw new Error(\n `Failed to create service integration. Rate limit exceeded.`,\n );\n default: // 201\n break;\n }\n\n let result: PagerDutyIntegrationResponse;\n try {\n result = (await response.json()) as PagerDutyIntegrationResponse;\n\n return result.integration.integration_key ?? '';\n } catch (error) {\n throw new Error(`Failed to parse service information: ${error}`);\n }\n}\n\nexport async function isEventNoiseReductionEnabled(\n account?: string,\n): Promise<boolean> {\n let response: Response;\n const baseUrl = getApiBaseUrl(account);\n const token = await getAuthToken(account);\n\n const options: RequestInit = {\n method: 'GET',\n headers: {\n Authorization: token,\n Accept: 'application/vnd.pagerduty+json;version=2',\n 'Content-Type': 'application/json',\n },\n };\n\n try {\n response = await fetch(`${baseUrl}/abilities`, options);\n } catch (error) {\n throw new Error(`Failed to read abilities: ${error}`);\n }\n\n switch (response.status) {\n case 401:\n throw new Error(\n `Failed to read abilities. Caller did not supply credentials or did not provide the correct credentials.`,\n );\n case 403:\n throw new Error(\n `Failed to read abilities. Caller is not authorized to view the requested resource.`,\n );\n case 429:\n throw new Error(`Failed to read abilities. Rate limit exceeded.`);\n default: // 200\n break;\n }\n\n let result: PagerDutyAbilitiesResponse;\n try {\n result = (await response.json()) as PagerDutyAbilitiesResponse;\n\n if (\n result.abilities.includes('preview_intelligent_alert_grouping') &&\n result.abilities.includes('time_based_alert_grouping')\n ) {\n return true;\n }\n\n return false;\n } catch (error) {\n throw new Error(`Failed to parse abilities information: ${error}`);\n }\n}\n\nfunction readOptionalString(key: string): string | undefined {\n if (!_config) {\n return _legacyConfig.getOptionalString(key);\n }\n\n return _config.getOptionalString(key);\n}\n\nfunction readOptionalObject(key: string): JsonValue | undefined {\n if (!_config) {\n return _legacyConfig.getOptional(key);\n }\n\n return _config.getOptional(key);\n}\n\nfunction readString(key: string): string {\n if (!_config) {\n return _legacyConfig.getString(key);\n }\n\n return _config.getString(key);\n}\n\nexport async function getAccountByEscalationPolicyId(\n escalationPolicyId: string,\n): Promise<string> {\n const escalationPoliciesList: PagerDutyEscalationPolicy[] =\n await getAllEscalationPolicies();\n\n // find escalation policy by id and return account\n const escalationPolicy = escalationPoliciesList.find(\n policy => policy.id === escalationPolicyId,\n );\n\n return escalationPolicy?.account ?? '';\n}\n\nasync function getEscalationPolicies(\n offset: number,\n limit: number,\n account?: string,\n): Promise<[Boolean, PagerDutyEscalationPolicy[]]> {\n let response: Response;\n const params = `total=true&sort_by=name&offset=${offset}&limit=${limit}`;\n const options: RequestInit = {\n method: 'GET',\n headers: {\n Authorization: await getAuthToken(account),\n Accept: 'application/vnd.pagerduty+json;version=2',\n 'Content-Type': 'application/json',\n },\n };\n\n const apiBaseUrl = getApiBaseUrl(account);\n const baseUrl = `${apiBaseUrl}/escalation_policies`;\n\n try {\n response = await fetch(`${baseUrl}?${params}`, options);\n } catch (error) {\n throw new Error(`Failed to retrieve escalation policies: ${error}`);\n }\n\n switch (response.status) {\n case 400:\n throw new HttpError(\n 'Failed to list escalation policies. Caller provided invalid arguments.',\n 400,\n );\n case 401:\n throw new HttpError(\n 'Failed to list escalation policies. Caller did not supply credentials or did not provide the correct credentials.',\n 401,\n );\n case 403:\n throw new HttpError(\n 'Failed to list escalation policies. Caller is not authorized to view the requested resource.',\n 403,\n );\n case 429:\n throw new HttpError(\n 'Failed to list escalation policies. Rate limit exceeded.',\n 429,\n );\n default: // 200\n break;\n }\n\n let result: PagerDutyEscalationPoliciesResponse;\n try {\n result = (await response.json()) as PagerDutyEscalationPoliciesResponse;\n\n return [result.more ?? false, result.escalation_policies];\n } catch (error) {\n throw new HttpError(\n `Failed to parse escalation policy information: ${error}`,\n 500,\n );\n }\n}\n\nexport async function getAllEscalationPolicies(): Promise<\n PagerDutyEscalationPolicy[]\n> {\n const limit = 50;\n let offset = 0;\n let moreResults = false;\n let results: PagerDutyEscalationPolicy[] = [];\n\n await Promise.all(\n Object.keys(EndpointConfig).map(async account => {\n try {\n // reset offset value\n offset = 0;\n\n do {\n const res = await getEscalationPolicies(offset, limit, account);\n\n // set account for each escalation policy\n res[1].forEach(policy => {\n policy.account = account;\n });\n\n // update results\n results = results.concat(res[1]);\n\n // if more results exist\n if (res[0] === true) {\n moreResults = true;\n offset += limit;\n } else {\n moreResults = false;\n }\n } while (moreResults === true);\n } catch (error) {\n if (error instanceof HttpError) {\n throw error;\n } else {\n throw new HttpError(`${error}`, 500);\n }\n }\n }),\n );\n\n return results;\n}\n"],"names":["getAuthToken","fetch","HttpError"],"mappings":"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"}
|
|
@@ -0,0 +1,260 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var backstagePluginCommon = require('@pagerduty/backstage-plugin-common');
|
|
4
|
+
|
|
5
|
+
let authPersistence;
|
|
6
|
+
let isLegacyConfig = false;
|
|
7
|
+
let _config;
|
|
8
|
+
let _legacyConfig;
|
|
9
|
+
let _logger;
|
|
10
|
+
async function checkForOAuthToken(tokenId) {
|
|
11
|
+
if (authPersistence.accountTokens[tokenId]?.authToken !== "" && authPersistence.accountTokens[tokenId]?.authToken.includes("Bearer")) {
|
|
12
|
+
if (authPersistence.accountTokens[tokenId].authTokenExpiryDate > Date.now()) {
|
|
13
|
+
return true;
|
|
14
|
+
}
|
|
15
|
+
_logger.info("OAuth token expired, renewing");
|
|
16
|
+
await loadAuthConfig({
|
|
17
|
+
config: _config,
|
|
18
|
+
legacyConfig: _legacyConfig,
|
|
19
|
+
logger: _logger
|
|
20
|
+
});
|
|
21
|
+
return authPersistence.accountTokens[tokenId].authTokenExpiryDate > Date.now();
|
|
22
|
+
}
|
|
23
|
+
return false;
|
|
24
|
+
}
|
|
25
|
+
async function getAuthToken(accountId) {
|
|
26
|
+
if (!authPersistence?.accountTokens) {
|
|
27
|
+
_logger.debug("Auth config not loaded. Loading auth config...");
|
|
28
|
+
await loadAuthConfig({
|
|
29
|
+
config: _config,
|
|
30
|
+
legacyConfig: _legacyConfig,
|
|
31
|
+
logger: _logger
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
if (isLegacyConfig) {
|
|
35
|
+
_logger.debug("Using legacy config for auth token retrieval.");
|
|
36
|
+
if (authPersistence.accountTokens.default.authToken !== "" && (await checkForOAuthToken("default") || authPersistence.accountTokens.default.authToken.includes("Token"))) {
|
|
37
|
+
return authPersistence.accountTokens.default.authToken;
|
|
38
|
+
}
|
|
39
|
+
} else {
|
|
40
|
+
_logger.debug("Using new config for auth token retrieval.");
|
|
41
|
+
if (accountId && accountId !== "") {
|
|
42
|
+
if (authPersistence.accountTokens[accountId]?.authToken !== "" && (await checkForOAuthToken(accountId) || authPersistence.accountTokens[accountId]?.authToken.includes("Token"))) {
|
|
43
|
+
return authPersistence.accountTokens[accountId].authToken;
|
|
44
|
+
}
|
|
45
|
+
} else {
|
|
46
|
+
const defaultFallback = authPersistence.defaultAccount ?? "";
|
|
47
|
+
_logger.debug("No account ID provided. Using default account token.");
|
|
48
|
+
if (authPersistence.accountTokens[defaultFallback]?.authToken !== "" && (await checkForOAuthToken(defaultFallback) || authPersistence.accountTokens[defaultFallback]?.authToken.includes(
|
|
49
|
+
"Token"
|
|
50
|
+
))) {
|
|
51
|
+
return authPersistence.accountTokens[defaultFallback].authToken;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
return "";
|
|
56
|
+
}
|
|
57
|
+
async function loadAuthConfig({
|
|
58
|
+
config,
|
|
59
|
+
legacyConfig,
|
|
60
|
+
logger
|
|
61
|
+
}) {
|
|
62
|
+
try {
|
|
63
|
+
const defaultAccountId = "default";
|
|
64
|
+
_config = config;
|
|
65
|
+
_legacyConfig = legacyConfig;
|
|
66
|
+
_logger = logger;
|
|
67
|
+
authPersistence = {
|
|
68
|
+
accountTokens: {},
|
|
69
|
+
config: _config,
|
|
70
|
+
legacyConfig: _legacyConfig,
|
|
71
|
+
logger: _logger
|
|
72
|
+
};
|
|
73
|
+
if (!readOptionalObject("pagerDuty.accounts")) {
|
|
74
|
+
isLegacyConfig = true;
|
|
75
|
+
logger.warn(
|
|
76
|
+
"No PagerDuty accounts configuration found in config file. Reverting to legacy configuration."
|
|
77
|
+
);
|
|
78
|
+
if (!readOptionalString("pagerDuty.apiToken")) {
|
|
79
|
+
logger.warn(
|
|
80
|
+
"No PagerDuty API token found in config file. Trying OAuth token instead..."
|
|
81
|
+
);
|
|
82
|
+
if (!readOptionalObject("pagerDuty.oauth")) {
|
|
83
|
+
logger.error(
|
|
84
|
+
"No PagerDuty OAuth configuration found in config file."
|
|
85
|
+
);
|
|
86
|
+
} else if (!readOptionalString("pagerDuty.oauth.clientId") || !readOptionalString("pagerDuty.oauth.clientSecret") || !readOptionalString("pagerDuty.oauth.subDomain")) {
|
|
87
|
+
logger.error(
|
|
88
|
+
"Missing required PagerDuty OAuth parameters in config file. 'clientId', 'clientSecret', and 'subDomain' are required. 'region' is optional."
|
|
89
|
+
);
|
|
90
|
+
} else {
|
|
91
|
+
const tokenInfo = await getOAuthToken(
|
|
92
|
+
readString("pagerDuty.oauth.clientId"),
|
|
93
|
+
readString("pagerDuty.oauth.clientSecret"),
|
|
94
|
+
readString("pagerDuty.oauth.subDomain"),
|
|
95
|
+
readOptionalString("pagerDuty.oauth.region") ?? "us"
|
|
96
|
+
);
|
|
97
|
+
authPersistence.accountTokens[defaultAccountId] = tokenInfo;
|
|
98
|
+
logger.debug("PagerDuty OAuth configuration loaded successfully.");
|
|
99
|
+
}
|
|
100
|
+
} else {
|
|
101
|
+
authPersistence.accountTokens[defaultAccountId] = {
|
|
102
|
+
authToken: `Token token=${readOptionalString("pagerDuty.apiToken")}`,
|
|
103
|
+
authTokenExpiryDate: Date.now() + 36e5 * 24 * 365 * 2
|
|
104
|
+
// 2 years
|
|
105
|
+
};
|
|
106
|
+
logger.debug("PagerDuty API token loaded successfully.");
|
|
107
|
+
}
|
|
108
|
+
} else {
|
|
109
|
+
logger.debug(
|
|
110
|
+
"New PagerDuty accounts configuration found in config file."
|
|
111
|
+
);
|
|
112
|
+
isLegacyConfig = false;
|
|
113
|
+
const accounts = JSON.parse(
|
|
114
|
+
JSON.stringify(readOptionalObject("pagerDuty.accounts"))
|
|
115
|
+
);
|
|
116
|
+
if (accounts && accounts?.length === 1) {
|
|
117
|
+
logger.debug(
|
|
118
|
+
"Only one account found in config file. Setting it as default."
|
|
119
|
+
);
|
|
120
|
+
authPersistence.defaultAccount = accounts[0].id;
|
|
121
|
+
}
|
|
122
|
+
await Promise.all(
|
|
123
|
+
accounts?.map(async (account) => {
|
|
124
|
+
const maskedAccountId = maskString(account.id);
|
|
125
|
+
if (account.isDefault && !authPersistence.defaultAccount) {
|
|
126
|
+
logger.debug(
|
|
127
|
+
`Default account found in config file. Setting it as default.`
|
|
128
|
+
);
|
|
129
|
+
authPersistence.defaultAccount = account.id;
|
|
130
|
+
}
|
|
131
|
+
if (!account.apiToken) {
|
|
132
|
+
logger.warn(
|
|
133
|
+
"No PagerDuty API token found in config file. Trying OAuth token instead..."
|
|
134
|
+
);
|
|
135
|
+
if (!account.oauth) {
|
|
136
|
+
logger.error(
|
|
137
|
+
"No PagerDuty OAuth configuration found in config file."
|
|
138
|
+
);
|
|
139
|
+
} else if (!account.oauth.clientId || !account.oauth.clientSecret || !account.oauth.subDomain) {
|
|
140
|
+
logger.error(
|
|
141
|
+
"Missing required PagerDuty OAuth parameters in config file. 'clientId', 'clientSecret', and 'subDomain' are required. 'region' is optional."
|
|
142
|
+
);
|
|
143
|
+
} else {
|
|
144
|
+
const tokenInfo = await getOAuthToken(
|
|
145
|
+
account.oauth.clientId,
|
|
146
|
+
account.oauth.clientSecret,
|
|
147
|
+
account.oauth.subDomain,
|
|
148
|
+
account.oauth.region ?? "us"
|
|
149
|
+
);
|
|
150
|
+
authPersistence.accountTokens[account.id] = tokenInfo;
|
|
151
|
+
logger.debug(
|
|
152
|
+
`PagerDuty OAuth configuration loaded successfully for account ${maskedAccountId}.`
|
|
153
|
+
);
|
|
154
|
+
}
|
|
155
|
+
} else {
|
|
156
|
+
authPersistence.accountTokens[account.id] = {
|
|
157
|
+
authToken: `Token token=${account.apiToken}`,
|
|
158
|
+
authTokenExpiryDate: Date.now() + 36e5 * 24 * 365 * 2
|
|
159
|
+
// 2 years
|
|
160
|
+
};
|
|
161
|
+
logger.debug(
|
|
162
|
+
`PagerDuty API token loaded successfully for account ${maskedAccountId}.`
|
|
163
|
+
);
|
|
164
|
+
}
|
|
165
|
+
})
|
|
166
|
+
);
|
|
167
|
+
if (!authPersistence.defaultAccount) {
|
|
168
|
+
logger.error(
|
|
169
|
+
"No default account found in config file. One account must be marked as default."
|
|
170
|
+
);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
} catch (error) {
|
|
174
|
+
logger.error(
|
|
175
|
+
`Unable to retrieve valid PagerDuty AUTH configuration from config file: ${error}`
|
|
176
|
+
);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
function readOptionalString(key) {
|
|
180
|
+
if (!_config) {
|
|
181
|
+
return _legacyConfig.getOptionalString(key);
|
|
182
|
+
}
|
|
183
|
+
return _config.getOptionalString(key);
|
|
184
|
+
}
|
|
185
|
+
function readOptionalObject(key) {
|
|
186
|
+
if (!_config) {
|
|
187
|
+
return _legacyConfig.getOptional(key);
|
|
188
|
+
}
|
|
189
|
+
return _config.getOptional(key);
|
|
190
|
+
}
|
|
191
|
+
function readString(key) {
|
|
192
|
+
if (!_config) {
|
|
193
|
+
return _legacyConfig.getString(key);
|
|
194
|
+
}
|
|
195
|
+
return _config.getString(key);
|
|
196
|
+
}
|
|
197
|
+
async function getOAuthToken(clientId, clientSecret, subDomain, region) {
|
|
198
|
+
if (!clientId || !clientSecret || !subDomain) {
|
|
199
|
+
throw new Error("Missing required PagerDuty OAuth parameters.");
|
|
200
|
+
}
|
|
201
|
+
const scopes = `
|
|
202
|
+
abilities.read
|
|
203
|
+
analytics.read
|
|
204
|
+
change_events.read
|
|
205
|
+
escalation_policies.read
|
|
206
|
+
incidents.read
|
|
207
|
+
oncalls.read
|
|
208
|
+
schedules.read
|
|
209
|
+
services.read
|
|
210
|
+
services.write
|
|
211
|
+
standards.read
|
|
212
|
+
teams.read
|
|
213
|
+
users.read
|
|
214
|
+
vendors.read
|
|
215
|
+
`;
|
|
216
|
+
const urlencoded = new URLSearchParams();
|
|
217
|
+
urlencoded.append("grant_type", "client_credentials");
|
|
218
|
+
urlencoded.append("client_id", clientId);
|
|
219
|
+
urlencoded.append("client_secret", clientSecret);
|
|
220
|
+
urlencoded.append("scope", `as_account-${region}.${subDomain} ${scopes}`);
|
|
221
|
+
let response;
|
|
222
|
+
const options = {
|
|
223
|
+
method: "POST",
|
|
224
|
+
headers: {
|
|
225
|
+
"Content-Type": "application/x-www-form-urlencoded"
|
|
226
|
+
},
|
|
227
|
+
body: urlencoded
|
|
228
|
+
};
|
|
229
|
+
const baseUrl = "https://identity.pagerduty.com/oauth/token";
|
|
230
|
+
try {
|
|
231
|
+
response = await fetch(baseUrl, options);
|
|
232
|
+
} catch (error) {
|
|
233
|
+
throw new Error(`Failed to retrieve oauth token: ${error}`);
|
|
234
|
+
}
|
|
235
|
+
switch (response.status) {
|
|
236
|
+
case 400:
|
|
237
|
+
throw new backstagePluginCommon.HttpError(
|
|
238
|
+
"Failed to retrieve valid token. Bad Request - Invalid arguments provided.",
|
|
239
|
+
400
|
|
240
|
+
);
|
|
241
|
+
case 401:
|
|
242
|
+
throw new backstagePluginCommon.HttpError(
|
|
243
|
+
"Failed to retrieve valid token. Forbidden - Invalid credentials provided.",
|
|
244
|
+
401
|
|
245
|
+
);
|
|
246
|
+
}
|
|
247
|
+
const authResponse = await response.json();
|
|
248
|
+
const result = {
|
|
249
|
+
authToken: `Bearer ${authResponse.access_token}`,
|
|
250
|
+
authTokenExpiryDate: Date.now() + authResponse.expires_in * 1e3
|
|
251
|
+
};
|
|
252
|
+
return result;
|
|
253
|
+
}
|
|
254
|
+
function maskString(str) {
|
|
255
|
+
return str[0] + "*".repeat(str.length - 2) + str.slice(-1);
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
exports.getAuthToken = getAuthToken;
|
|
259
|
+
exports.loadAuthConfig = loadAuthConfig;
|
|
260
|
+
//# sourceMappingURL=auth.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auth.cjs.js","sources":["../../src/auth/auth.ts"],"sourcesContent":["import {\n LoggerService,\n RootConfigService,\n} from '@backstage/backend-plugin-api';\nimport {\n HttpError,\n PagerDutyAccountConfig,\n} from '@pagerduty/backstage-plugin-common';\nimport { Config } from '@backstage/config';\n\nexport type LoadAuthConfigProps = {\n config: RootConfigService | undefined;\n legacyConfig: Config;\n logger: LoggerService;\n};\n\ntype JsonValue = boolean | number | string | null | JsonArray | JsonObject;\n\ninterface JsonObject {\n [x: string]: JsonValue;\n}\n\ntype JsonArray = JsonValue[];\n\ntype AccountTokenInfo = {\n authToken: string;\n authTokenExpiryDate: number;\n};\n\ntype Auth = {\n accountTokens: Record<string, AccountTokenInfo>;\n defaultAccount?: string;\n config?: RootConfigService;\n legacyConfig?: Config;\n logger?: LoggerService;\n};\n\nlet authPersistence: Auth;\nlet isLegacyConfig = false;\nlet _config: RootConfigService | undefined;\nlet _legacyConfig: Config;\nlet _logger: LoggerService;\n\nasync function checkForOAuthToken(tokenId: string): Promise<boolean> {\n if (\n authPersistence.accountTokens[tokenId]?.authToken !== '' &&\n authPersistence.accountTokens[tokenId]?.authToken.includes('Bearer')\n ) {\n if (\n authPersistence.accountTokens[tokenId].authTokenExpiryDate > Date.now()\n ) {\n return true;\n }\n _logger.info('OAuth token expired, renewing');\n await loadAuthConfig({\n config: _config,\n legacyConfig: _legacyConfig,\n logger: _logger,\n });\n return (\n authPersistence.accountTokens[tokenId].authTokenExpiryDate > Date.now()\n );\n }\n return false;\n}\n\nexport async function getAuthToken(accountId?: string): Promise<string> {\n // if authPersistence is not initialized, load the auth config\n if (!authPersistence?.accountTokens) {\n _logger.debug('Auth config not loaded. Loading auth config...');\n await loadAuthConfig({\n config: _config,\n legacyConfig: _legacyConfig,\n logger: _logger,\n });\n }\n\n if (isLegacyConfig) {\n _logger.debug('Using legacy config for auth token retrieval.');\n if (\n authPersistence.accountTokens.default.authToken !== '' &&\n ((await checkForOAuthToken('default')) ||\n authPersistence.accountTokens.default.authToken.includes('Token'))\n ) {\n return authPersistence.accountTokens.default.authToken;\n }\n } else {\n _logger.debug('Using new config for auth token retrieval.');\n // check if accountId is provided\n if (accountId && accountId !== '') {\n if (\n authPersistence.accountTokens[accountId]?.authToken !== '' &&\n ((await checkForOAuthToken(accountId)) ||\n authPersistence.accountTokens[accountId]?.authToken.includes('Token'))\n ) {\n return authPersistence.accountTokens[accountId].authToken;\n }\n } else {\n // return default account token if accountId is not provided\n const defaultFallback = authPersistence.defaultAccount ?? '';\n _logger.debug('No account ID provided. Using default account token.');\n\n if (\n authPersistence.accountTokens[defaultFallback]?.authToken !== '' &&\n ((await checkForOAuthToken(defaultFallback)) ||\n authPersistence.accountTokens[defaultFallback]?.authToken.includes(\n 'Token',\n ))\n ) {\n return authPersistence.accountTokens[defaultFallback].authToken;\n }\n }\n }\n\n return '';\n}\n\nexport async function loadAuthConfig({\n config,\n legacyConfig,\n logger,\n}: LoadAuthConfigProps) {\n try {\n const defaultAccountId = 'default';\n\n // set config and logger\n _config = config;\n _legacyConfig = legacyConfig;\n _logger = logger;\n\n // initiliaze the authPersistence in-memory object\n authPersistence = {\n accountTokens: {},\n config: _config,\n legacyConfig: _legacyConfig,\n logger: _logger,\n };\n\n // check if new accounts config is present\n if (!readOptionalObject('pagerDuty.accounts')) {\n isLegacyConfig = true;\n logger.warn(\n 'No PagerDuty accounts configuration found in config file. Reverting to legacy configuration.',\n );\n\n if (!readOptionalString('pagerDuty.apiToken')) {\n logger.warn(\n 'No PagerDuty API token found in config file. Trying OAuth token instead...',\n );\n\n if (!readOptionalObject('pagerDuty.oauth')) {\n logger.error(\n 'No PagerDuty OAuth configuration found in config file.',\n );\n } else if (\n !readOptionalString('pagerDuty.oauth.clientId') ||\n !readOptionalString('pagerDuty.oauth.clientSecret') ||\n !readOptionalString('pagerDuty.oauth.subDomain')\n ) {\n logger.error(\n \"Missing required PagerDuty OAuth parameters in config file. 'clientId', 'clientSecret', and 'subDomain' are required. 'region' is optional.\",\n );\n } else {\n const tokenInfo: AccountTokenInfo = await getOAuthToken(\n readString('pagerDuty.oauth.clientId'),\n readString('pagerDuty.oauth.clientSecret'),\n readString('pagerDuty.oauth.subDomain'),\n readOptionalString('pagerDuty.oauth.region') ?? 'us',\n );\n\n authPersistence.accountTokens[defaultAccountId] = tokenInfo;\n\n logger.debug('PagerDuty OAuth configuration loaded successfully.');\n }\n } else {\n authPersistence.accountTokens[defaultAccountId] = {\n authToken: `Token token=${readOptionalString('pagerDuty.apiToken')}`,\n authTokenExpiryDate: Date.now() + 3600000 * 24 * 365 * 2, // 2 years\n };\n\n logger.debug('PagerDuty API token loaded successfully.');\n }\n } else {\n // new accounts config is present\n logger.debug(\n 'New PagerDuty accounts configuration found in config file.',\n );\n isLegacyConfig = false;\n\n const accounts: PagerDutyAccountConfig[] = JSON.parse(\n JSON.stringify(readOptionalObject('pagerDuty.accounts')),\n );\n\n if (accounts && accounts?.length === 1) {\n logger.debug(\n 'Only one account found in config file. Setting it as default.',\n );\n authPersistence.defaultAccount = accounts[0].id;\n }\n\n await Promise.all(\n accounts?.map(async account => {\n const maskedAccountId = maskString(account.id);\n\n if (account.isDefault && !authPersistence.defaultAccount) {\n logger.debug(\n `Default account found in config file. Setting it as default.`,\n );\n authPersistence.defaultAccount = account.id;\n }\n\n if (!account.apiToken) {\n logger.warn(\n 'No PagerDuty API token found in config file. Trying OAuth token instead...',\n );\n\n if (!account.oauth) {\n logger.error(\n 'No PagerDuty OAuth configuration found in config file.',\n );\n } else if (\n !account.oauth.clientId ||\n !account.oauth.clientSecret ||\n !account.oauth.subDomain\n ) {\n logger.error(\n \"Missing required PagerDuty OAuth parameters in config file. 'clientId', 'clientSecret', and 'subDomain' are required. 'region' is optional.\",\n );\n } else {\n const tokenInfo: AccountTokenInfo = await getOAuthToken(\n account.oauth.clientId,\n account.oauth.clientSecret,\n account.oauth.subDomain,\n account.oauth.region ?? 'us',\n );\n\n authPersistence.accountTokens[account.id] = tokenInfo;\n\n logger.debug(\n `PagerDuty OAuth configuration loaded successfully for account ${maskedAccountId}.`,\n );\n }\n } else {\n authPersistence.accountTokens[account.id] = {\n authToken: `Token token=${account.apiToken}`,\n authTokenExpiryDate: Date.now() + 3600000 * 24 * 365 * 2, // 2 years\n };\n\n logger.debug(\n `PagerDuty API token loaded successfully for account ${maskedAccountId}.`,\n );\n }\n }),\n );\n\n if (!authPersistence.defaultAccount) {\n logger.error(\n 'No default account found in config file. One account must be marked as default.',\n );\n }\n }\n } catch (error) {\n logger.error(\n `Unable to retrieve valid PagerDuty AUTH configuration from config file: ${error}`,\n );\n }\n}\n\nfunction readOptionalString(key: string): string | undefined {\n if (!_config) {\n return _legacyConfig.getOptionalString(key);\n }\n\n return _config.getOptionalString(key);\n}\n\nfunction readOptionalObject(key: string): JsonValue | undefined {\n if (!_config) {\n return _legacyConfig.getOptional(key);\n }\n\n return _config.getOptional(key);\n}\n\nfunction readString(key: string): string {\n if (!_config) {\n return _legacyConfig.getString(key);\n }\n\n return _config.getString(key);\n}\n\nasync function getOAuthToken(\n clientId: string,\n clientSecret: string,\n subDomain: string,\n region: string,\n): Promise<AccountTokenInfo> {\n // check if required parameters are provided\n if (!clientId || !clientSecret || !subDomain) {\n throw new Error('Missing required PagerDuty OAuth parameters.');\n }\n\n // define the scopes required for the OAuth token\n const scopes = `\n abilities.read \n analytics.read\n change_events.read \n escalation_policies.read \n incidents.read \n oncalls.read \n schedules.read \n services.read \n services.write \n standards.read\n teams.read \n users.read \n vendors.read\n `;\n\n // encode the parameters for the request\n const urlencoded = new URLSearchParams();\n urlencoded.append('grant_type', 'client_credentials');\n urlencoded.append('client_id', clientId);\n urlencoded.append('client_secret', clientSecret);\n urlencoded.append('scope', `as_account-${region}.${subDomain} ${scopes}`);\n\n let response: Response;\n const options: RequestInit = {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded',\n },\n body: urlencoded,\n };\n const baseUrl = 'https://identity.pagerduty.com/oauth/token';\n\n try {\n response = await fetch(baseUrl, options);\n } catch (error) {\n throw new Error(`Failed to retrieve oauth token: ${error}`);\n }\n\n switch (response.status) {\n case 400:\n throw new HttpError(\n 'Failed to retrieve valid token. Bad Request - Invalid arguments provided.',\n 400,\n );\n case 401:\n throw new HttpError(\n 'Failed to retrieve valid token. Forbidden - Invalid credentials provided.',\n 401,\n );\n default: // 200\n break;\n }\n\n const authResponse = await response.json();\n\n const result: AccountTokenInfo = {\n authToken: `Bearer ${authResponse.access_token}`,\n authTokenExpiryDate: Date.now() + authResponse.expires_in * 1000,\n };\n\n return result;\n}\n\nfunction maskString(str: string): string {\n return str[0] + '*'.repeat(str.length - 2) + str.slice(-1);\n}\n"],"names":["HttpError"],"mappings":";;;;AAqCA,IAAI,eAAA;AACJ,IAAI,cAAA,GAAiB,KAAA;AACrB,IAAI,OAAA;AACJ,IAAI,aAAA;AACJ,IAAI,OAAA;AAEJ,eAAe,mBAAmB,OAAA,EAAmC;AACnE,EAAA,IACE,eAAA,CAAgB,aAAA,CAAc,OAAO,CAAA,EAAG,SAAA,KAAc,EAAA,IACtD,eAAA,CAAgB,aAAA,CAAc,OAAO,CAAA,EAAG,SAAA,CAAU,QAAA,CAAS,QAAQ,CAAA,EACnE;AACA,IAAA,IACE,gBAAgB,aAAA,CAAc,OAAO,EAAE,mBAAA,GAAsB,IAAA,CAAK,KAAI,EACtE;AACA,MAAA,OAAO,IAAA;AAAA,IACT;AACA,IAAA,OAAA,CAAQ,KAAK,+BAA+B,CAAA;AAC5C,IAAA,MAAM,cAAA,CAAe;AAAA,MACnB,MAAA,EAAQ,OAAA;AAAA,MACR,YAAA,EAAc,aAAA;AAAA,MACd,MAAA,EAAQ;AAAA,KACT,CAAA;AACD,IAAA,OACE,gBAAgB,aAAA,CAAc,OAAO,CAAA,CAAE,mBAAA,GAAsB,KAAK,GAAA,EAAI;AAAA,EAE1E;AACA,EAAA,OAAO,KAAA;AACT;AAEA,eAAsB,aAAa,SAAA,EAAqC;AAEtE,EAAA,IAAI,CAAC,iBAAiB,aAAA,EAAe;AACnC,IAAA,OAAA,CAAQ,MAAM,gDAAgD,CAAA;AAC9D,IAAA,MAAM,cAAA,CAAe;AAAA,MACnB,MAAA,EAAQ,OAAA;AAAA,MACR,YAAA,EAAc,aAAA;AAAA,MACd,MAAA,EAAQ;AAAA,KACT,CAAA;AAAA,EACH;AAEA,EAAA,IAAI,cAAA,EAAgB;AAClB,IAAA,OAAA,CAAQ,MAAM,+CAA+C,CAAA;AAC7D,IAAA,IACE,eAAA,CAAgB,aAAA,CAAc,OAAA,CAAQ,SAAA,KAAc,OAClD,MAAM,kBAAA,CAAmB,SAAS,CAAA,IAClC,gBAAgB,aAAA,CAAc,OAAA,CAAQ,SAAA,CAAU,QAAA,CAAS,OAAO,CAAA,CAAA,EAClE;AACA,MAAA,OAAO,eAAA,CAAgB,cAAc,OAAA,CAAQ,SAAA;AAAA,IAC/C;AAAA,EACF,CAAA,MAAO;AACL,IAAA,OAAA,CAAQ,MAAM,4CAA4C,CAAA;AAE1D,IAAA,IAAI,SAAA,IAAa,cAAc,EAAA,EAAI;AACjC,MAAA,IACE,gBAAgB,aAAA,CAAc,SAAS,CAAA,EAAG,SAAA,KAAc,OACtD,MAAM,kBAAA,CAAmB,SAAS,CAAA,IAClC,gBAAgB,aAAA,CAAc,SAAS,GAAG,SAAA,CAAU,QAAA,CAAS,OAAO,CAAA,CAAA,EACtE;AACA,QAAA,OAAO,eAAA,CAAgB,aAAA,CAAc,SAAS,CAAA,CAAE,SAAA;AAAA,MAClD;AAAA,IACF,CAAA,MAAO;AAEL,MAAA,MAAM,eAAA,GAAkB,gBAAgB,cAAA,IAAkB,EAAA;AAC1D,MAAA,OAAA,CAAQ,MAAM,sDAAsD,CAAA;AAEpE,MAAA,IACE,eAAA,CAAgB,aAAA,CAAc,eAAe,CAAA,EAAG,cAAc,EAAA,KAC5D,MAAM,kBAAA,CAAmB,eAAe,CAAA,IACxC,eAAA,CAAgB,aAAA,CAAc,eAAe,GAAG,SAAA,CAAU,QAAA;AAAA,QACxD;AAAA,OACF,CAAA,EACF;AACA,QAAA,OAAO,eAAA,CAAgB,aAAA,CAAc,eAAe,CAAA,CAAE,SAAA;AAAA,MACxD;AAAA,IACF;AAAA,EACF;AAEA,EAAA,OAAO,EAAA;AACT;AAEA,eAAsB,cAAA,CAAe;AAAA,EACnC,MAAA;AAAA,EACA,YAAA;AAAA,EACA;AACF,CAAA,EAAwB;AACtB,EAAA,IAAI;AACF,IAAA,MAAM,gBAAA,GAAmB,SAAA;AAGzB,IAAA,OAAA,GAAU,MAAA;AACV,IAAA,aAAA,GAAgB,YAAA;AAChB,IAAA,OAAA,GAAU,MAAA;AAGV,IAAA,eAAA,GAAkB;AAAA,MAChB,eAAe,EAAC;AAAA,MAChB,MAAA,EAAQ,OAAA;AAAA,MACR,YAAA,EAAc,aAAA;AAAA,MACd,MAAA,EAAQ;AAAA,KACV;AAGA,IAAA,IAAI,CAAC,kBAAA,CAAmB,oBAAoB,CAAA,EAAG;AAC7C,MAAA,cAAA,GAAiB,IAAA;AACjB,MAAA,MAAA,CAAO,IAAA;AAAA,QACL;AAAA,OACF;AAEA,MAAA,IAAI,CAAC,kBAAA,CAAmB,oBAAoB,CAAA,EAAG;AAC7C,QAAA,MAAA,CAAO,IAAA;AAAA,UACL;AAAA,SACF;AAEA,QAAA,IAAI,CAAC,kBAAA,CAAmB,iBAAiB,CAAA,EAAG;AAC1C,UAAA,MAAA,CAAO,KAAA;AAAA,YACL;AAAA,WACF;AAAA,QACF,CAAA,MAAA,IACE,CAAC,kBAAA,CAAmB,0BAA0B,CAAA,IAC9C,CAAC,kBAAA,CAAmB,8BAA8B,CAAA,IAClD,CAAC,kBAAA,CAAmB,2BAA2B,CAAA,EAC/C;AACA,UAAA,MAAA,CAAO,KAAA;AAAA,YACL;AAAA,WACF;AAAA,QACF,CAAA,MAAO;AACL,UAAA,MAAM,YAA8B,MAAM,aAAA;AAAA,YACxC,WAAW,0BAA0B,CAAA;AAAA,YACrC,WAAW,8BAA8B,CAAA;AAAA,YACzC,WAAW,2BAA2B,CAAA;AAAA,YACtC,kBAAA,CAAmB,wBAAwB,CAAA,IAAK;AAAA,WAClD;AAEA,UAAA,eAAA,CAAgB,aAAA,CAAc,gBAAgB,CAAA,GAAI,SAAA;AAElD,UAAA,MAAA,CAAO,MAAM,oDAAoD,CAAA;AAAA,QACnE;AAAA,MACF,CAAA,MAAO;AACL,QAAA,eAAA,CAAgB,aAAA,CAAc,gBAAgB,CAAA,GAAI;AAAA,UAChD,SAAA,EAAW,CAAA,YAAA,EAAe,kBAAA,CAAmB,oBAAoB,CAAC,CAAA,CAAA;AAAA,UAClE,qBAAqB,IAAA,CAAK,GAAA,EAAI,GAAI,IAAA,GAAU,KAAK,GAAA,GAAM;AAAA;AAAA,SACzD;AAEA,QAAA,MAAA,CAAO,MAAM,0CAA0C,CAAA;AAAA,MACzD;AAAA,IACF,CAAA,MAAO;AAEL,MAAA,MAAA,CAAO,KAAA;AAAA,QACL;AAAA,OACF;AACA,MAAA,cAAA,GAAiB,KAAA;AAEjB,MAAA,MAAM,WAAqC,IAAA,CAAK,KAAA;AAAA,QAC9C,IAAA,CAAK,SAAA,CAAU,kBAAA,CAAmB,oBAAoB,CAAC;AAAA,OACzD;AAEA,MAAA,IAAI,QAAA,IAAY,QAAA,EAAU,MAAA,KAAW,CAAA,EAAG;AACtC,QAAA,MAAA,CAAO,KAAA;AAAA,UACL;AAAA,SACF;AACA,QAAA,eAAA,CAAgB,cAAA,GAAiB,QAAA,CAAS,CAAC,CAAA,CAAE,EAAA;AAAA,MAC/C;AAEA,MAAA,MAAM,OAAA,CAAQ,GAAA;AAAA,QACZ,QAAA,EAAU,GAAA,CAAI,OAAM,OAAA,KAAW;AAC7B,UAAA,MAAM,eAAA,GAAkB,UAAA,CAAW,OAAA,CAAQ,EAAE,CAAA;AAE7C,UAAA,IAAI,OAAA,CAAQ,SAAA,IAAa,CAAC,eAAA,CAAgB,cAAA,EAAgB;AACxD,YAAA,MAAA,CAAO,KAAA;AAAA,cACL,CAAA,4DAAA;AAAA,aACF;AACA,YAAA,eAAA,CAAgB,iBAAiB,OAAA,CAAQ,EAAA;AAAA,UAC3C;AAEA,UAAA,IAAI,CAAC,QAAQ,QAAA,EAAU;AACrB,YAAA,MAAA,CAAO,IAAA;AAAA,cACL;AAAA,aACF;AAEA,YAAA,IAAI,CAAC,QAAQ,KAAA,EAAO;AAClB,cAAA,MAAA,CAAO,KAAA;AAAA,gBACL;AAAA,eACF;AAAA,YACF,CAAA,MAAA,IACE,CAAC,OAAA,CAAQ,KAAA,CAAM,QAAA,IACf,CAAC,OAAA,CAAQ,KAAA,CAAM,YAAA,IACf,CAAC,OAAA,CAAQ,KAAA,CAAM,SAAA,EACf;AACA,cAAA,MAAA,CAAO,KAAA;AAAA,gBACL;AAAA,eACF;AAAA,YACF,CAAA,MAAO;AACL,cAAA,MAAM,YAA8B,MAAM,aAAA;AAAA,gBACxC,QAAQ,KAAA,CAAM,QAAA;AAAA,gBACd,QAAQ,KAAA,CAAM,YAAA;AAAA,gBACd,QAAQ,KAAA,CAAM,SAAA;AAAA,gBACd,OAAA,CAAQ,MAAM,MAAA,IAAU;AAAA,eAC1B;AAEA,cAAA,eAAA,CAAgB,aAAA,CAAc,OAAA,CAAQ,EAAE,CAAA,GAAI,SAAA;AAE5C,cAAA,MAAA,CAAO,KAAA;AAAA,gBACL,iEAAiE,eAAe,CAAA,CAAA;AAAA,eAClF;AAAA,YACF;AAAA,UACF,CAAA,MAAO;AACL,YAAA,eAAA,CAAgB,aAAA,CAAc,OAAA,CAAQ,EAAE,CAAA,GAAI;AAAA,cAC1C,SAAA,EAAW,CAAA,YAAA,EAAe,OAAA,CAAQ,QAAQ,CAAA,CAAA;AAAA,cAC1C,qBAAqB,IAAA,CAAK,GAAA,EAAI,GAAI,IAAA,GAAU,KAAK,GAAA,GAAM;AAAA;AAAA,aACzD;AAEA,YAAA,MAAA,CAAO,KAAA;AAAA,cACL,uDAAuD,eAAe,CAAA,CAAA;AAAA,aACxE;AAAA,UACF;AAAA,QACF,CAAC;AAAA,OACH;AAEA,MAAA,IAAI,CAAC,gBAAgB,cAAA,EAAgB;AACnC,QAAA,MAAA,CAAO,KAAA;AAAA,UACL;AAAA,SACF;AAAA,MACF;AAAA,IACF;AAAA,EACF,SAAS,KAAA,EAAO;AACd,IAAA,MAAA,CAAO,KAAA;AAAA,MACL,2EAA2E,KAAK,CAAA;AAAA,KAClF;AAAA,EACF;AACF;AAEA,SAAS,mBAAmB,GAAA,EAAiC;AAC3D,EAAA,IAAI,CAAC,OAAA,EAAS;AACZ,IAAA,OAAO,aAAA,CAAc,kBAAkB,GAAG,CAAA;AAAA,EAC5C;AAEA,EAAA,OAAO,OAAA,CAAQ,kBAAkB,GAAG,CAAA;AACtC;AAEA,SAAS,mBAAmB,GAAA,EAAoC;AAC9D,EAAA,IAAI,CAAC,OAAA,EAAS;AACZ,IAAA,OAAO,aAAA,CAAc,YAAY,GAAG,CAAA;AAAA,EACtC;AAEA,EAAA,OAAO,OAAA,CAAQ,YAAY,GAAG,CAAA;AAChC;AAEA,SAAS,WAAW,GAAA,EAAqB;AACvC,EAAA,IAAI,CAAC,OAAA,EAAS;AACZ,IAAA,OAAO,aAAA,CAAc,UAAU,GAAG,CAAA;AAAA,EACpC;AAEA,EAAA,OAAO,OAAA,CAAQ,UAAU,GAAG,CAAA;AAC9B;AAEA,eAAe,aAAA,CACb,QAAA,EACA,YAAA,EACA,SAAA,EACA,MAAA,EAC2B;AAE3B,EAAA,IAAI,CAAC,QAAA,IAAY,CAAC,YAAA,IAAgB,CAAC,SAAA,EAAW;AAC5C,IAAA,MAAM,IAAI,MAAM,8CAA8C,CAAA;AAAA,EAChE;AAGA,EAAA,MAAM,MAAA,GAAS;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAAA,CAAA;AAiBf,EAAA,MAAM,UAAA,GAAa,IAAI,eAAA,EAAgB;AACvC,EAAA,UAAA,CAAW,MAAA,CAAO,cAAc,oBAAoB,CAAA;AACpD,EAAA,UAAA,CAAW,MAAA,CAAO,aAAa,QAAQ,CAAA;AACvC,EAAA,UAAA,CAAW,MAAA,CAAO,iBAAiB,YAAY,CAAA;AAC/C,EAAA,UAAA,CAAW,MAAA,CAAO,SAAS,CAAA,WAAA,EAAc,MAAM,IAAI,SAAS,CAAA,CAAA,EAAI,MAAM,CAAA,CAAE,CAAA;AAExE,EAAA,IAAI,QAAA;AACJ,EAAA,MAAM,OAAA,GAAuB;AAAA,IAC3B,MAAA,EAAQ,MAAA;AAAA,IACR,OAAA,EAAS;AAAA,MACP,cAAA,EAAgB;AAAA,KAClB;AAAA,IACA,IAAA,EAAM;AAAA,GACR;AACA,EAAA,MAAM,OAAA,GAAU,4CAAA;AAEhB,EAAA,IAAI;AACF,IAAA,QAAA,GAAW,MAAM,KAAA,CAAM,OAAA,EAAS,OAAO,CAAA;AAAA,EACzC,SAAS,KAAA,EAAO;AACd,IAAA,MAAM,IAAI,KAAA,CAAM,CAAA,gCAAA,EAAmC,KAAK,CAAA,CAAE,CAAA;AAAA,EAC5D;AAEA,EAAA,QAAQ,SAAS,MAAA;AAAQ,IACvB,KAAK,GAAA;AACH,MAAA,MAAM,IAAIA,+BAAA;AAAA,QACR,2EAAA;AAAA,QACA;AAAA,OACF;AAAA,IACF,KAAK,GAAA;AACH,MAAA,MAAM,IAAIA,+BAAA;AAAA,QACR,2EAAA;AAAA,QACA;AAAA,OACF;AAEA;AAGJ,EAAA,MAAM,YAAA,GAAe,MAAM,QAAA,CAAS,IAAA,EAAK;AAEzC,EAAA,MAAM,MAAA,GAA2B;AAAA,IAC/B,SAAA,EAAW,CAAA,OAAA,EAAU,YAAA,CAAa,YAAY,CAAA,CAAA;AAAA,IAC9C,mBAAA,EAAqB,IAAA,CAAK,GAAA,EAAI,GAAI,aAAa,UAAA,GAAa;AAAA,GAC9D;AAEA,EAAA,OAAO,MAAA;AACT;AAEA,SAAS,WAAW,GAAA,EAAqB;AACvC,EAAA,OAAO,GAAA,CAAI,CAAC,CAAA,GAAI,GAAA,CAAI,MAAA,CAAO,GAAA,CAAI,MAAA,GAAS,CAAC,CAAA,GAAI,GAAA,CAAI,KAAA,CAAM,EAAE,CAAA;AAC3D;;;;;"}
|