',
10 | timeout: 30000,
11 | os: ServiceOS.LINUX,
12 | serviceAuthType:'ACCESS_TOKEN',
13 | useCloudHostedBrowsers: true // Set to false if you want to only use reporting and not cloud hosted browsers
14 | }),
15 | {
16 | /*
17 | Playwright Testing service reporter is added by default.
18 | This will override any reporter options specified in the base playwright config.
19 | If you are using more reporters, please update your configuration accordingly.
20 | */
21 | reporter: [['list'], ['@azure/microsoft-playwright-testing/reporter'],['junit', { outputFile: 'test-results/results.xml' }]],
22 | }
23 | );
24 |
--------------------------------------------------------------------------------
/template.env:
--------------------------------------------------------------------------------
1 | ENV=dev
2 | CI=false
3 | CLIENT_ID=""
4 | CLIENT_SECRET=""
5 | TENANT_ID=""
6 | ENVIRONMENT="Public"
7 | EFFECTIVE_USERNAME = ""
--------------------------------------------------------------------------------
/test-cases/placeholder.txt:
--------------------------------------------------------------------------------
1 | This is used to make sure the test-cases folder exists on Git Import.
--------------------------------------------------------------------------------
/test-generation/Execute-XMLAQuery.ps1:
--------------------------------------------------------------------------------
1 | param (
2 | [string]$ClientId,
3 | [string]$ClientSecret,
4 | [string]$TenantId,
5 | [string]$XmlaQuery,
6 | [string]$DataSource,
7 | [string]$DatasetName,
8 | [string]$AccessToken
9 | )
10 |
11 | #Install Powershell Module if Needed
12 | if (Get-Module -ListAvailable -Name "SqlServer") {
13 | # Do Nothing
14 | } else {
15 | Install-Module -Name SqlServer -Scope CurrentUser -AllowClobber -Force
16 | }
17 |
18 | # Setup the connection string
19 | $secret = $ClientSecret | ConvertTo-SecureString -AsPlainText -Force
20 | $credentials = [System.Management.Automation.PSCredential]::new($ClientId,$secret)
21 |
22 | try {
23 | $Result = Invoke-ASCmd -Server $DataSource `
24 | -Database $DatasetName `
25 | -Query $XmlaQuery `
26 | -Credential $credentials `
27 | -TenantId $TenantId -ServicePrincipal
28 |
29 | #Remove unicode chars for brackets and spaces from XML node names
30 | $Result = $Result -replace '_x[0-9A-z]{4}_', '';
31 |
32 | $Result
33 |
34 | } catch {
35 | Write-Host "Error: $_"
36 | }
37 |
--------------------------------------------------------------------------------
/test-generation/app.js:
--------------------------------------------------------------------------------
1 | const express = require('express');
2 | const path = require('path');
3 | const { exec } = require('child_process');
4 | const xml2js = require('xml2js');
5 | const { ConfidentialClientApplication } = require('@azure/msal-node');
6 | const { get } = require('http');
7 | const app = express();
8 | const port = 3000;
9 | // Load environment variables from .env file
10 | require('dotenv').config({ path: path.resolve(__dirname, '../.env') });
11 | // Use environment variables from .env file
12 | const tenantId = process.env.TENANT_ID;
13 | const clientId = process.env.CLIENT_ID;
14 | const clientSecret = process.env.CLIENT_SECRET;
15 | const environment = process.env.ENVIRONMENT;
16 | const testSettings = getAPIEndpoints(environment);
17 | // Handle file uploads
18 | const multer = require('multer');
19 | const fs = require('fs');
20 |
21 | const msalConfig = {
22 | auth: {
23 | clientId: clientId,
24 | authority: `${testSettings.loginUrl}/${tenantId}`,
25 | clientSecret: clientSecret
26 | }
27 | };
28 | const cca = new ConfidentialClientApplication(msalConfig);
29 | // Define the scopes for the access token
30 | const tokenRequest = { scopes: [`${testSettings.resourceUrl}/.default`] };
31 |
32 | // Function to get API endpoints based on the environment
33 | function getAPIEndpoints(environment) {
34 | // Default endpoints
35 | let endpoints = {
36 | apiPrefix: 'https://api.powerbi.com',
37 | xmlaPrefix: 'powerbi://api.powerbi.com',
38 | webPrefix: 'https://app.powerbi.com',
39 | resourceUrl: 'https://analysis.windows.net/powerbi/api',
40 | embedUrl: 'https://app.powerbi.com/reportEmbed',
41 | loginUrl: 'https://login.microsoftonline.com'
42 | };
43 |
44 | // Switch case to set endpoints based on the environment
45 | switch (environment) {
46 | case "Public":
47 | break;
48 | case "Germany":
49 | endpoints.apiPrefix = "https://api.powerbi.de";
50 | endpoints.xmlaPrefix = "powerbi://api.powerbi.de";
51 | endpoints.webPrefix = "https://app.powerbi.de";
52 | endpoints.resourceUrl = "https://analysis.cloudapi.de/powerbi/api";
53 | endpoints.embedUrl = "https://app.powerbi.de/reportEmbed";
54 | endpoints.loginUrl = "https://login.microsoftonline.com";
55 | break;
56 | case "China":
57 | endpoints.apiPrefix = "https://api.powerbi.cn";
58 | endpoints.xmlaPrefix = "powerbi://api.powerbi.cn";
59 | endpoints.webPrefix = "https://app.powerbigov.cn";
60 | endpoints.resourceUrl = "https://analysis.chinacloudapi.cn/powerbi/api";
61 | endpoints.embedUrl = "https://app.powerbi.cn/reportEmbed";
62 | endpoints.loginUrl = "https://login.partner.microsoftonline.cn";
63 | break;
64 | case "USGov":
65 | endpoints.apiPrefix = "https://api.powerbigov.us";
66 | endpoints.xmlaPrefix = "powerbi://api.powerbigov.us";
67 | endpoints.webPrefix = "https://app.powerbigov.us";
68 | endpoints.resourceUrl = "https://analysis.usgovcloudapi.net/powerbi/api";
69 | endpoints.embedUrl = "https://app.powerbigov.us/reportEmbed";
70 | endpoints.loginUrl = "https://login.microsoftonline.com";
71 | break;
72 | case "USGovHigh":
73 | endpoints.apiPrefix = "https://api.high.powerbigov.us";
74 | endpoints.xmlaPrefix = "powerbi://api.high.powerbigov.us";
75 | endpoints.webPrefix = "https://app.high.powerbigov.us";
76 | endpoints.resourceUrl = "https://analysis.high.usgovcloudapi.net/powerbi/api";
77 | endpoints.embedUrl = "https://app.high.powerbigov.us/reportEmbed";
78 | endpoints.loginUrl = "https://login.microsoftonline.us";
79 | break;
80 | case "USGovDoD":
81 | endpoints.apiPrefix = "https://api.mil.powerbi.us";
82 | endpoints.xmlaPrefix = "powerbi://api.mil.powerbi.us";
83 | endpoints.webPrefix = "https://app.mil.powerbi.us";
84 | endpoints.resourceUrl = "https://analysis.dod.usgovcloudapi.net/powerbi/api";
85 | endpoints.embedUrl = "https://app.mil.powerbi.us/reportEmbed";
86 | endpoints.loginUrl = "https://login.microsoftonline.us";
87 | break;
88 | default:
89 | break;
90 | }
91 |
92 | // Return the endpoints
93 | return endpoints;
94 | }
95 |
96 | // Create the 'test-cases' directory if it doesn't exist
97 | const testCasesDir = path.join(__dirname, '../test-cases');
98 | if (!fs.existsSync(testCasesDir)) {
99 | fs.mkdirSync(testCasesDir, { recursive: true }); // Ensure the directory is created
100 | }
101 |
102 | // Configure multer for file uploads
103 | const storage = multer.diskStorage({
104 | destination: function (req, file, cb) {
105 | cb(null, testCasesDir); // Save to 'test-cases' folder
106 | },
107 | filename: function (req, file, cb) {
108 | cb(null, file.originalname); // Use the original file name
109 | }
110 | });
111 |
112 | const upload = multer({ storage: storage });
113 |
114 |
115 | // ENDPOINTS
116 |
117 | // Get Access Token
118 | async function getAccessToken() {
119 | const apiPrefix = getAPIEndpoints(process.env.ENVIRONMENT).apiPrefix;
120 | const tokenResponse = await cca.acquireTokenByClientCredential(tokenRequest);
121 | return { accessToken: tokenResponse.accessToken, apiUrl: apiPrefix, effectiveUserName: process.env.EFFECTIVE_USERNAME }
122 | }
123 |
124 | // Route to get access token
125 | app.get('/getToken', async (req, res) => {
126 | try {
127 | const response = await getAccessToken();
128 | res.json(response);
129 | } catch (error) {
130 | console.error('Error acquiring token:', error);
131 | res.status(500).json({ error: 'Failed to get access token' });
132 | }
133 | });
134 |
135 | // Route to execute PowerShell script
136 | app.get('/executeScript', async (req, res) => {
137 | const workspaceName = req.query.workspaceName;
138 | const datasetName = req.query.datasetName;
139 |
140 | if (!workspaceName || !datasetName) {
141 | return res.status(400).json({ error: "Workspace Name and Dataset Name are required" });
142 | }
143 |
144 | try {
145 | // Get access token
146 | const { accessToken } = await getAccessToken();
147 |
148 | // Path to your PowerShell script
149 | const scriptPath = path.join(__dirname, 'Execute-XMLAQuery.ps1');
150 |
151 | // Define the dataset name and XMLA query based on the report
152 | const xmlaQuery = "EVALUATE INFO.ROLES()";
153 | const dataSource = `${testSettings.xmlaPrefix}/v1.0/myorg/${workspaceName}`;
154 |
155 | // Prepare the PowerShell command
156 | const command = `pwsh.exe -ExecutionPolicy Bypass -File "${scriptPath}" -ClientId "${clientId}" -ClientSecret "${clientSecret}" -TenantId "${tenantId}" -XmlaQuery "${xmlaQuery}" -DataSource "${dataSource}" -DatasetName "${datasetName}" -AccessToken "${accessToken}"`;
157 |
158 | // Execute the PowerShell script
159 | exec(command, (error, stdout, stderr) => {
160 | if (error) {
161 | console.error(`Error executing PowerShell script: ${error.message}`);
162 | res.status(500).json({ error: 'Failed to execute script' });
163 | return;
164 | }
165 | if (stderr) {
166 | console.error(`PowerShell script error: ${stderr}`);
167 | res.status(500).json({ error: 'PowerShell error' });
168 | return;
169 | }
170 |
171 | console.log("Query Results:", stdout);
172 |
173 | // Parse the XMLA response
174 | const parser = new xml2js.Parser();
175 | parser.parseString(stdout, (err, result) => {
176 | if (err) {
177 | return res.status(500).send("Error parsing XML");
178 | }
179 |
180 | // Navigate to the rows in the parsed XML, check if rows exist
181 | const rows = result?.['return']?.['root']?.[0]?.['row'];
182 |
183 | if (!rows || rows.length === 0) {
184 | return res.json({ message: "No rows found" });
185 | }
186 |
187 | // Extract the values for the C2 nodes, checking if the node exists
188 | const roles = rows.map(row => {
189 | // Check if C2 exists and is an array with at least one value
190 | return row['C2'] && row['C2'].length > 0 ? row['C2'][0] : 'C2 node not found';
191 | });
192 |
193 | res.json({ roles });
194 | })
195 | });
196 |
197 | } catch (error) {
198 | console.error('Error executing script:', error);
199 | res.status(500).json({ error: 'Failed to execute script' });
200 | }
201 | });
202 |
203 | // Serve index.html
204 | app.get('/', (req, res) => {
205 | res.sendFile(path.join(__dirname, 'views', 'index.html'));
206 | });
207 |
208 | // Route to upload a CSV test file
209 | app.post('/uploadCsv', upload.single('testFile'), (req, res) => {
210 | // Check if file was uploaded
211 | if (!req.file) {
212 | return res.status(400).json({ error: 'No file uploaded' });
213 | }
214 |
215 | res.json({ message: 'File uploaded successfully', fileName: req.file.originalname });
216 | });
217 |
218 | // Route to upload a JSON test file
219 | app.post('/uploadJson', upload.single('testFile'), (req, res) => {
220 | // Check if file was uploaded
221 | if (!req.file) {
222 | return res.status(400).json({ error: 'No file uploaded' });
223 | }
224 |
225 | res.json({ message: 'File uploaded successfully', fileName: req.file.originalname });
226 | });
227 |
228 | app.listen(port, () => {
229 | console.log(`App running at http://localhost:${port}`);
230 | });
231 |
--------------------------------------------------------------------------------
/test-generation/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "test-generation",
3 | "version": "1.0.0",
4 | "lockfileVersion": 3,
5 | "requires": true,
6 | "packages": {
7 | "": {
8 | "name": "test-generation",
9 | "version": "1.0.0",
10 | "license": "ISC",
11 | "dependencies": {
12 | "@azure/msal-node": "^2.15.0",
13 | "axios": "^1.7.7",
14 | "child_process": "^1.0.2",
15 | "dotenv": "^16.4.5",
16 | "edge": "^7.10.1",
17 | "express": "^4.21.1",
18 | "fs": "^0.0.1-security",
19 | "multer": "^1.4.5-lts.1",
20 | "xml2js": "^0.6.2"
21 | }
22 | },
23 | "node_modules/@azure/msal-common": {
24 | "version": "14.15.0",
25 | "resolved": "https://registry.npmjs.org/@azure/msal-common/-/msal-common-14.15.0.tgz",
26 | "integrity": "sha512-ImAQHxmpMneJ/4S8BRFhjt1MZ3bppmpRPYYNyzeQPeFN288YKbb8TmmISQEbtfkQ1BPASvYZU5doIZOPBAqENQ==",
27 | "engines": {
28 | "node": ">=0.8.0"
29 | }
30 | },
31 | "node_modules/@azure/msal-node": {
32 | "version": "2.15.0",
33 | "resolved": "https://registry.npmjs.org/@azure/msal-node/-/msal-node-2.15.0.tgz",
34 | "integrity": "sha512-gVPW8YLz92ZeCibQH2QUw96odJoiM3k/ZPH3f2HxptozmH6+OnyyvKXo/Egg39HAM230akarQKHf0W74UHlh0Q==",
35 | "dependencies": {
36 | "@azure/msal-common": "14.15.0",
37 | "jsonwebtoken": "^9.0.0",
38 | "uuid": "^8.3.0"
39 | },
40 | "engines": {
41 | "node": ">=16"
42 | }
43 | },
44 | "node_modules/accepts": {
45 | "version": "1.3.8",
46 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
47 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
48 | "dependencies": {
49 | "mime-types": "~2.1.34",
50 | "negotiator": "0.6.3"
51 | },
52 | "engines": {
53 | "node": ">= 0.6"
54 | }
55 | },
56 | "node_modules/append-field": {
57 | "version": "1.0.0",
58 | "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz",
59 | "integrity": "sha512-klpgFSWLW1ZEs8svjfb7g4qWY0YS5imI82dTg+QahUvJ8YqAY0P10Uk8tTyh9ZGuYEZEMaeJYCF5BFuX552hsw=="
60 | },
61 | "node_modules/array-flatten": {
62 | "version": "1.1.1",
63 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
64 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg=="
65 | },
66 | "node_modules/asynckit": {
67 | "version": "0.4.0",
68 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
69 | "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q=="
70 | },
71 | "node_modules/axios": {
72 | "version": "1.8.4",
73 | "resolved": "https://registry.npmjs.org/axios/-/axios-1.8.4.tgz",
74 | "integrity": "sha512-eBSYY4Y68NNlHbHBMdeDmKNtDgXWhQsJcGqzO3iLUM0GraQFSS9cVgPX5I9b3lbdFKyYoAEGAZF1DwhTaljNAw==",
75 | "license": "MIT",
76 | "dependencies": {
77 | "follow-redirects": "^1.15.6",
78 | "form-data": "^4.0.0",
79 | "proxy-from-env": "^1.1.0"
80 | }
81 | },
82 | "node_modules/body-parser": {
83 | "version": "1.20.3",
84 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz",
85 | "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==",
86 | "dependencies": {
87 | "bytes": "3.1.2",
88 | "content-type": "~1.0.5",
89 | "debug": "2.6.9",
90 | "depd": "2.0.0",
91 | "destroy": "1.2.0",
92 | "http-errors": "2.0.0",
93 | "iconv-lite": "0.4.24",
94 | "on-finished": "2.4.1",
95 | "qs": "6.13.0",
96 | "raw-body": "2.5.2",
97 | "type-is": "~1.6.18",
98 | "unpipe": "1.0.0"
99 | },
100 | "engines": {
101 | "node": ">= 0.8",
102 | "npm": "1.2.8000 || >= 1.4.16"
103 | }
104 | },
105 | "node_modules/buffer-equal-constant-time": {
106 | "version": "1.0.1",
107 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
108 | "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA=="
109 | },
110 | "node_modules/buffer-from": {
111 | "version": "1.1.2",
112 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
113 | "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ=="
114 | },
115 | "node_modules/busboy": {
116 | "version": "1.6.0",
117 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz",
118 | "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==",
119 | "dependencies": {
120 | "streamsearch": "^1.1.0"
121 | },
122 | "engines": {
123 | "node": ">=10.16.0"
124 | }
125 | },
126 | "node_modules/bytes": {
127 | "version": "3.1.2",
128 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
129 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
130 | "engines": {
131 | "node": ">= 0.8"
132 | }
133 | },
134 | "node_modules/call-bind": {
135 | "version": "1.0.7",
136 | "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz",
137 | "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==",
138 | "dependencies": {
139 | "es-define-property": "^1.0.0",
140 | "es-errors": "^1.3.0",
141 | "function-bind": "^1.1.2",
142 | "get-intrinsic": "^1.2.4",
143 | "set-function-length": "^1.2.1"
144 | },
145 | "engines": {
146 | "node": ">= 0.4"
147 | },
148 | "funding": {
149 | "url": "https://github.com/sponsors/ljharb"
150 | }
151 | },
152 | "node_modules/call-bind-apply-helpers": {
153 | "version": "1.0.2",
154 | "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
155 | "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
156 | "license": "MIT",
157 | "dependencies": {
158 | "es-errors": "^1.3.0",
159 | "function-bind": "^1.1.2"
160 | },
161 | "engines": {
162 | "node": ">= 0.4"
163 | }
164 | },
165 | "node_modules/child_process": {
166 | "version": "1.0.2",
167 | "resolved": "https://registry.npmjs.org/child_process/-/child_process-1.0.2.tgz",
168 | "integrity": "sha512-Wmza/JzL0SiWz7kl6MhIKT5ceIlnFPJX+lwUGj7Clhy5MMldsSoJR0+uvRzOS5Kv45Mq7t1PoE8TsOA9bzvb6g=="
169 | },
170 | "node_modules/combined-stream": {
171 | "version": "1.0.8",
172 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
173 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
174 | "dependencies": {
175 | "delayed-stream": "~1.0.0"
176 | },
177 | "engines": {
178 | "node": ">= 0.8"
179 | }
180 | },
181 | "node_modules/concat-stream": {
182 | "version": "1.6.2",
183 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz",
184 | "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==",
185 | "engines": [
186 | "node >= 0.8"
187 | ],
188 | "dependencies": {
189 | "buffer-from": "^1.0.0",
190 | "inherits": "^2.0.3",
191 | "readable-stream": "^2.2.2",
192 | "typedarray": "^0.0.6"
193 | }
194 | },
195 | "node_modules/content-disposition": {
196 | "version": "0.5.4",
197 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz",
198 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==",
199 | "dependencies": {
200 | "safe-buffer": "5.2.1"
201 | },
202 | "engines": {
203 | "node": ">= 0.6"
204 | }
205 | },
206 | "node_modules/content-type": {
207 | "version": "1.0.5",
208 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
209 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
210 | "engines": {
211 | "node": ">= 0.6"
212 | }
213 | },
214 | "node_modules/cookie": {
215 | "version": "0.7.1",
216 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz",
217 | "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==",
218 | "engines": {
219 | "node": ">= 0.6"
220 | }
221 | },
222 | "node_modules/cookie-signature": {
223 | "version": "1.0.6",
224 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
225 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ=="
226 | },
227 | "node_modules/core-util-is": {
228 | "version": "1.0.3",
229 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz",
230 | "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ=="
231 | },
232 | "node_modules/debug": {
233 | "version": "2.6.9",
234 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
235 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
236 | "dependencies": {
237 | "ms": "2.0.0"
238 | }
239 | },
240 | "node_modules/define-data-property": {
241 | "version": "1.1.4",
242 | "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz",
243 | "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==",
244 | "dependencies": {
245 | "es-define-property": "^1.0.0",
246 | "es-errors": "^1.3.0",
247 | "gopd": "^1.0.1"
248 | },
249 | "engines": {
250 | "node": ">= 0.4"
251 | },
252 | "funding": {
253 | "url": "https://github.com/sponsors/ljharb"
254 | }
255 | },
256 | "node_modules/delayed-stream": {
257 | "version": "1.0.0",
258 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
259 | "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
260 | "engines": {
261 | "node": ">=0.4.0"
262 | }
263 | },
264 | "node_modules/depd": {
265 | "version": "2.0.0",
266 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
267 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
268 | "engines": {
269 | "node": ">= 0.8"
270 | }
271 | },
272 | "node_modules/destroy": {
273 | "version": "1.2.0",
274 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz",
275 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==",
276 | "engines": {
277 | "node": ">= 0.8",
278 | "npm": "1.2.8000 || >= 1.4.16"
279 | }
280 | },
281 | "node_modules/dotenv": {
282 | "version": "16.4.5",
283 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.5.tgz",
284 | "integrity": "sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==",
285 | "engines": {
286 | "node": ">=12"
287 | },
288 | "funding": {
289 | "url": "https://dotenvx.com"
290 | }
291 | },
292 | "node_modules/dunder-proto": {
293 | "version": "1.0.1",
294 | "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
295 | "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
296 | "license": "MIT",
297 | "dependencies": {
298 | "call-bind-apply-helpers": "^1.0.1",
299 | "es-errors": "^1.3.0",
300 | "gopd": "^1.2.0"
301 | },
302 | "engines": {
303 | "node": ">= 0.4"
304 | }
305 | },
306 | "node_modules/ecdsa-sig-formatter": {
307 | "version": "1.0.11",
308 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
309 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
310 | "dependencies": {
311 | "safe-buffer": "^5.0.1"
312 | }
313 | },
314 | "node_modules/edge": {
315 | "version": "7.10.1",
316 | "resolved": "https://registry.npmjs.org/edge/-/edge-7.10.1.tgz",
317 | "integrity": "sha512-3rTG+z6houHRamNK7aTqm6rQnFi19H5yYGh0lXNBS5RlP6vQIGlBBwELEaUBR+h9WxSb225AUis4afiDg2w3yw==",
318 | "hasInstallScript": true,
319 | "dependencies": {
320 | "edge-cs": "1.2.1",
321 | "nan": "^2.0.9"
322 | },
323 | "engines": {
324 | "node": ">= 0.8"
325 | }
326 | },
327 | "node_modules/edge-cs": {
328 | "version": "1.2.1",
329 | "resolved": "https://registry.npmjs.org/edge-cs/-/edge-cs-1.2.1.tgz",
330 | "integrity": "sha512-N5KAeMBhhCbCGmb5oiHJ0KcuTksZzo8lg+uByEUCZAdnLPHizHkk6ZIuuTD63eez7+W25ZSvsGS5+FtymxFoKw==",
331 | "hasInstallScript": true,
332 | "engines": {
333 | "node": ">= 0.8"
334 | }
335 | },
336 | "node_modules/ee-first": {
337 | "version": "1.1.1",
338 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
339 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="
340 | },
341 | "node_modules/encodeurl": {
342 | "version": "2.0.0",
343 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz",
344 | "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==",
345 | "engines": {
346 | "node": ">= 0.8"
347 | }
348 | },
349 | "node_modules/es-define-property": {
350 | "version": "1.0.1",
351 | "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
352 | "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
353 | "license": "MIT",
354 | "engines": {
355 | "node": ">= 0.4"
356 | }
357 | },
358 | "node_modules/es-errors": {
359 | "version": "1.3.0",
360 | "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
361 | "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
362 | "engines": {
363 | "node": ">= 0.4"
364 | }
365 | },
366 | "node_modules/es-object-atoms": {
367 | "version": "1.1.1",
368 | "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
369 | "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
370 | "license": "MIT",
371 | "dependencies": {
372 | "es-errors": "^1.3.0"
373 | },
374 | "engines": {
375 | "node": ">= 0.4"
376 | }
377 | },
378 | "node_modules/es-set-tostringtag": {
379 | "version": "2.1.0",
380 | "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz",
381 | "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==",
382 | "license": "MIT",
383 | "dependencies": {
384 | "es-errors": "^1.3.0",
385 | "get-intrinsic": "^1.2.6",
386 | "has-tostringtag": "^1.0.2",
387 | "hasown": "^2.0.2"
388 | },
389 | "engines": {
390 | "node": ">= 0.4"
391 | }
392 | },
393 | "node_modules/escape-html": {
394 | "version": "1.0.3",
395 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
396 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="
397 | },
398 | "node_modules/etag": {
399 | "version": "1.8.1",
400 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
401 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
402 | "engines": {
403 | "node": ">= 0.6"
404 | }
405 | },
406 | "node_modules/express": {
407 | "version": "4.21.2",
408 | "resolved": "https://registry.npmjs.org/express/-/express-4.21.2.tgz",
409 | "integrity": "sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA==",
410 | "license": "MIT",
411 | "dependencies": {
412 | "accepts": "~1.3.8",
413 | "array-flatten": "1.1.1",
414 | "body-parser": "1.20.3",
415 | "content-disposition": "0.5.4",
416 | "content-type": "~1.0.4",
417 | "cookie": "0.7.1",
418 | "cookie-signature": "1.0.6",
419 | "debug": "2.6.9",
420 | "depd": "2.0.0",
421 | "encodeurl": "~2.0.0",
422 | "escape-html": "~1.0.3",
423 | "etag": "~1.8.1",
424 | "finalhandler": "1.3.1",
425 | "fresh": "0.5.2",
426 | "http-errors": "2.0.0",
427 | "merge-descriptors": "1.0.3",
428 | "methods": "~1.1.2",
429 | "on-finished": "2.4.1",
430 | "parseurl": "~1.3.3",
431 | "path-to-regexp": "0.1.12",
432 | "proxy-addr": "~2.0.7",
433 | "qs": "6.13.0",
434 | "range-parser": "~1.2.1",
435 | "safe-buffer": "5.2.1",
436 | "send": "0.19.0",
437 | "serve-static": "1.16.2",
438 | "setprototypeof": "1.2.0",
439 | "statuses": "2.0.1",
440 | "type-is": "~1.6.18",
441 | "utils-merge": "1.0.1",
442 | "vary": "~1.1.2"
443 | },
444 | "engines": {
445 | "node": ">= 0.10.0"
446 | },
447 | "funding": {
448 | "type": "opencollective",
449 | "url": "https://opencollective.com/express"
450 | }
451 | },
452 | "node_modules/finalhandler": {
453 | "version": "1.3.1",
454 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz",
455 | "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==",
456 | "dependencies": {
457 | "debug": "2.6.9",
458 | "encodeurl": "~2.0.0",
459 | "escape-html": "~1.0.3",
460 | "on-finished": "2.4.1",
461 | "parseurl": "~1.3.3",
462 | "statuses": "2.0.1",
463 | "unpipe": "~1.0.0"
464 | },
465 | "engines": {
466 | "node": ">= 0.8"
467 | }
468 | },
469 | "node_modules/follow-redirects": {
470 | "version": "1.15.9",
471 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz",
472 | "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==",
473 | "funding": [
474 | {
475 | "type": "individual",
476 | "url": "https://github.com/sponsors/RubenVerborgh"
477 | }
478 | ],
479 | "engines": {
480 | "node": ">=4.0"
481 | },
482 | "peerDependenciesMeta": {
483 | "debug": {
484 | "optional": true
485 | }
486 | }
487 | },
488 | "node_modules/form-data": {
489 | "version": "4.0.4",
490 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.4.tgz",
491 | "integrity": "sha512-KrGhL9Q4zjj0kiUt5OO4Mr/A/jlI2jDYs5eHBpYHPcBEVSiipAvn2Ko2HnPe20rmcuuvMHNdZFp+4IlGTMF0Ow==",
492 | "license": "MIT",
493 | "dependencies": {
494 | "asynckit": "^0.4.0",
495 | "combined-stream": "^1.0.8",
496 | "es-set-tostringtag": "^2.1.0",
497 | "hasown": "^2.0.2",
498 | "mime-types": "^2.1.12"
499 | },
500 | "engines": {
501 | "node": ">= 6"
502 | }
503 | },
504 | "node_modules/forwarded": {
505 | "version": "0.2.0",
506 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
507 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
508 | "engines": {
509 | "node": ">= 0.6"
510 | }
511 | },
512 | "node_modules/fresh": {
513 | "version": "0.5.2",
514 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
515 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==",
516 | "engines": {
517 | "node": ">= 0.6"
518 | }
519 | },
520 | "node_modules/fs": {
521 | "version": "0.0.1-security",
522 | "resolved": "https://registry.npmjs.org/fs/-/fs-0.0.1-security.tgz",
523 | "integrity": "sha512-3XY9e1pP0CVEUCdj5BmfIZxRBTSDycnbqhIOGec9QYtmVH2fbLpj86CFWkrNOkt/Fvty4KZG5lTglL9j/gJ87w=="
524 | },
525 | "node_modules/function-bind": {
526 | "version": "1.1.2",
527 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
528 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
529 | "funding": {
530 | "url": "https://github.com/sponsors/ljharb"
531 | }
532 | },
533 | "node_modules/get-intrinsic": {
534 | "version": "1.3.0",
535 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
536 | "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
537 | "license": "MIT",
538 | "dependencies": {
539 | "call-bind-apply-helpers": "^1.0.2",
540 | "es-define-property": "^1.0.1",
541 | "es-errors": "^1.3.0",
542 | "es-object-atoms": "^1.1.1",
543 | "function-bind": "^1.1.2",
544 | "get-proto": "^1.0.1",
545 | "gopd": "^1.2.0",
546 | "has-symbols": "^1.1.0",
547 | "hasown": "^2.0.2",
548 | "math-intrinsics": "^1.1.0"
549 | },
550 | "engines": {
551 | "node": ">= 0.4"
552 | },
553 | "funding": {
554 | "url": "https://github.com/sponsors/ljharb"
555 | }
556 | },
557 | "node_modules/get-proto": {
558 | "version": "1.0.1",
559 | "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
560 | "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
561 | "license": "MIT",
562 | "dependencies": {
563 | "dunder-proto": "^1.0.1",
564 | "es-object-atoms": "^1.0.0"
565 | },
566 | "engines": {
567 | "node": ">= 0.4"
568 | }
569 | },
570 | "node_modules/gopd": {
571 | "version": "1.2.0",
572 | "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
573 | "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
574 | "license": "MIT",
575 | "engines": {
576 | "node": ">= 0.4"
577 | },
578 | "funding": {
579 | "url": "https://github.com/sponsors/ljharb"
580 | }
581 | },
582 | "node_modules/has-property-descriptors": {
583 | "version": "1.0.2",
584 | "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz",
585 | "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==",
586 | "dependencies": {
587 | "es-define-property": "^1.0.0"
588 | },
589 | "funding": {
590 | "url": "https://github.com/sponsors/ljharb"
591 | }
592 | },
593 | "node_modules/has-symbols": {
594 | "version": "1.1.0",
595 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
596 | "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
597 | "license": "MIT",
598 | "engines": {
599 | "node": ">= 0.4"
600 | },
601 | "funding": {
602 | "url": "https://github.com/sponsors/ljharb"
603 | }
604 | },
605 | "node_modules/has-tostringtag": {
606 | "version": "1.0.2",
607 | "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz",
608 | "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==",
609 | "license": "MIT",
610 | "dependencies": {
611 | "has-symbols": "^1.0.3"
612 | },
613 | "engines": {
614 | "node": ">= 0.4"
615 | },
616 | "funding": {
617 | "url": "https://github.com/sponsors/ljharb"
618 | }
619 | },
620 | "node_modules/hasown": {
621 | "version": "2.0.2",
622 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
623 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
624 | "dependencies": {
625 | "function-bind": "^1.1.2"
626 | },
627 | "engines": {
628 | "node": ">= 0.4"
629 | }
630 | },
631 | "node_modules/http-errors": {
632 | "version": "2.0.0",
633 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz",
634 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==",
635 | "dependencies": {
636 | "depd": "2.0.0",
637 | "inherits": "2.0.4",
638 | "setprototypeof": "1.2.0",
639 | "statuses": "2.0.1",
640 | "toidentifier": "1.0.1"
641 | },
642 | "engines": {
643 | "node": ">= 0.8"
644 | }
645 | },
646 | "node_modules/iconv-lite": {
647 | "version": "0.4.24",
648 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
649 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
650 | "dependencies": {
651 | "safer-buffer": ">= 2.1.2 < 3"
652 | },
653 | "engines": {
654 | "node": ">=0.10.0"
655 | }
656 | },
657 | "node_modules/inherits": {
658 | "version": "2.0.4",
659 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
660 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
661 | },
662 | "node_modules/ipaddr.js": {
663 | "version": "1.9.1",
664 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
665 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
666 | "engines": {
667 | "node": ">= 0.10"
668 | }
669 | },
670 | "node_modules/isarray": {
671 | "version": "1.0.0",
672 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
673 | "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ=="
674 | },
675 | "node_modules/jsonwebtoken": {
676 | "version": "9.0.2",
677 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz",
678 | "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==",
679 | "dependencies": {
680 | "jws": "^3.2.2",
681 | "lodash.includes": "^4.3.0",
682 | "lodash.isboolean": "^3.0.3",
683 | "lodash.isinteger": "^4.0.4",
684 | "lodash.isnumber": "^3.0.3",
685 | "lodash.isplainobject": "^4.0.6",
686 | "lodash.isstring": "^4.0.1",
687 | "lodash.once": "^4.0.0",
688 | "ms": "^2.1.1",
689 | "semver": "^7.5.4"
690 | },
691 | "engines": {
692 | "node": ">=12",
693 | "npm": ">=6"
694 | }
695 | },
696 | "node_modules/jsonwebtoken/node_modules/ms": {
697 | "version": "2.1.3",
698 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
699 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
700 | },
701 | "node_modules/jwa": {
702 | "version": "1.4.1",
703 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz",
704 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==",
705 | "dependencies": {
706 | "buffer-equal-constant-time": "1.0.1",
707 | "ecdsa-sig-formatter": "1.0.11",
708 | "safe-buffer": "^5.0.1"
709 | }
710 | },
711 | "node_modules/jws": {
712 | "version": "3.2.2",
713 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz",
714 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==",
715 | "dependencies": {
716 | "jwa": "^1.4.1",
717 | "safe-buffer": "^5.0.1"
718 | }
719 | },
720 | "node_modules/lodash.includes": {
721 | "version": "4.3.0",
722 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz",
723 | "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w=="
724 | },
725 | "node_modules/lodash.isboolean": {
726 | "version": "3.0.3",
727 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz",
728 | "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg=="
729 | },
730 | "node_modules/lodash.isinteger": {
731 | "version": "4.0.4",
732 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz",
733 | "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA=="
734 | },
735 | "node_modules/lodash.isnumber": {
736 | "version": "3.0.3",
737 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz",
738 | "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw=="
739 | },
740 | "node_modules/lodash.isplainobject": {
741 | "version": "4.0.6",
742 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
743 | "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA=="
744 | },
745 | "node_modules/lodash.isstring": {
746 | "version": "4.0.1",
747 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz",
748 | "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw=="
749 | },
750 | "node_modules/lodash.once": {
751 | "version": "4.1.1",
752 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz",
753 | "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg=="
754 | },
755 | "node_modules/math-intrinsics": {
756 | "version": "1.1.0",
757 | "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
758 | "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
759 | "license": "MIT",
760 | "engines": {
761 | "node": ">= 0.4"
762 | }
763 | },
764 | "node_modules/media-typer": {
765 | "version": "0.3.0",
766 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
767 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
768 | "engines": {
769 | "node": ">= 0.6"
770 | }
771 | },
772 | "node_modules/merge-descriptors": {
773 | "version": "1.0.3",
774 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz",
775 | "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==",
776 | "funding": {
777 | "url": "https://github.com/sponsors/sindresorhus"
778 | }
779 | },
780 | "node_modules/methods": {
781 | "version": "1.1.2",
782 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
783 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==",
784 | "engines": {
785 | "node": ">= 0.6"
786 | }
787 | },
788 | "node_modules/mime": {
789 | "version": "1.6.0",
790 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
791 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
792 | "bin": {
793 | "mime": "cli.js"
794 | },
795 | "engines": {
796 | "node": ">=4"
797 | }
798 | },
799 | "node_modules/mime-db": {
800 | "version": "1.52.0",
801 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
802 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
803 | "engines": {
804 | "node": ">= 0.6"
805 | }
806 | },
807 | "node_modules/mime-types": {
808 | "version": "2.1.35",
809 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
810 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
811 | "dependencies": {
812 | "mime-db": "1.52.0"
813 | },
814 | "engines": {
815 | "node": ">= 0.6"
816 | }
817 | },
818 | "node_modules/minimist": {
819 | "version": "1.2.8",
820 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
821 | "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
822 | "funding": {
823 | "url": "https://github.com/sponsors/ljharb"
824 | }
825 | },
826 | "node_modules/mkdirp": {
827 | "version": "0.5.6",
828 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz",
829 | "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==",
830 | "dependencies": {
831 | "minimist": "^1.2.6"
832 | },
833 | "bin": {
834 | "mkdirp": "bin/cmd.js"
835 | }
836 | },
837 | "node_modules/ms": {
838 | "version": "2.0.0",
839 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
840 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="
841 | },
842 | "node_modules/multer": {
843 | "version": "1.4.5-lts.1",
844 | "resolved": "https://registry.npmjs.org/multer/-/multer-1.4.5-lts.1.tgz",
845 | "integrity": "sha512-ywPWvcDMeH+z9gQq5qYHCCy+ethsk4goepZ45GLD63fOu0YcNecQxi64nDs3qluZB+murG3/D4dJ7+dGctcCQQ==",
846 | "dependencies": {
847 | "append-field": "^1.0.0",
848 | "busboy": "^1.0.0",
849 | "concat-stream": "^1.5.2",
850 | "mkdirp": "^0.5.4",
851 | "object-assign": "^4.1.1",
852 | "type-is": "^1.6.4",
853 | "xtend": "^4.0.0"
854 | },
855 | "engines": {
856 | "node": ">= 6.0.0"
857 | }
858 | },
859 | "node_modules/nan": {
860 | "version": "2.22.0",
861 | "resolved": "https://registry.npmjs.org/nan/-/nan-2.22.0.tgz",
862 | "integrity": "sha512-nbajikzWTMwsW+eSsNm3QwlOs7het9gGJU5dDZzRTQGk03vyBOauxgI4VakDzE0PtsGTmXPsXTbbjVhRwR5mpw=="
863 | },
864 | "node_modules/negotiator": {
865 | "version": "0.6.3",
866 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
867 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==",
868 | "engines": {
869 | "node": ">= 0.6"
870 | }
871 | },
872 | "node_modules/object-assign": {
873 | "version": "4.1.1",
874 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
875 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
876 | "engines": {
877 | "node": ">=0.10.0"
878 | }
879 | },
880 | "node_modules/object-inspect": {
881 | "version": "1.13.2",
882 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.2.tgz",
883 | "integrity": "sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==",
884 | "engines": {
885 | "node": ">= 0.4"
886 | },
887 | "funding": {
888 | "url": "https://github.com/sponsors/ljharb"
889 | }
890 | },
891 | "node_modules/on-finished": {
892 | "version": "2.4.1",
893 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
894 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
895 | "dependencies": {
896 | "ee-first": "1.1.1"
897 | },
898 | "engines": {
899 | "node": ">= 0.8"
900 | }
901 | },
902 | "node_modules/parseurl": {
903 | "version": "1.3.3",
904 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
905 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
906 | "engines": {
907 | "node": ">= 0.8"
908 | }
909 | },
910 | "node_modules/path-to-regexp": {
911 | "version": "0.1.12",
912 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz",
913 | "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==",
914 | "license": "MIT"
915 | },
916 | "node_modules/process-nextick-args": {
917 | "version": "2.0.1",
918 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
919 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="
920 | },
921 | "node_modules/proxy-addr": {
922 | "version": "2.0.7",
923 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
924 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
925 | "dependencies": {
926 | "forwarded": "0.2.0",
927 | "ipaddr.js": "1.9.1"
928 | },
929 | "engines": {
930 | "node": ">= 0.10"
931 | }
932 | },
933 | "node_modules/proxy-from-env": {
934 | "version": "1.1.0",
935 | "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
936 | "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
937 | },
938 | "node_modules/qs": {
939 | "version": "6.13.0",
940 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz",
941 | "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==",
942 | "dependencies": {
943 | "side-channel": "^1.0.6"
944 | },
945 | "engines": {
946 | "node": ">=0.6"
947 | },
948 | "funding": {
949 | "url": "https://github.com/sponsors/ljharb"
950 | }
951 | },
952 | "node_modules/range-parser": {
953 | "version": "1.2.1",
954 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
955 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
956 | "engines": {
957 | "node": ">= 0.6"
958 | }
959 | },
960 | "node_modules/raw-body": {
961 | "version": "2.5.2",
962 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz",
963 | "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==",
964 | "dependencies": {
965 | "bytes": "3.1.2",
966 | "http-errors": "2.0.0",
967 | "iconv-lite": "0.4.24",
968 | "unpipe": "1.0.0"
969 | },
970 | "engines": {
971 | "node": ">= 0.8"
972 | }
973 | },
974 | "node_modules/readable-stream": {
975 | "version": "2.3.8",
976 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz",
977 | "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==",
978 | "dependencies": {
979 | "core-util-is": "~1.0.0",
980 | "inherits": "~2.0.3",
981 | "isarray": "~1.0.0",
982 | "process-nextick-args": "~2.0.0",
983 | "safe-buffer": "~5.1.1",
984 | "string_decoder": "~1.1.1",
985 | "util-deprecate": "~1.0.1"
986 | }
987 | },
988 | "node_modules/readable-stream/node_modules/safe-buffer": {
989 | "version": "5.1.2",
990 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
991 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
992 | },
993 | "node_modules/safe-buffer": {
994 | "version": "5.2.1",
995 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
996 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
997 | "funding": [
998 | {
999 | "type": "github",
1000 | "url": "https://github.com/sponsors/feross"
1001 | },
1002 | {
1003 | "type": "patreon",
1004 | "url": "https://www.patreon.com/feross"
1005 | },
1006 | {
1007 | "type": "consulting",
1008 | "url": "https://feross.org/support"
1009 | }
1010 | ]
1011 | },
1012 | "node_modules/safer-buffer": {
1013 | "version": "2.1.2",
1014 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1015 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
1016 | },
1017 | "node_modules/sax": {
1018 | "version": "1.4.1",
1019 | "resolved": "https://registry.npmjs.org/sax/-/sax-1.4.1.tgz",
1020 | "integrity": "sha512-+aWOz7yVScEGoKNd4PA10LZ8sk0A/z5+nXQG5giUO5rprX9jgYsTdov9qCchZiPIZezbZH+jRut8nPodFAX4Jg=="
1021 | },
1022 | "node_modules/semver": {
1023 | "version": "7.6.3",
1024 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz",
1025 | "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==",
1026 | "bin": {
1027 | "semver": "bin/semver.js"
1028 | },
1029 | "engines": {
1030 | "node": ">=10"
1031 | }
1032 | },
1033 | "node_modules/send": {
1034 | "version": "0.19.0",
1035 | "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz",
1036 | "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==",
1037 | "dependencies": {
1038 | "debug": "2.6.9",
1039 | "depd": "2.0.0",
1040 | "destroy": "1.2.0",
1041 | "encodeurl": "~1.0.2",
1042 | "escape-html": "~1.0.3",
1043 | "etag": "~1.8.1",
1044 | "fresh": "0.5.2",
1045 | "http-errors": "2.0.0",
1046 | "mime": "1.6.0",
1047 | "ms": "2.1.3",
1048 | "on-finished": "2.4.1",
1049 | "range-parser": "~1.2.1",
1050 | "statuses": "2.0.1"
1051 | },
1052 | "engines": {
1053 | "node": ">= 0.8.0"
1054 | }
1055 | },
1056 | "node_modules/send/node_modules/encodeurl": {
1057 | "version": "1.0.2",
1058 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
1059 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==",
1060 | "engines": {
1061 | "node": ">= 0.8"
1062 | }
1063 | },
1064 | "node_modules/send/node_modules/ms": {
1065 | "version": "2.1.3",
1066 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
1067 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
1068 | },
1069 | "node_modules/serve-static": {
1070 | "version": "1.16.2",
1071 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz",
1072 | "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==",
1073 | "dependencies": {
1074 | "encodeurl": "~2.0.0",
1075 | "escape-html": "~1.0.3",
1076 | "parseurl": "~1.3.3",
1077 | "send": "0.19.0"
1078 | },
1079 | "engines": {
1080 | "node": ">= 0.8.0"
1081 | }
1082 | },
1083 | "node_modules/set-function-length": {
1084 | "version": "1.2.2",
1085 | "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz",
1086 | "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==",
1087 | "dependencies": {
1088 | "define-data-property": "^1.1.4",
1089 | "es-errors": "^1.3.0",
1090 | "function-bind": "^1.1.2",
1091 | "get-intrinsic": "^1.2.4",
1092 | "gopd": "^1.0.1",
1093 | "has-property-descriptors": "^1.0.2"
1094 | },
1095 | "engines": {
1096 | "node": ">= 0.4"
1097 | }
1098 | },
1099 | "node_modules/setprototypeof": {
1100 | "version": "1.2.0",
1101 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
1102 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="
1103 | },
1104 | "node_modules/side-channel": {
1105 | "version": "1.0.6",
1106 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz",
1107 | "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==",
1108 | "dependencies": {
1109 | "call-bind": "^1.0.7",
1110 | "es-errors": "^1.3.0",
1111 | "get-intrinsic": "^1.2.4",
1112 | "object-inspect": "^1.13.1"
1113 | },
1114 | "engines": {
1115 | "node": ">= 0.4"
1116 | },
1117 | "funding": {
1118 | "url": "https://github.com/sponsors/ljharb"
1119 | }
1120 | },
1121 | "node_modules/statuses": {
1122 | "version": "2.0.1",
1123 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz",
1124 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==",
1125 | "engines": {
1126 | "node": ">= 0.8"
1127 | }
1128 | },
1129 | "node_modules/streamsearch": {
1130 | "version": "1.1.0",
1131 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz",
1132 | "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==",
1133 | "engines": {
1134 | "node": ">=10.0.0"
1135 | }
1136 | },
1137 | "node_modules/string_decoder": {
1138 | "version": "1.1.1",
1139 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
1140 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
1141 | "dependencies": {
1142 | "safe-buffer": "~5.1.0"
1143 | }
1144 | },
1145 | "node_modules/string_decoder/node_modules/safe-buffer": {
1146 | "version": "5.1.2",
1147 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1148 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1149 | },
1150 | "node_modules/toidentifier": {
1151 | "version": "1.0.1",
1152 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
1153 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
1154 | "engines": {
1155 | "node": ">=0.6"
1156 | }
1157 | },
1158 | "node_modules/type-is": {
1159 | "version": "1.6.18",
1160 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1161 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1162 | "dependencies": {
1163 | "media-typer": "0.3.0",
1164 | "mime-types": "~2.1.24"
1165 | },
1166 | "engines": {
1167 | "node": ">= 0.6"
1168 | }
1169 | },
1170 | "node_modules/typedarray": {
1171 | "version": "0.0.6",
1172 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz",
1173 | "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA=="
1174 | },
1175 | "node_modules/unpipe": {
1176 | "version": "1.0.0",
1177 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1178 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
1179 | "engines": {
1180 | "node": ">= 0.8"
1181 | }
1182 | },
1183 | "node_modules/util-deprecate": {
1184 | "version": "1.0.2",
1185 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
1186 | "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw=="
1187 | },
1188 | "node_modules/utils-merge": {
1189 | "version": "1.0.1",
1190 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1191 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==",
1192 | "engines": {
1193 | "node": ">= 0.4.0"
1194 | }
1195 | },
1196 | "node_modules/uuid": {
1197 | "version": "8.3.2",
1198 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
1199 | "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==",
1200 | "bin": {
1201 | "uuid": "dist/bin/uuid"
1202 | }
1203 | },
1204 | "node_modules/vary": {
1205 | "version": "1.1.2",
1206 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1207 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
1208 | "engines": {
1209 | "node": ">= 0.8"
1210 | }
1211 | },
1212 | "node_modules/xml2js": {
1213 | "version": "0.6.2",
1214 | "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.6.2.tgz",
1215 | "integrity": "sha512-T4rieHaC1EXcES0Kxxj4JWgaUQHDk+qwHcYOCFHfiwKz7tOVPLq7Hjq9dM1WCMhylqMEfP7hMcOIChvotiZegA==",
1216 | "dependencies": {
1217 | "sax": ">=0.6.0",
1218 | "xmlbuilder": "~11.0.0"
1219 | },
1220 | "engines": {
1221 | "node": ">=4.0.0"
1222 | }
1223 | },
1224 | "node_modules/xmlbuilder": {
1225 | "version": "11.0.1",
1226 | "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz",
1227 | "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==",
1228 | "engines": {
1229 | "node": ">=4.0"
1230 | }
1231 | },
1232 | "node_modules/xtend": {
1233 | "version": "4.0.2",
1234 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
1235 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
1236 | "engines": {
1237 | "node": ">=0.4"
1238 | }
1239 | }
1240 | }
1241 | }
1242 |
--------------------------------------------------------------------------------
/test-generation/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "test-generation",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "app.js",
6 | "scripts": {
7 | "test": "echo \"Error: no test specified\" && exit 1"
8 | },
9 | "keywords": [],
10 | "author": "",
11 | "license": "ISC",
12 | "dependencies": {
13 | "@azure/msal-node": "^2.15.0",
14 | "axios": "^1.7.7",
15 | "child_process": "^1.0.2",
16 | "dotenv": "^16.4.5",
17 | "edge": "^7.10.1",
18 | "express": "^4.21.1",
19 | "fs": "^0.0.1-security",
20 | "multer": "^1.4.5-lts.1",
21 | "xml2js": "^0.6.2"
22 | }
23 | }
24 |
--------------------------------------------------------------------------------
/test-generation/views/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Power BI Region, Workspaces, Reports, and Bookmarks
7 |
8 |
9 |
10 |
11 |
12 |
19 |
20 |
21 |
22 |
23 | Generate Test Cases
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
742 |
743 |
744 |
--------------------------------------------------------------------------------
/tests/paginated.spec.ts:
--------------------------------------------------------------------------------
1 | import { test, expect } from '@playwright/test';
2 | import { chromium } from 'playwright';
3 | import { getAccessToken, getPaginatedEmbedToken, TestSettings, getAPIEndpoints, PaginatedEmbedInfo } from '../helper-functions/token-helpers';
4 | // Used for local testings
5 | import { readJSONFilesFromFolder as readJSONFilesFromFolder } from '../helper-functions/file-reader';
6 | import { logToConsole } from '../helper-functions/logging';
7 |
8 | /* VARIABLES */
9 | if (!process.env.CLIENT_ID || !process.env.CLIENT_SECRET || !process.env.TENANT_ID || !process.env.ENVIRONMENT) {
10 | throw new Error('Missing required environment variables.');
11 | }
12 |
13 | // Initialize the environment variables
14 | let testRecords: Array;
15 | let endPoints;
16 | // Access environment variables
17 | let testSettings: TestSettings = {
18 | clientId: process.env.CLIENT_ID,
19 | clientSecret: process.env.CLIENT_SECRET,
20 | tenantId: process.env.TENANT_ID,
21 | environment: process.env.ENVIRONMENT as Environment
22 | };
23 |
24 | // Parse the JSON file locally
25 | testRecords = readJSONFilesFromFolder('./test-cases');
26 | //testRecords = testRecords.slice(0,2);
27 | endPoints = getAPIEndpoints(testSettings.environment);
28 | // Set logging for debugging
29 | let isVerboseLogging: boolean = true;
30 | // Get Access Token
31 | let testAccessToken: string | undefined = ""
32 |
33 | // Make sure to get token
34 | // Assume testing takes less then an hour at this point (before token expires)
35 | test.beforeAll(async () => {
36 | testAccessToken = await getAccessToken(testSettings);
37 | //isVerboseLogging = true;
38 | });
39 |
40 | /* TESTS */
41 |
42 | // Test to check if the access token is accessible
43 | test('test if access token can be generated from the environment variables provided (Paginated Report).', async ({ }) => {
44 | logToConsole('##[debug]Test if access token can be generated from the environment variables provided (Paginated Report).', isVerboseLogging);
45 | const token = testAccessToken
46 | //logToConsole('******** ' + token + '********', isVerboseLogging);
47 | expect(token).not.toBeUndefined();
48 | });
49 |
50 | // Test each record to check if the embed token is accessible
51 | testRecords.forEach((record) => {
52 | test(`test ${record.test_case} - '${record.report_name}' embed token is accessible ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''}`, async ({ }) => {
53 | logToConsole(`##[debug]test ${record.test_case} - '${record.report_name}' embed token is accessible ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''}`, isVerboseLogging);
54 | const tmpEmbedInfo: PaginatedEmbedInfo = {
55 | datasets: record.dataset_ids,
56 | reports: [{ id: record.report_id }]
57 | };
58 |
59 | //logToConsole('------' + testAccessToken + '-------', isVerboseLogging);
60 | const embedToken = await getPaginatedEmbedToken(tmpEmbedInfo, endPoints, testAccessToken);
61 | expect(embedToken).not.toBeUndefined();
62 | });
63 | }// end for
64 | );// end test
65 |
66 | // Test for visual errors
67 | testRecords.forEach((record) => {
68 | test(`test ${record.test_case} - '${record.report_name}' for visual errors ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''}, Link: ${endPoints.webPrefix}/groups/${record.workspace_id}/rdlreports/${record.report_id}${record.report_parameters_string != '' && record.report_parameters_string !== undefined ? "?" + record.report_parameters_string : ''} `, async ({ browser }) => {
69 | logToConsole(`##[debug]test ${record.test_case} - '${record.report_name}' for visual errors ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''}, Link: ${endPoints.webPrefix}/groups/${record.workspace_id}/rdlreports/${record.report_id}${record.report_parameters_string != '' && record.report_parameters_string !== undefined ? "?" + record.report_parameters_string : ''}`,isVerboseLogging);
70 | //logToConsole(record, isVerboseLogging);
71 | //const accessToken = await getAccessToken(testSettings);
72 | browser = await chromium.launch({ args: ['--disable-web-security'], headless: false });
73 | const context = await browser.newContext();
74 | const page = await context.newPage();
75 |
76 | // Get the embedURL for the paginated report
77 | const reportResponse = await fetch(`${endPoints.apiPrefix}/v1.0/myorg/groups/${record.workspace_id}/reports/${record.report_id}`, {
78 | method: 'GET',
79 | headers: {
80 | 'Authorization': `Bearer ${testAccessToken}`
81 | }
82 | }).then(res => res.json());
83 |
84 | await page.goto('about:blank');
85 | await page.addScriptTag({ url: 'https://cdnjs.cloudflare.com/ajax/libs/powerbi-client/2.23.7/powerbi.min.js' });
86 |
87 | const tmpEmbedInfo: PaginatedEmbedInfo = {
88 | datasets: record.dataset_ids,
89 | reports: [{ id: record.report_id }]
90 | };
91 |
92 | const embedToken = await getPaginatedEmbedToken(tmpEmbedInfo, endPoints, testAccessToken);
93 |
94 | let reportInfo = {
95 | reportId: record.report_id,
96 | parameterValues: record.report_parameters,
97 | embedUrl: reportResponse.embedUrl, // use one from report response
98 | embedToken: embedToken,
99 | endpoints: endPoints,
100 | waitSeconds: record.wait_seconds || 20
101 | };
102 |
103 | // Evaluate the page and check for visual errors
104 | let test = await page.evaluate(async (reportInfo: any) => {
105 | var pbi = window['powerbi-client'];
106 | var models = window['powerbi-client'].models;
107 | let embedConfiguration: any = {
108 | type: 'report',
109 | id: reportInfo.reportId,
110 | embedUrl: reportInfo.embedUrl,
111 | accessToken: reportInfo.embedToken,
112 | tokenType: models.TokenType.Embed,
113 | permissions: models.Permissions.Read,
114 | viewMode: models.ViewMode.View,
115 | parameterValues: reportInfo.parameterValues
116 | };
117 |
118 | // Add parameters if they exist
119 | if(reportInfo.parameterValues){
120 | embedConfiguration['parameterValues'] = reportInfo.parameterValues;
121 | }
122 |
123 | // Initialize the powerbi service
124 | const powerbi = new pbi.service.Service(pbi.factories.hpmFactory, pbi.factories.wpmpFactory, pbi.factories.routerFactory);
125 | let embed = powerbi.embed(document.body, embedConfiguration);
126 |
127 | // Wait for the report to typically load
128 | let pauseProme = new Promise((resolve) => {
129 | setTimeout(() => {
130 | resolve();
131 | }, reportInfo.waitSeconds * 1000);
132 | });
133 | let result = await pauseProme;
134 | return result;
135 | }, reportInfo)
136 |
137 | await page.waitForLoadState('networkidle'); // Ensure all JS-loaded content is finished
138 | // Get the main page's HTML
139 | let fullPageHTML = await page.content();
140 | // Find all iframes on the page
141 | const frames = page.frames().filter(frame => frame !== page.mainFrame());
142 |
143 | for (const frame of frames) {
144 | const frameContent = await frame.evaluate(() => document.documentElement.outerHTML);
145 | const frameUrl = frame.url();
146 | fullPageHTML += `\n\n${frameContent}\n`;
147 | }
148 |
149 | const searchForModal = fullPageHTML.search('"ms-Dialog-content');
150 | expect(searchForModal).toBe(-1);
151 | });
152 | });
--------------------------------------------------------------------------------
/tests/pbi.spec.ts:
--------------------------------------------------------------------------------
1 | import { test, expect } from '@playwright/test';
2 | import { chromium } from 'playwright';
3 | import {
4 | getAccessToken,
5 | TestSettings, getAPIEndpoints,
6 | getReportEmbedToken,
7 | createReportEmbedInfo
8 |
9 | } from '../helper-functions/token-helpers';
10 | // Used for local testings
11 | import { IReportEmbedConfiguration } from 'powerbi-client';
12 | import { readCSVFilesFromFolder } from '../helper-functions/file-reader';
13 | import { logToConsole } from '../helper-functions/logging';
14 |
15 | /* VARIABLES */
16 | if (!process.env.CLIENT_ID || !process.env.CLIENT_SECRET || !process.env.TENANT_ID || !process.env.ENVIRONMENT) {
17 | throw new Error('Missing required environment variables.');
18 | }
19 |
20 | // Initialize the environment variables
21 | let testRecords: Array;
22 | let endPoints;
23 | // Access environment variables
24 | let testSettings: TestSettings = {
25 | clientId: process.env.CLIENT_ID,
26 | clientSecret: process.env.CLIENT_SECRET,
27 | tenantId: process.env.TENANT_ID,
28 | environment: process.env.ENVIRONMENT as Environment
29 | };
30 |
31 | // Parse the JSON file locally
32 | testRecords = readCSVFilesFromFolder('./test-cases');
33 | //testRecords = testRecords.slice(0,2);
34 | endPoints = getAPIEndpoints(testSettings.environment);
35 | // Set logging for debugging
36 | let isVerboseLogging: boolean = true;
37 | // Get Access Token
38 | let testAccessToken: string | undefined = ""
39 |
40 | // Make sure to get token
41 | // Assume testing takes less then an hour at this point (before token expires)
42 | test.beforeAll(async () => {
43 | testAccessToken = await getAccessToken(testSettings);
44 | //isVerboseLogging = true;
45 | });
46 |
47 | /* TESTS */
48 |
49 | // Test to check if the access token is accessible
50 | test('test if access token can be generated from the environment variables provided.', async ({ }) => {
51 | logToConsole('##[debug]Test if access token can be generated from the environment variables provided.', isVerboseLogging);
52 | const token = testAccessToken
53 | //logToConsole('******** ' + token + '********', isVerboseLogging);
54 | expect(token).not.toBeUndefined();
55 | });
56 |
57 | // Test each record to check if the embed token is accessible
58 | testRecords.forEach((record) => {
59 | test(`test ${record.test_case} - '${record.report_name}' embed token is accessible ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''} ${record.bookmark_id != '' && record.bookmark_id !== undefined ? "(Bookmark: " + record.bookmark_name + ")" : ''}`, async ({ }) => {
60 | logToConsole(`##[debug]test ${record.test_case} - '${record.report_name}' embed token is accessible ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''} ${record.bookmark_id != '' && record.bookmark_id !== undefined ? "(Bookmark: " + record.bookmark_name + ")" : ''}`, isVerboseLogging);
61 |
62 | const tmpEmbedInfo = createReportEmbedInfo(record);
63 | const embedToken = await getReportEmbedToken(tmpEmbedInfo, endPoints, testAccessToken);
64 | //logToConsole('------' + testAccessToken + '-------', isVerboseLogging);
65 | //const embedToken = await getEmbedToken(tmpEmbedInfo, endPoints, testAccessToken);
66 | expect(embedToken).not.toBeUndefined();
67 | });
68 | }// end for
69 | );// end test
70 |
71 | // Test for visual errors
72 | testRecords.forEach((record) => {
73 | test(`test ${record.test_case} - '${record.report_name}' for visual errors ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''} ${record.bookmark_id != '' && record.bookmark_id !== undefined ? "(Bookmark: " + record.bookmark_name + ")" : ''}, Link: ${endPoints.webPrefix}/groups/${record.workspace_id}/reports/${record.report_id}/${record.page_id}${record.bookmark_id != '' && record.bookmark_id !== undefined ? "?bookmarkGuid=" + record.bookmark_id : ""} `, async ({ browser }) => {
74 | logToConsole(`##[debug]test ${record.test_case} - '${record.report_name}' for visual errors ${record.role != '' && record.role !== undefined ? "(Role: " + record.role + ") " : ''} ${record.bookmark_id != '' && record.bookmark_id !== undefined ? "(Bookmark: " + record.bookmark_name + ")" : ''}, Link: ${endPoints.webPrefix}/groups/${record.workspace_id}/reports/${record.report_id}/${record.page_id}${record.bookmark_id != '' && record.bookmark_id !== undefined ? "?bookmarkGuid=" + record.bookmark_id : ""}`, isVerboseLogging);
75 | const accessToken = await getAccessToken(testSettings);
76 | browser = await chromium.launch({ args: ['--disable-web-security'], headless: false });
77 | const context = await browser.newContext();
78 | const page = await context.newPage();
79 |
80 | // Get the embedURL for the report
81 | const reportResponse = await fetch(`${endPoints.apiPrefix}/v1.0/myorg/groups/${record.workspace_id}/reports/${record.report_id}`, {
82 | method: 'GET',
83 | headers: {
84 | 'Authorization': `Bearer ${testAccessToken}`
85 | }
86 | }).then(res => res.json());
87 |
88 | await page.goto('about:blank');
89 | await page.addScriptTag({ url: 'https://cdnjs.cloudflare.com/ajax/libs/powerbi-client/2.23.1/powerbi.min.js' });
90 |
91 | const tmpEmbedInfo = createReportEmbedInfo(record);
92 | // Get the embedURL for the Power BI report
93 | const embedToken = await getReportEmbedToken(tmpEmbedInfo, endPoints, testAccessToken);
94 |
95 | let reportInfo = {
96 | reportId: record.report_id,
97 | page_id: record.page_id,
98 | embedUrl: reportResponse.embedUrl, // use the one from the report response
99 | embedToken: embedToken,
100 | endpoints: endPoints
101 | };
102 |
103 | // Handle the bookmark
104 | if (record.bookmark_id != '' && record.bookmark_id !== undefined) {
105 | reportInfo['bookmark_id'] = record.bookmark_id;
106 | }
107 |
108 | // Evaluate the page and check for visual errors
109 | let test = await page.evaluate(async (reportInfo: any) => {
110 | var pbi = window['powerbi-client'];
111 | var models = window['powerbi-client'].models;
112 | let embedConfiguration: IReportEmbedConfiguration = {
113 | type: 'report',
114 | id: reportInfo.reportId,
115 | pageName: reportInfo.page_id,
116 | embedUrl: reportInfo.embedUrl,
117 | accessToken: reportInfo.embedToken,
118 | tokenType: models.TokenType.Embed,
119 | permissions: models.Permissions.Read,
120 | viewMode: models.ViewMode.View
121 | };
122 |
123 | // Apply bookmark if it exists
124 | if (reportInfo.bookmark_id && reportInfo.bookmark_id.trim() !== "") {
125 | embedConfiguration = {
126 | type: 'report',
127 | id: reportInfo.reportId,
128 | pageName: reportInfo.page_id,
129 | bookmark: {
130 | name: reportInfo.bookmark_id
131 | },
132 | embedUrl: reportInfo.embedUrl,
133 | accessToken: reportInfo.embedToken,
134 | tokenType: models.TokenType.Embed,
135 | permissions: models.Permissions.Read,
136 | viewMode: models.ViewMode.View
137 | };
138 | }// apply bookmark
139 |
140 | // Initialize the powerbi service
141 | const powerbi = new pbi.service.Service(pbi.factories.hpmFactory, pbi.factories.wpmpFactory, pbi.factories.routerFactory);
142 | let embed = powerbi.embed(document.body, embedConfiguration);
143 |
144 | // Wait for the report to render or error out using the promises
145 | const once = {
146 | once: true,
147 | };
148 | let testErrorPromise = new Promise((resolve) => {
149 | document.body.addEventListener('error', async function (event: any) {
150 | resolve(event);
151 | }, once);
152 | });
153 | let testRenderedPromise = new Promise((resolve) => {
154 | document.body.addEventListener('rendered', async function (event: any) {
155 | resolve();// resolve undefined
156 | }, once);
157 | });
158 |
159 | // Wait for the report to render or error out using the race condition
160 | let result = await Promise.race([testErrorPromise, testRenderedPromise]);
161 | return result === undefined ? "passed" : "failed";
162 |
163 | }, reportInfo)
164 |
165 | expect(test).toBe("passed");
166 | });
167 | });
--------------------------------------------------------------------------------