Compare commits
	
		
			6 Commits
		
	
	
		
			snackbar-o
			...
			AUT-499
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 466d58b3d2 | ||
|   | c0f0415f70 | ||
|   | 23c40c89ee | ||
|   | 77f84944c7 | ||
|   | 8a8be21d56 | ||
|   | e5c4e18fd5 | 
| @@ -28,7 +28,7 @@ cd packages/web | |||||||
| rm -rf .env | rm -rf .env | ||||||
| echo " | echo " | ||||||
| PORT=$WEB_PORT | PORT=$WEB_PORT | ||||||
| REACT_APP_BACKEND_URL=http://localhost:$BACKEND_PORT | REACT_APP_GRAPHQL_URL=http://localhost:$BACKEND_PORT/graphql | ||||||
| " >> .env | " >> .env | ||||||
| cd $CURRENT_DIR | cd $CURRENT_DIR | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|       "version": "latest" |       "version": "latest" | ||||||
|     }, |     }, | ||||||
|     "ghcr.io/devcontainers/features/node:1": { |     "ghcr.io/devcontainers/features/node:1": { | ||||||
|       "version": 18 |       "version": 16 | ||||||
|     }, |     }, | ||||||
|     "ghcr.io/devcontainers/features/common-utils:1": { |     "ghcr.io/devcontainers/features/common-utils:1": { | ||||||
|       "username": "vscode", |       "username": "vscode", | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | module.exports = { | ||||||
|  |   root: true, | ||||||
|  |   parser: '@typescript-eslint/parser', | ||||||
|  |   plugins: ['@typescript-eslint'], | ||||||
|  |   extends: [ | ||||||
|  |     'eslint:recommended', | ||||||
|  |     'plugin:@typescript-eslint/recommended', | ||||||
|  |     'prettier', | ||||||
|  |   ], | ||||||
|  |   overrides: [ | ||||||
|  |     { | ||||||
|  |       files: ['**/*.test.ts', '**/test/**/*.ts'], | ||||||
|  |       rules: { | ||||||
|  |         '@typescript-eslint/ban-ts-comment': ['off'], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  | }; | ||||||
							
								
								
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -22,7 +22,7 @@ jobs: | |||||||
|       - run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner." |       - run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner." | ||||||
|       - run: echo "🖥️ The workflow is now ready to test your code on the runner." |       - run: echo "🖥️ The workflow is now ready to test your code on the runner." | ||||||
|       - run: yarn --frozen-lockfile |       - run: yarn --frozen-lockfile | ||||||
|       - run: cd packages/backend && yarn lint |       - run: yarn lint | ||||||
|       - run: echo "🍏 This job's status is ${{ job.status }}." |       - run: echo "🍏 This job's status is ${{ job.status }}." | ||||||
|   start-backend-server: |   start-backend-server: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
								
							| @@ -62,9 +62,8 @@ jobs: | |||||||
|         run: yarn && yarn lerna bootstrap |         run: yarn && yarn lerna bootstrap | ||||||
|       - name: Install Playwright Browsers |       - name: Install Playwright Browsers | ||||||
|         run: yarn playwright install --with-deps |         run: yarn playwright install --with-deps | ||||||
|       - name: Build Automatisch web |       - name: Build Automatisch | ||||||
|         working-directory: ./packages/web |         run: yarn lerna run --scope=@*/{web,cli} build | ||||||
|         run: yarn build |  | ||||||
|         env: |         env: | ||||||
|           # Keep this until we clean up warnings in build processes |           # Keep this until we clean up warnings in build processes | ||||||
|           CI: false |           CI: false | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|     "start": "lerna run --stream --parallel --scope=@*/{web,backend} dev", |     "start": "lerna run --stream --parallel --scope=@*/{web,backend} dev", | ||||||
|     "start:web": "lerna run --stream --scope=@*/web dev", |     "start:web": "lerna run --stream --scope=@*/web dev", | ||||||
|     "start:backend": "lerna run --stream --scope=@*/backend dev", |     "start:backend": "lerna run --stream --scope=@*/backend dev", | ||||||
|  |     "lint": "lerna run --no-bail --stream --parallel --scope=@*/{web,backend} lint", | ||||||
|     "build:docs": "cd ./packages/docs && yarn install && yarn build" |     "build:docs": "cd ./packages/docs && yarn install && yarn build" | ||||||
|   }, |   }, | ||||||
|   "workspaces": { |   "workspaces": { | ||||||
| @@ -20,6 +21,8 @@ | |||||||
|     ] |     ] | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|  |     "@typescript-eslint/eslint-plugin": "^5.9.1", | ||||||
|  |     "@typescript-eslint/parser": "^5.9.1", | ||||||
|     "eslint": "^8.13.0", |     "eslint": "^8.13.0", | ||||||
|     "eslint-config-prettier": "^8.3.0", |     "eslint-config-prettier": "^8.3.0", | ||||||
|     "eslint-plugin-prettier": "^4.0.0", |     "eslint-plugin-prettier": "^4.0.0", | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ | |||||||
|     "start:worker": "node src/worker.js", |     "start:worker": "node src/worker.js", | ||||||
|     "pretest": "APP_ENV=test node ./test/setup/prepare-test-env.js", |     "pretest": "APP_ENV=test node ./test/setup/prepare-test-env.js", | ||||||
|     "test": "APP_ENV=test vitest run", |     "test": "APP_ENV=test vitest run", | ||||||
|     "lint": "eslint .", |     "lint": "eslint . --ignore-path ../../.eslintignore", | ||||||
|     "db:create": "node ./bin/database/create.js", |     "db:create": "node ./bin/database/create.js", | ||||||
|     "db:seed:user": "node ./bin/database/seed-user.js", |     "db:seed:user": "node ./bin/database/seed-user.js", | ||||||
|     "db:drop": "node ./bin/database/drop.js", |     "db:drop": "node ./bin/database/drop.js", | ||||||
| @@ -33,18 +33,19 @@ | |||||||
|     "axios": "1.6.0", |     "axios": "1.6.0", | ||||||
|     "bcrypt": "^5.0.1", |     "bcrypt": "^5.0.1", | ||||||
|     "bullmq": "^3.0.0", |     "bullmq": "^3.0.0", | ||||||
|  |     "copyfiles": "^2.4.1", | ||||||
|     "cors": "^2.8.5", |     "cors": "^2.8.5", | ||||||
|     "crypto-js": "^4.1.1", |     "crypto-js": "^4.1.1", | ||||||
|     "debug": "~2.6.9", |     "debug": "~2.6.9", | ||||||
|     "dotenv": "^10.0.0", |     "dotenv": "^10.0.0", | ||||||
|     "express": "~4.18.2", |     "express": "~4.18.2", | ||||||
|     "express-async-handler": "^1.2.0", |  | ||||||
|     "express-basic-auth": "^1.2.1", |     "express-basic-auth": "^1.2.1", | ||||||
|     "express-graphql": "^0.12.0", |     "express-graphql": "^0.12.0", | ||||||
|     "fast-xml-parser": "^4.0.11", |     "fast-xml-parser": "^4.0.11", | ||||||
|     "graphql-middleware": "^6.1.15", |     "graphql-middleware": "^6.1.15", | ||||||
|     "graphql-shield": "^7.5.0", |     "graphql-shield": "^7.5.0", | ||||||
|     "graphql-tools": "^8.2.0", |     "graphql-tools": "^8.2.0", | ||||||
|  |     "graphql-type-json": "^0.3.2", | ||||||
|     "handlebars": "^4.7.7", |     "handlebars": "^4.7.7", | ||||||
|     "http-errors": "~1.6.3", |     "http-errors": "~1.6.3", | ||||||
|     "http-proxy-agent": "^7.0.0", |     "http-proxy-agent": "^7.0.0", | ||||||
| @@ -67,6 +68,7 @@ | |||||||
|     "pluralize": "^8.0.0", |     "pluralize": "^8.0.0", | ||||||
|     "raw-body": "^2.5.2", |     "raw-body": "^2.5.2", | ||||||
|     "showdown": "^2.1.0", |     "showdown": "^2.1.0", | ||||||
|  |     "stripe": "^11.13.0", | ||||||
|     "winston": "^3.7.1", |     "winston": "^3.7.1", | ||||||
|     "xmlrpc": "^1.3.2" |     "xmlrpc": "^1.3.2" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Acknowledge incident', | ||||||
|  |   key: 'acknowledgeIncident', | ||||||
|  |   description: 'Acknowledges an incident.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Incident ID', | ||||||
|  |       key: 'incidentId', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         'This serves as the incident ID that requires your acknowledgment.', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Acknowledged by', | ||||||
|  |       key: 'acknowledgedBy', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         "This refers to the individual's name, email, or another form of identification that the person who acknowledged the incident has provided.", | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const acknowledgedBy = $.step.parameters.acknowledgedBy; | ||||||
|  |     const incidentId = $.step.parameters.incidentId; | ||||||
|  |  | ||||||
|  |     const body = { | ||||||
|  |       acknowledged_by: acknowledgedBy, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const response = await $.http.post( | ||||||
|  |       `/v2/incidents/${incidentId}/acknowledge`, | ||||||
|  |       body | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ raw: response.data.data }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,120 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Create incident', | ||||||
|  |   key: 'createIncident', | ||||||
|  |   description: 'Creates an incident that informs the team.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Brief Summary', | ||||||
|  |       key: 'briefSummary', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       variables: true, | ||||||
|  |       description: 'A short description outlining the issue.', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Description', | ||||||
|  |       key: 'description', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         'An elaborate description of the situation, offering insights into what is occurring, along with instructions to reproduce the problem.', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Requester Email', | ||||||
|  |       key: 'requesterEmail', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         'This represents the email address of the individual who initiated the incident request.', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Alert Settings - Call', | ||||||
|  |       key: 'alertSettingsCall', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Should we call the on-call person?', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Yes', value: 'true' }, | ||||||
|  |         { label: 'No', value: 'false' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Alert Settings - Text', | ||||||
|  |       key: 'alertSettingsText', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Should we text the on-call person?', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Yes', value: 'true' }, | ||||||
|  |         { label: 'No', value: 'false' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Alert Settings - Email', | ||||||
|  |       key: 'alertSettingsEmail', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Should we email the on-call person?', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Yes', value: 'true' }, | ||||||
|  |         { label: 'No', value: 'false' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Alert Settings - Push Notification', | ||||||
|  |       key: 'alertSettingsPushNotification', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Should we send a push notification to the on-call person?', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Yes', value: 'true' }, | ||||||
|  |         { label: 'No', value: 'false' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Team Alert Wait Time', | ||||||
|  |       key: 'teamAlertWaitTime', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         "What is the time threshold for acknowledgment before escalating to the entire team? (Specify in seconds) - Use a negative value to indicate no team alert if the on-call person doesn't respond, and use 0 for an immediate alert to the entire team.", | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { | ||||||
|  |       briefSummary, | ||||||
|  |       description, | ||||||
|  |       requesterEmail, | ||||||
|  |       alertSettingsCall, | ||||||
|  |       alertSettingsText, | ||||||
|  |       alertSettingsEmail, | ||||||
|  |       alertSettingsPushNotification, | ||||||
|  |       teamAlertWaitTime, | ||||||
|  |     } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const body = { | ||||||
|  |       summary: briefSummary, | ||||||
|  |       description, | ||||||
|  |       requester_email: requesterEmail, | ||||||
|  |       call: alertSettingsCall, | ||||||
|  |       sms: alertSettingsText, | ||||||
|  |       email: alertSettingsEmail, | ||||||
|  |       push: alertSettingsPushNotification, | ||||||
|  |       team_wait: teamAlertWaitTime, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const response = await $.http.post('/v2/incidents', body); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ raw: response.data.data }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,25 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Find incident', | ||||||
|  |   key: 'findIncident', | ||||||
|  |   description: 'finds an incident.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Incident ID', | ||||||
|  |       key: 'incidentId', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       variables: true, | ||||||
|  |       description: 'ID for querying incidents.', | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const incidentId = $.step.parameters.incidentId; | ||||||
|  |  | ||||||
|  |     const response = await $.http.get(`/v2/incidents/${incidentId}`); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ raw: response.data.data }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
							
								
								
									
										11
									
								
								packages/backend/src/apps/better-stack/actions/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/src/apps/better-stack/actions/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | import acknowledgeIncident from './acknowledge-incident/index.js'; | ||||||
|  | import createIncident from './create-incident/index.js'; | ||||||
|  | import findIncident from './find-incident/index.js'; | ||||||
|  | import resolveIncident from './resolve-incident/index.js'; | ||||||
|  |  | ||||||
|  | export default [ | ||||||
|  |   acknowledgeIncident, | ||||||
|  |   createIncident, | ||||||
|  |   findIncident, | ||||||
|  |   resolveIncident, | ||||||
|  | ]; | ||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Resolve incident', | ||||||
|  |   key: 'resolveIncident', | ||||||
|  |   description: 'Resolves an incident.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Incident ID', | ||||||
|  |       key: 'incidentId', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         'This represents the identification for an incident that requires resolution.', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Resolved by', | ||||||
|  |       key: 'resolvedBy', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         "This refers to the individual's name, email, or another form of identification that the person who resolved the incident has provided.", | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const resolvedBy = $.step.parameters.resolvedBy; | ||||||
|  |     const incidentId = $.step.parameters.incidentId; | ||||||
|  |  | ||||||
|  |     const body = { | ||||||
|  |       resolved_by: resolvedBy, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const response = await $.http.post( | ||||||
|  |       `/v2/incidents/${incidentId}/resolve`, | ||||||
|  |       body | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ raw: response.data.data }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
							
								
								
									
										21
									
								
								packages/backend/src/apps/better-stack/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								packages/backend/src/apps/better-stack/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | <?xml version="1.0" standalone="no"?> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN" | ||||||
|  |  "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd"> | ||||||
|  | <svg version="1.0" xmlns="http://www.w3.org/2000/svg" | ||||||
|  |  width="200.000000pt" height="200.000000pt" viewBox="0 0 200.000000 200.000000" | ||||||
|  |  preserveAspectRatio="xMidYMid meet"> | ||||||
|  |  | ||||||
|  | <g transform="translate(0.000000,200.000000) scale(0.100000,-0.100000)" | ||||||
|  | fill="#000" stroke="none"> | ||||||
|  | <path d="M0 1000 l0 -1000 1000 0 1000 0 0 1000 0 1000 -1000 0 -1000 0 0 | ||||||
|  | -1000z m1162 460 c14 -11 113 -184 232 -408 228 -429 231 -439 175 -486 -35 | ||||||
|  | -30 -30 -29 -140 -15 -89 12 -123 25 -152 56 -9 11 -72 147 -140 304 -113 263 | ||||||
|  | -124 284 -149 287 -14 2 -29 10 -32 17 -8 21 67 214 94 242 28 29 78 30 112 3z | ||||||
|  | m-340 -148 c10 -10 72 -175 139 -367 114 -325 121 -351 108 -374 -8 -14 -27 | ||||||
|  | -32 -41 -41 -25 -13 -34 -12 -126 18 -55 18 -111 43 -125 56 -19 17 -40 67 | ||||||
|  | -76 182 -36 112 -58 164 -73 176 l-22 16 27 99 c63 224 66 232 95 248 31 17 | ||||||
|  | 69 12 94 -13z m-314 -219 c16 -15 26 -59 56 -243 42 -262 43 -285 17 -300 -11 | ||||||
|  | -5 -24 -10 -30 -10 -19 0 -140 114 -150 141 -7 20 -4 76 10 191 10 90 19 171 | ||||||
|  | 19 181 0 18 33 57 49 57 5 0 18 -8 29 -17z"/> | ||||||
|  | </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										33
									
								
								packages/backend/src/apps/better-stack/auth/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								packages/backend/src/apps/better-stack/auth/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | import verifyCredentials from './verify-credentials.js'; | ||||||
|  | import isStillVerified from './is-still-verified.js'; | ||||||
|  |  | ||||||
|  | export default { | ||||||
|  |   fields: [ | ||||||
|  |     { | ||||||
|  |       key: 'screenName', | ||||||
|  |       label: 'Screen Name', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: | ||||||
|  |         'Screen name of your connection to be used on Automatisch UI.', | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'apiKey', | ||||||
|  |       label: 'API Key', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: 'Better Stack API key of your account.', | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   verifyCredentials, | ||||||
|  |   isStillVerified, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | import verifyCredentials from './verify-credentials.js'; | ||||||
|  |  | ||||||
|  | const isStillVerified = async ($) => { | ||||||
|  |   await verifyCredentials($); | ||||||
|  |   return true; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default isStillVerified; | ||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | const verifyCredentials = async ($) => { | ||||||
|  |   await $.http.get('/v2/metadata'); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     screenName: $.auth.data.screenName, | ||||||
|  |     apiKey: $.auth.data.apiKey, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default verifyCredentials; | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | const addAuthHeader = ($, requestConfig) => { | ||||||
|  |   if ($.auth.data?.apiKey) { | ||||||
|  |     requestConfig.headers.Authorization = `Bearer ${$.auth.data.apiKey}`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default addAuthHeader; | ||||||
							
								
								
									
										18
									
								
								packages/backend/src/apps/better-stack/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								packages/backend/src/apps/better-stack/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | import defineApp from '../../helpers/define-app.js'; | ||||||
|  | import addAuthHeader from './common/add-auth-header.js'; | ||||||
|  | import auth from './auth/index.js'; | ||||||
|  | import actions from './actions/index.js'; | ||||||
|  |  | ||||||
|  | export default defineApp({ | ||||||
|  |   name: 'Better Stack', | ||||||
|  |   key: 'better-stack', | ||||||
|  |   iconUrl: '{BASE_URL}/apps/better-stack/assets/favicon.svg', | ||||||
|  |   authDocUrl: 'https://automatisch.io/docs/apps/better-stack/connection', | ||||||
|  |   supportsConnections: true, | ||||||
|  |   baseUrl: 'https://betterstack.com', | ||||||
|  |   apiBaseUrl: 'https://uptime.betterstack.com/api', | ||||||
|  |   primaryColor: '000000', | ||||||
|  |   beforeRequest: [addAuthHeader], | ||||||
|  |   auth, | ||||||
|  |   actions, | ||||||
|  | }); | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| import defineAction from '../../../../helpers/define-action.js'; |  | ||||||
|  |  | ||||||
| export default defineAction({ |  | ||||||
|   name: 'Get value', |  | ||||||
|   key: 'getValue', |  | ||||||
|   description: 'Get value from the persistent datastore.', |  | ||||||
|   arguments: [ |  | ||||||
|     { |  | ||||||
|       label: 'Key', |  | ||||||
|       key: 'key', |  | ||||||
|       type: 'string', |  | ||||||
|       required: true, |  | ||||||
|       description: 'The key of your value to get.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|   ], |  | ||||||
|  |  | ||||||
|   async run($) { |  | ||||||
|     const keyValuePair = await $.datastore.get({ |  | ||||||
|       key: $.step.parameters.key, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     $.setActionItem({ |  | ||||||
|       raw: keyValuePair, |  | ||||||
|     }); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| import getValue from './get-value/index.js'; |  | ||||||
| import setValue from './set-value/index.js'; |  | ||||||
|  |  | ||||||
| export default [getValue, setValue]; |  | ||||||
| @@ -1,36 +0,0 @@ | |||||||
| import defineAction from '../../../../helpers/define-action.js'; |  | ||||||
|  |  | ||||||
| export default defineAction({ |  | ||||||
|   name: 'Set value', |  | ||||||
|   key: 'setValue', |  | ||||||
|   description: 'Set value to the persistent datastore.', |  | ||||||
|   arguments: [ |  | ||||||
|     { |  | ||||||
|       label: 'Key', |  | ||||||
|       key: 'key', |  | ||||||
|       type: 'string', |  | ||||||
|       required: true, |  | ||||||
|       description: 'The key of your value to set.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       label: 'Value', |  | ||||||
|       key: 'value', |  | ||||||
|       type: 'string', |  | ||||||
|       required: true, |  | ||||||
|       description: 'The value to set.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|   ], |  | ||||||
|  |  | ||||||
|   async run($) { |  | ||||||
|     const keyValuePair = await $.datastore.set({ |  | ||||||
|       key: $.step.parameters.key, |  | ||||||
|       value: $.step.parameters.value, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     $.setActionItem({ |  | ||||||
|       raw: keyValuePair, |  | ||||||
|     }); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| <?xml version="1.0"?> |  | ||||||
| <svg xmlns="http://www.w3.org/2000/svg" fill="#000000" width="800px" height="800px" viewBox="0 0 32 32" id="icon"> |  | ||||||
|   <defs> |  | ||||||
|     <style>.cls-1{fill:none;}</style> |  | ||||||
|   </defs> |  | ||||||
|   <title>datastore</title> |  | ||||||
|   <circle cx="23" cy="23" r="1"/> |  | ||||||
|   <rect x="8" y="22" width="12" height="2"/> |  | ||||||
|   <circle cx="23" cy="9" r="1"/> |  | ||||||
|   <rect x="8" y="8" width="12" height="2"/> |  | ||||||
|   <path d="M26,14a2,2,0,0,0,2-2V6a2,2,0,0,0-2-2H6A2,2,0,0,0,4,6v6a2,2,0,0,0,2,2H8v4H6a2,2,0,0,0-2,2v6a2,2,0,0,0,2,2H26a2,2,0,0,0,2-2V20a2,2,0,0,0-2-2H24V14ZM6,6H26v6H6ZM26,26H6V20H26Zm-4-8H10V14H22Z"/> |  | ||||||
|   <rect id="_Transparent_Rectangle_" data-name="<Transparent Rectangle>" class="cls-1" width="32" height="32"/> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 704 B | 
| @@ -1,14 +0,0 @@ | |||||||
| import defineApp from '../../helpers/define-app.js'; |  | ||||||
| import actions from './actions/index.js'; |  | ||||||
|  |  | ||||||
| export default defineApp({ |  | ||||||
|   name: 'Datastore', |  | ||||||
|   key: 'datastore', |  | ||||||
|   iconUrl: '{BASE_URL}/apps/datastore/assets/favicon.svg', |  | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/datastore/connection', |  | ||||||
|   supportsConnections: false, |  | ||||||
|   baseUrl: '', |  | ||||||
|   apiBaseUrl: '', |  | ||||||
|   primaryColor: '001F52', |  | ||||||
|   actions, |  | ||||||
| }); |  | ||||||
| @@ -1,6 +1,5 @@ | |||||||
| import defineAction from '../../../../helpers/define-action.js'; | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
| import base64ToString from './transformers/base64-to-string.js'; |  | ||||||
| import capitalize from './transformers/capitalize.js'; | import capitalize from './transformers/capitalize.js'; | ||||||
| import extractEmailAddress from './transformers/extract-email-address.js'; | import extractEmailAddress from './transformers/extract-email-address.js'; | ||||||
| import extractNumber from './transformers/extract-number.js'; | import extractNumber from './transformers/extract-number.js'; | ||||||
| @@ -9,12 +8,10 @@ import lowercase from './transformers/lowercase.js'; | |||||||
| import markdownToHtml from './transformers/markdown-to-html.js'; | import markdownToHtml from './transformers/markdown-to-html.js'; | ||||||
| import pluralize from './transformers/pluralize.js'; | import pluralize from './transformers/pluralize.js'; | ||||||
| import replace from './transformers/replace.js'; | import replace from './transformers/replace.js'; | ||||||
| import stringToBase64 from './transformers/string-to-base64.js'; |  | ||||||
| import trimWhitespace from './transformers/trim-whitespace.js'; | import trimWhitespace from './transformers/trim-whitespace.js'; | ||||||
| import useDefaultValue from './transformers/use-default-value.js'; | import useDefaultValue from './transformers/use-default-value.js'; | ||||||
|  |  | ||||||
| const transformers = { | const transformers = { | ||||||
|   base64ToString, |  | ||||||
|   capitalize, |   capitalize, | ||||||
|   extractEmailAddress, |   extractEmailAddress, | ||||||
|   extractNumber, |   extractNumber, | ||||||
| @@ -23,7 +20,6 @@ const transformers = { | |||||||
|   markdownToHtml, |   markdownToHtml, | ||||||
|   pluralize, |   pluralize, | ||||||
|   replace, |   replace, | ||||||
|   stringToBase64, |  | ||||||
|   trimWhitespace, |   trimWhitespace, | ||||||
|   useDefaultValue, |   useDefaultValue, | ||||||
| }; | }; | ||||||
| @@ -41,7 +37,6 @@ export default defineAction({ | |||||||
|       required: true, |       required: true, | ||||||
|       variables: true, |       variables: true, | ||||||
|       options: [ |       options: [ | ||||||
|         { label: 'Base64 to String', value: 'base64ToString' }, |  | ||||||
|         { label: 'Capitalize', value: 'capitalize' }, |         { label: 'Capitalize', value: 'capitalize' }, | ||||||
|         { label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' }, |         { label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' }, | ||||||
|         { label: 'Convert Markdown to HTML', value: 'markdownToHtml' }, |         { label: 'Convert Markdown to HTML', value: 'markdownToHtml' }, | ||||||
| @@ -50,7 +45,6 @@ export default defineAction({ | |||||||
|         { label: 'Lowercase', value: 'lowercase' }, |         { label: 'Lowercase', value: 'lowercase' }, | ||||||
|         { label: 'Pluralize', value: 'pluralize' }, |         { label: 'Pluralize', value: 'pluralize' }, | ||||||
|         { label: 'Replace', value: 'replace' }, |         { label: 'Replace', value: 'replace' }, | ||||||
|         { label: 'String to Base64', value: 'stringToBase64' }, |  | ||||||
|         { label: 'Trim Whitespace', value: 'trimWhitespace' }, |         { label: 'Trim Whitespace', value: 'trimWhitespace' }, | ||||||
|         { label: 'Use Default Value', value: 'useDefaultValue' }, |         { label: 'Use Default Value', value: 'useDefaultValue' }, | ||||||
|       ], |       ], | ||||||
|   | |||||||
| @@ -1,8 +0,0 @@ | |||||||
| const base64ToString = ($) => { |  | ||||||
|   const input = $.step.parameters.input; |  | ||||||
|   const decodedString = Buffer.from(input, 'base64').toString('utf8'); |  | ||||||
|  |  | ||||||
|   return decodedString; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default base64ToString; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| const stringtoBase64 = ($) => { |  | ||||||
|   const input = $.step.parameters.input; |  | ||||||
|   const base64String = Buffer.from(input).toString('base64'); |  | ||||||
|  |  | ||||||
|   return base64String; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default stringtoBase64; |  | ||||||
| @@ -1,4 +1,3 @@ | |||||||
| import base64ToString from './text/base64-to-string.js'; |  | ||||||
| import capitalize from './text/capitalize.js'; | import capitalize from './text/capitalize.js'; | ||||||
| import extractEmailAddress from './text/extract-email-address.js'; | import extractEmailAddress from './text/extract-email-address.js'; | ||||||
| import extractNumber from './text/extract-number.js'; | import extractNumber from './text/extract-number.js'; | ||||||
| @@ -7,7 +6,6 @@ import lowercase from './text/lowercase.js'; | |||||||
| import markdownToHtml from './text/markdown-to-html.js'; | import markdownToHtml from './text/markdown-to-html.js'; | ||||||
| import pluralize from './text/pluralize.js'; | import pluralize from './text/pluralize.js'; | ||||||
| import replace from './text/replace.js'; | import replace from './text/replace.js'; | ||||||
| import stringToBase64 from './text/string-to-base64.js'; |  | ||||||
| import trimWhitespace from './text/trim-whitespace.js'; | import trimWhitespace from './text/trim-whitespace.js'; | ||||||
| import useDefaultValue from './text/use-default-value.js'; | import useDefaultValue from './text/use-default-value.js'; | ||||||
| import performMathOperation from './numbers/perform-math-operation.js'; | import performMathOperation from './numbers/perform-math-operation.js'; | ||||||
| @@ -17,7 +15,6 @@ import formatPhoneNumber from './numbers/format-phone-number.js'; | |||||||
| import formatDateTime from './date-time/format-date-time.js'; | import formatDateTime from './date-time/format-date-time.js'; | ||||||
|  |  | ||||||
| const options = { | const options = { | ||||||
|   base64ToString, |  | ||||||
|   capitalize, |   capitalize, | ||||||
|   extractEmailAddress, |   extractEmailAddress, | ||||||
|   extractNumber, |   extractNumber, | ||||||
| @@ -26,7 +23,6 @@ const options = { | |||||||
|   markdownToHtml, |   markdownToHtml, | ||||||
|   pluralize, |   pluralize, | ||||||
|   replace, |   replace, | ||||||
|   stringToBase64, |  | ||||||
|   trimWhitespace, |   trimWhitespace, | ||||||
|   useDefaultValue, |   useDefaultValue, | ||||||
|   performMathOperation, |   performMathOperation, | ||||||
|   | |||||||
| @@ -1,12 +0,0 @@ | |||||||
| const base64ToString = [ |  | ||||||
|   { |  | ||||||
|     label: 'Input', |  | ||||||
|     key: 'input', |  | ||||||
|     type: 'string', |  | ||||||
|     required: true, |  | ||||||
|     description: 'Text that will be converted from Base64 to string.', |  | ||||||
|     variables: true, |  | ||||||
|   }, |  | ||||||
| ]; |  | ||||||
|  |  | ||||||
| export default base64ToString; |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| const stringToBase64 = [ |  | ||||||
|   { |  | ||||||
|     label: 'Input', |  | ||||||
|     key: 'input', |  | ||||||
|     type: 'string', |  | ||||||
|     required: true, |  | ||||||
|     description: 'Text that will be converted to Base64.', |  | ||||||
|     variables: true, |  | ||||||
|   }, |  | ||||||
| ]; |  | ||||||
|  |  | ||||||
| export default stringToBase64; |  | ||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import FormData from 'form-data'; | ||||||
| import defineAction from '../../../../helpers/define-action.js'; | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
| export default defineAction({ | export default defineAction({ | ||||||
| @@ -5,51 +6,45 @@ export default defineAction({ | |||||||
|   key: 'newChat', |   key: 'newChat', | ||||||
|   description: 'Create a new chat session for Helix AI.', |   description: 'Create a new chat session for Helix AI.', | ||||||
|   arguments: [ |   arguments: [ | ||||||
|     { |  | ||||||
|       label: 'Session ID', |  | ||||||
|       key: 'sessionId', |  | ||||||
|       type: 'string', |  | ||||||
|       required: false, |  | ||||||
|       description: |  | ||||||
|         'ID of the chat session to continue. Leave empty to start a new chat.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       label: 'System Prompt', |  | ||||||
|       key: 'systemPrompt', |  | ||||||
|       type: 'string', |  | ||||||
|       required: false, |  | ||||||
|       description: |  | ||||||
|         'Optional system prompt to start the chat with. It will be used only for new chat sessions.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|     { |     { | ||||||
|       label: 'Input', |       label: 'Input', | ||||||
|       key: 'input', |       key: 'input', | ||||||
|       type: 'string', |       type: 'string', | ||||||
|       required: true, |       required: true, | ||||||
|       description: 'User input to start the chat with.', |       description: 'Prompt to start the chat with.', | ||||||
|       variables: true, |       variables: true, | ||||||
|     }, |     }, | ||||||
|   ], |   ], | ||||||
|  |  | ||||||
|   async run($) { |   async run($) { | ||||||
|     const response = await $.http.post('/api/v1/sessions/chat', { |     const formData = new FormData(); | ||||||
|       session_id: $.step.parameters.sessionId, |     formData.append('input', $.step.parameters.input); | ||||||
|       system: $.step.parameters.systemPrompt, |     formData.append('mode', 'inference'); | ||||||
|       messages: [ |     formData.append('type', 'text'); | ||||||
|         { |  | ||||||
|           role: 'user', |     const sessionResponse = await $.http.post('/api/v1/sessions', formData, { | ||||||
|           content: { |       headers: { | ||||||
|             content_type: 'text', |         ...formData.getHeaders(), | ||||||
|             parts: [$.step.parameters.input], |  | ||||||
|       }, |       }, | ||||||
|         }, |  | ||||||
|       ], |  | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     const sessionId = sessionResponse.data.id; | ||||||
|  |  | ||||||
|  |     let chatGenerated = false; | ||||||
|  |  | ||||||
|  |     while (!chatGenerated) { | ||||||
|  |       const response = await $.http.get(`/api/v1/sessions/${sessionId}`); | ||||||
|  |  | ||||||
|  |       const message = | ||||||
|  |         response.data.interactions[response.data.interactions.length - 1]; | ||||||
|  |  | ||||||
|  |       if (message.creator === 'system' && message.state === 'complete') { | ||||||
|         $.setActionItem({ |         $.setActionItem({ | ||||||
|       raw: response.data, |           raw: message, | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|  |         chatGenerated = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|   }, |   }, | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -18,9 +18,7 @@ const port = process.env.PORT || '3000'; | |||||||
| const serveWebAppSeparately = | const serveWebAppSeparately = | ||||||
|   process.env.SERVE_WEB_APP_SEPARATELY === 'true' ? true : false; |   process.env.SERVE_WEB_APP_SEPARATELY === 'true' ? true : false; | ||||||
|  |  | ||||||
| let apiUrl = new URL( | let apiUrl = new URL(`${protocol}://${host}:${port}`).toString(); | ||||||
|   process.env.API_URL || `${protocol}://${host}:${port}` |  | ||||||
| ).toString(); |  | ||||||
| apiUrl = apiUrl.substring(0, apiUrl.length - 1); | apiUrl = apiUrl.substring(0, apiUrl.length - 1); | ||||||
|  |  | ||||||
| // use apiUrl by default, which has less priority over the following cases | // use apiUrl by default, which has less priority over the following cases | ||||||
| @@ -90,10 +88,6 @@ const appConfig = { | |||||||
|   licenseKey: process.env.LICENSE_KEY, |   licenseKey: process.env.LICENSE_KEY, | ||||||
|   sentryDsn: process.env.SENTRY_DSN, |   sentryDsn: process.env.SENTRY_DSN, | ||||||
|   CI: process.env.CI === 'true', |   CI: process.env.CI === 'true', | ||||||
|   disableNotificationsPage: process.env.DISABLE_NOTIFICATIONS_PAGE === 'true', |  | ||||||
|   disableFavicon: process.env.DISABLE_FAVICON === 'true', |  | ||||||
|   additionalDrawerLink: process.env.ADDITIONAL_DRAWER_LINK, |  | ||||||
|   additionalDrawerLinkText: process.env.ADDITIONAL_DRAWER_LINK_TEXT, |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| if (!appConfig.encryptionKey) { | if (!appConfig.encryptionKey) { | ||||||
|   | |||||||
| @@ -1,10 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import AppAuthClient from '../../../../../models/app-auth-client.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const appAuthClient = await AppAuthClient.query() |  | ||||||
|     .findById(request.params.appAuthClientId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   renderObject(response, appAuthClient); |  | ||||||
| }; |  | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user.js'; |  | ||||||
| import getAdminAppAuthClientMock from '../../../../../../test/mocks/rest/api/v1/admin/get-app-auth-client.js'; |  | ||||||
| import { createAppAuthClient } from '../../../../../../test/factories/app-auth-client.js'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/app-auth-clients/:appAuthClientId', () => { |  | ||||||
|   let currentUser, currentUserRole, currentAppAuthClient, token; |  | ||||||
|  |  | ||||||
|   describe('with valid license key', () => { |  | ||||||
|     beforeEach(async () => { |  | ||||||
|       vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|       currentUserRole = await createRole({ key: 'admin' }); |  | ||||||
|       currentUser = await createUser({ roleId: currentUserRole.id }); |  | ||||||
|       currentAppAuthClient = await createAppAuthClient(); |  | ||||||
|  |  | ||||||
|       token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     it('should return specified app auth client info', async () => { |  | ||||||
|       const response = await request(app) |  | ||||||
|         .get(`/api/v1/admin/app-auth-clients/${currentAppAuthClient.id}`) |  | ||||||
|         .set('Authorization', token) |  | ||||||
|         .expect(200); |  | ||||||
|  |  | ||||||
|       const expectedPayload = getAdminAppAuthClientMock(currentAppAuthClient); |  | ||||||
|       expect(response.body).toEqual(expectedPayload); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     it('should return not found response for not existing app auth client UUID', async () => { |  | ||||||
|       const notExistingAppAuthClientUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|       await request(app) |  | ||||||
|         .get(`/api/v1/admin/app-auth-clients/${notExistingAppAuthClientUUID}`) |  | ||||||
|         .set('Authorization', token) |  | ||||||
|         .expect(404); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     it('should return bad request response for invalid UUID', async () => { |  | ||||||
|       await request(app) |  | ||||||
|         .get('/api/v1/admin/app-auth-clients/invalidAppAuthClientUUID') |  | ||||||
|         .set('Authorization', token) |  | ||||||
|         .expect(400); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import permissionCatalog from '../../../../../helpers/permission-catalog.ee.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   renderObject(response, permissionCatalog); |  | ||||||
| }; |  | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role.js'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user.js'; |  | ||||||
| import getPermissionsCatalogMock from '../../../../../../test/mocks/rest/api/v1/admin/permissions/get-permissions-catalog.ee.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/permissions/catalog', () => { |  | ||||||
|   let role, currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     role = await createRole({ key: 'admin' }); |  | ||||||
|     currentUser = await createUser({ roleId: role.id }); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return roles', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/admin/permissions/catalog') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getPermissionsCatalogMock(); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import Role from '../../../../../models/role.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const role = await Role.query() |  | ||||||
|     .leftJoinRelated({ |  | ||||||
|       permissions: true, |  | ||||||
|     }) |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       permissions: true, |  | ||||||
|     }) |  | ||||||
|     .findById(request.params.roleId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   renderObject(response, role); |  | ||||||
| }; |  | ||||||
| @@ -1,59 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role.js'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user.js'; |  | ||||||
| import { createPermission } from '../../../../../../test/factories/permission.js'; |  | ||||||
| import getRoleMock from '../../../../../../test/mocks/rest/api/v1/admin/roles/get-role.ee.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/roles/:roleId', () => { |  | ||||||
|   let role, currentUser, token, permissionOne, permissionTwo; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     role = await createRole({ key: 'admin' }); |  | ||||||
|     permissionOne = await createPermission({ roleId: role.id }); |  | ||||||
|     permissionTwo = await createPermission({ roleId: role.id }); |  | ||||||
|     currentUser = await createUser({ roleId: role.id }); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return role', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/admin/roles/${role.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getRoleMock(role, [ |  | ||||||
|       permissionOne, |  | ||||||
|       permissionTwo, |  | ||||||
|     ]); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for not existing role UUID', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const notExistingRoleUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get(`/api/v1/admin/roles/${notExistingRoleUUID}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return bad request response for invalid UUID', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/admin/roles/invalidRoleUUID') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(400); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import Role from '../../../../../models/role.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const roles = await Role.query().orderBy('name'); |  | ||||||
|  |  | ||||||
|   renderObject(response, roles); |  | ||||||
| }; |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role.js'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user.js'; |  | ||||||
| import getRolesMock from '../../../../../../test/mocks/rest/api/v1/admin/roles/get-roles.ee.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/roles', () => { |  | ||||||
|   let roleOne, roleTwo, currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     roleOne = await createRole({ key: 'admin' }); |  | ||||||
|     roleTwo = await createRole({ key: 'user' }); |  | ||||||
|     currentUser = await createUser({ roleId: roleOne.id }); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return roles', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/admin/roles') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getRolesMock([roleOne, roleTwo]); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import SamlAuthProvider from '../../../../../models/saml-auth-provider.ee.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const samlAuthProvider = await SamlAuthProvider.query() |  | ||||||
|     .findById(request.params.samlAuthProviderId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   renderObject(response, samlAuthProvider); |  | ||||||
| }; |  | ||||||
| @@ -1,57 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role.js'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user.js'; |  | ||||||
| import { createSamlAuthProvider } from '../../../../../../test/factories/saml-auth-provider.ee.js'; |  | ||||||
| import getSamlAuthProviderMock from '../../../../../../test/mocks/rest/api/v1/admin/saml-auth-providers/get-saml-auth-provider.ee.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/saml-auth-provider/:samlAuthProviderId', () => { |  | ||||||
|   let samlAuthProvider, currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     const role = await createRole({ key: 'admin' }); |  | ||||||
|     currentUser = await createUser({ roleId: role.id }); |  | ||||||
|     samlAuthProvider = await createSamlAuthProvider(); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return saml auth provider with specified id', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/admin/saml-auth-providers/${samlAuthProvider.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getSamlAuthProviderMock(samlAuthProvider); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for not existing saml auth provider UUID', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const notExistingSamlAuthProviderUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get( |  | ||||||
|         `/api/v1/admin/saml-auth-providers/${notExistingSamlAuthProviderUUID}` |  | ||||||
|       ) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return bad request response for invalid UUID', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/admin/saml-auth-providers/invalidSamlAuthProviderUUID') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(400); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import SamlAuthProvider from '../../../../../models/saml-auth-provider.ee.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const samlAuthProviders = await SamlAuthProvider.query().orderBy( |  | ||||||
|     'created_at', |  | ||||||
|     'desc' |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   renderObject(response, samlAuthProviders); |  | ||||||
| }; |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role.js'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user.js'; |  | ||||||
| import { createSamlAuthProvider } from '../../../../../../test/factories/saml-auth-provider.ee.js'; |  | ||||||
| import getSamlAuthProvidersMock from '../../../../../../test/mocks/rest/api/v1/admin/saml-auth-providers/get-saml-auth-providers.ee.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/saml-auth-providers', () => { |  | ||||||
|   let samlAuthProviderOne, samlAuthProviderTwo, currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     const role = await createRole({ key: 'admin' }); |  | ||||||
|     currentUser = await createUser({ roleId: role.id }); |  | ||||||
|  |  | ||||||
|     samlAuthProviderOne = await createSamlAuthProvider(); |  | ||||||
|     samlAuthProviderTwo = await createSamlAuthProvider(); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return saml auth providers', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/admin/saml-auth-providers') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getSamlAuthProvidersMock([ |  | ||||||
|       samlAuthProviderTwo, |  | ||||||
|       samlAuthProviderOne, |  | ||||||
|     ]); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import User from '../../../../../models/user.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const user = await User.query() |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       role: true, |  | ||||||
|     }) |  | ||||||
|     .findById(request.params.userId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   renderObject(response, user); |  | ||||||
| }; |  | ||||||
| @@ -1,55 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role'; |  | ||||||
| import getUserMock from '../../../../../../test/mocks/rest/api/v1/admin/users/get-user.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/users/:userId', () => { |  | ||||||
|   let currentUser, currentUserRole, anotherUser, anotherUserRole, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUserRole = await createRole({ key: 'admin' }); |  | ||||||
|     currentUser = await createUser({ roleId: currentUserRole.id }); |  | ||||||
|  |  | ||||||
|     anotherUser = await createUser(); |  | ||||||
|     anotherUserRole = await anotherUser.$relatedQuery('role'); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return specified user info', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/admin/users/${anotherUser.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getUserMock(anotherUser, anotherUserRole); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for not existing user UUID', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const notExistingUserUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get(`/api/v1/admin/users/${notExistingUserUUID}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return bad request response for invalid UUID', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/admin/users/invalidUserUUID') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(400); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| import { renderObject } from '../../../../../helpers/renderer.js'; |  | ||||||
| import User from '../../../../../models/user.js'; |  | ||||||
| import paginateRest from '../../../../../helpers/pagination-rest.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const usersQuery = User.query() |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       role: true, |  | ||||||
|     }) |  | ||||||
|     .orderBy('full_name', 'asc'); |  | ||||||
|  |  | ||||||
|   const users = await paginateRest(usersQuery, request.query.page); |  | ||||||
|  |  | ||||||
|   renderObject(response, users); |  | ||||||
| }; |  | ||||||
| @@ -1,49 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../../app'; |  | ||||||
| import createAuthTokenByUserId from '../../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createRole } from '../../../../../../test/factories/role'; |  | ||||||
| import { createUser } from '../../../../../../test/factories/user'; |  | ||||||
| import getUsersMock from '../../../../../../test/mocks/rest/api/v1/admin/users/get-users.js'; |  | ||||||
| import * as license from '../../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/admin/users', () => { |  | ||||||
|   let currentUser, currentUserRole, anotherUser, anotherUserRole, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUserRole = await createRole({ key: 'admin' }); |  | ||||||
|  |  | ||||||
|     currentUser = await createUser({ |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       fullName: 'Current User', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     anotherUserRole = await createRole({ |  | ||||||
|       key: 'anotherUser', |  | ||||||
|       name: 'Another user role', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     anotherUser = await createUser({ |  | ||||||
|       roleId: anotherUserRole.id, |  | ||||||
|       fullName: 'Another User', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return users data', async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/admin/users') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedResponsePayload = await getUsersMock( |  | ||||||
|       [anotherUser, currentUser], |  | ||||||
|       [anotherUserRole, currentUserRole] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
| import AppAuthClient from '../../../../models/app-auth-client.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const appAuthClient = await AppAuthClient.query() |  | ||||||
|     .findById(request.params.appAuthClientId) |  | ||||||
|     .where({ active: true }) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   renderObject(response, appAuthClient); |  | ||||||
| }; |  | ||||||
| @@ -1,48 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user.js'; |  | ||||||
| import getAppAuthClientMock from '../../../../../test/mocks/rest/api/v1/admin/get-app-auth-client.js'; |  | ||||||
| import { createAppAuthClient } from '../../../../../test/factories/app-auth-client.js'; |  | ||||||
| import * as license from '../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/app-auth-clients/:id', () => { |  | ||||||
|   let currentUser, currentAppAuthClient, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     currentAppAuthClient = await createAppAuthClient(); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return specified app auth client info', async () => { |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/app-auth-clients/${currentAppAuthClient.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getAppAuthClientMock(currentAppAuthClient); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for not existing app auth client ID', async () => { |  | ||||||
|     const notExistingAppAuthClientUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get(`/api/v1/app-auth-clients/${notExistingAppAuthClientUUID}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return bad request response for invalid UUID', async () => { |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/app-auth-clients/invalidAppAuthClientUUID') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(400); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| import App from '../../../../models/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const substeps = await App.findActionSubsteps( |  | ||||||
|     request.params.appKey, |  | ||||||
|     request.params.actionKey |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   renderObject(response, substeps); |  | ||||||
| }; |  | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import App from '../../../../models/app'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getActionSubstepsMock from '../../../../../test/mocks/rest/api/v1/apps/get-action-substeps.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/apps/:appKey/actions/:actionKey/substeps', () => { |  | ||||||
|   let currentUser, exampleApp, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|     exampleApp = await App.findOneByKey('github'); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the app auth info', async () => { |  | ||||||
|     const actions = await App.findActionsByKey('github'); |  | ||||||
|     const exampleAction = actions.find( |  | ||||||
|       (action) => action.key === 'createIssue' |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     const endpointUrl = `/api/v1/apps/${exampleApp.key}/actions/${exampleAction.key}/substeps`; |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(endpointUrl) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getActionSubstepsMock(exampleAction.substeps); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for invalid app key', async () => { |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/apps/invalid-app-key/actions/invalid-actions-key/substeps') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return empty array for invalid action key', async () => { |  | ||||||
|     const endpointUrl = `/api/v1/apps/${exampleApp.key}/actions/invalid-action-key/substeps`; |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(endpointUrl) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     expect(response.body.data).toEqual([]); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import App from '../../../../models/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const actions = await App.findActionsByKey(request.params.appKey); |  | ||||||
|  |  | ||||||
|   renderObject(response, actions, { serializer: 'Action' }); |  | ||||||
| }; |  | ||||||
| @@ -1,35 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import App from '../../../../models/app'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getActionsMock from '../../../../../test/mocks/rest/api/v1/apps/get-actions.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/apps/:appKey/actions', () => { |  | ||||||
|   let currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the app actions', async () => { |  | ||||||
|     const exampleApp = await App.findOneByKey('github'); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/apps/${exampleApp.key}/actions`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getActionsMock(exampleApp.actions); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for invalid app key', async () => { |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/apps/invalid-app-key/actions') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import App from '../../../../models/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const app = await App.findOneByKey(request.params.appKey); |  | ||||||
|  |  | ||||||
|   renderObject(response, app, { serializer: 'App' }); |  | ||||||
| }; |  | ||||||
| @@ -1,35 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import App from '../../../../models/app'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getAppMock from '../../../../../test/mocks/rest/api/v1/apps/get-app.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/apps/:appKey', () => { |  | ||||||
|   let currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the app info', async () => { |  | ||||||
|     const exampleApp = await App.findOneByKey('github'); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/apps/${exampleApp.key}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getAppMock(exampleApp); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for invalid app key', async () => { |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/apps/invalid-app-key') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| import App from '../../../../models/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   let apps = await App.findAll(request.query.name); |  | ||||||
|  |  | ||||||
|   if (request.query.onlyWithTriggers) { |  | ||||||
|     apps = apps.filter((app) => app.triggers?.length); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (request.query.onlyWithActions) { |  | ||||||
|     apps = apps.filter((app) => app.actions?.length); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   renderObject(response, apps, { serializer: 'App' }); |  | ||||||
| }; |  | ||||||
| @@ -1,63 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import App from '../../../../models/app'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getAppsMock from '../../../../../test/mocks/rest/api/v1/apps/get-apps.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/apps', () => { |  | ||||||
|   let currentUser, apps, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|     apps = await App.findAll(); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return all apps', async () => { |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/apps') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getAppsMock(apps); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return all apps filtered by name', async () => { |  | ||||||
|     const appsWithNameGit = apps.filter((app) => app.name.includes('Git')); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/apps?name=Git') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getAppsMock(appsWithNameGit); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return only the apps with triggers', async () => { |  | ||||||
|     const appsWithTriggers = apps.filter((app) => app.triggers?.length > 0); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/apps?onlyWithTriggers=true') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getAppsMock(appsWithTriggers); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return only the apps with actions', async () => { |  | ||||||
|     const appsWithActions = apps.filter((app) => app.actions?.length > 0); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/apps?onlyWithActions=true') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getAppsMock(appsWithActions); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import App from '../../../../models/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const auth = await App.findAuthByKey(request.params.appKey); |  | ||||||
|  |  | ||||||
|   renderObject(response, auth, { serializer: 'Auth' }); |  | ||||||
| }; |  | ||||||
| @@ -1,35 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import App from '../../../../models/app'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getAuthMock from '../../../../../test/mocks/rest/api/v1/apps/get-auth.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/apps/:appKey/auth', () => { |  | ||||||
|   let currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the app auth info', async () => { |  | ||||||
|     const exampleApp = await App.findOneByKey('github'); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/apps/${exampleApp.key}/auth`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getAuthMock(exampleApp.auth); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for invalid app key', async () => { |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/apps/invalid-app-key/auth') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| import App from '../../../../models/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const substeps = await App.findTriggerSubsteps( |  | ||||||
|     request.params.appKey, |  | ||||||
|     request.params.triggerKey |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   renderObject(response, substeps); |  | ||||||
| }; |  | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import App from '../../../../models/app'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getTriggerSubstepsMock from '../../../../../test/mocks/rest/api/v1/apps/get-trigger-substeps.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/apps/:appKey/triggers/:triggerKey/substeps', () => { |  | ||||||
|   let currentUser, exampleApp, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|     exampleApp = await App.findOneByKey('github'); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the app auth info', async () => { |  | ||||||
|     const triggers = await App.findTriggersByKey('github'); |  | ||||||
|     const exampleTrigger = triggers.find( |  | ||||||
|       (trigger) => trigger.key === 'newIssues' |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     const endpointUrl = `/api/v1/apps/${exampleApp.key}/triggers/${exampleTrigger.key}/substeps`; |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(endpointUrl) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getTriggerSubstepsMock(exampleTrigger.substeps); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for invalid app key', async () => { |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/apps/invalid-app-key/triggers/invalid-trigger-key/substeps') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return empty array for invalid trigger key', async () => { |  | ||||||
|     const endpointUrl = `/api/v1/apps/${exampleApp.key}/triggers/invalid-trigger-key/substeps`; |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(endpointUrl) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     expect(response.body.data).toEqual([]); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import App from '../../../../models/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const triggers = await App.findTriggersByKey(request.params.appKey); |  | ||||||
|  |  | ||||||
|   renderObject(response, triggers, { serializer: 'Trigger' }); |  | ||||||
| }; |  | ||||||
| @@ -1,35 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import App from '../../../../models/app'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getTriggersMock from '../../../../../test/mocks/rest/api/v1/apps/get-triggers.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/apps/:appKey/triggers', () => { |  | ||||||
|   let currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the app triggers', async () => { |  | ||||||
|     const exampleApp = await App.findOneByKey('github'); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/apps/${exampleApp.key}/triggers`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getTriggersMock(exampleApp.triggers); |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for invalid app key', async () => { |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/apps/invalid-app-key/triggers') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| import appConfig from '../../../../config/app.js'; |  | ||||||
| import { hasValidLicense } from '../../../../helpers/license.ee.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const info = { |  | ||||||
|     isCloud: appConfig.isCloud, |  | ||||||
|     isMation: appConfig.isMation, |  | ||||||
|     isEnterprise: await hasValidLicense(), |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   renderObject(response, info); |  | ||||||
| }; |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| import { vi, expect, describe, it } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import appConfig from '../../../../config/app.js'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import infoMock from '../../../../../test/mocks/rest/api/v1/automatisch/info.js'; |  | ||||||
| import * as license from '../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/automatisch/info', () => { |  | ||||||
|   it('should return Automatisch info', async () => { |  | ||||||
|     vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(false); |  | ||||||
|     vi.spyOn(appConfig, 'isMation', 'get').mockReturnValue(false); |  | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/automatisch/info') |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = infoMock(); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| import { getLicense } from '../../../../helpers/license.ee.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const license = await getLicense(); |  | ||||||
|  |  | ||||||
|   const computedLicense = { |  | ||||||
|     id: license ? license.id : null, |  | ||||||
|     name: license ? license.name : null, |  | ||||||
|     expireAt: license ? license.expireAt : null, |  | ||||||
|     verified: license ? true : false, |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   renderObject(response, computedLicense); |  | ||||||
| }; |  | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| import { vi, expect, describe, it } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import licenseMock from '../../../../../test/mocks/rest/api/v1/automatisch/license.js'; |  | ||||||
| import * as license from '../../../../helpers/license.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/automatisch/license', () => { |  | ||||||
|   it('should return Automatisch license info', async () => { |  | ||||||
|     vi.spyOn(license, 'getLicense').mockResolvedValue({ |  | ||||||
|       id: '123', |  | ||||||
|       name: 'license-name', |  | ||||||
|       expireAt: '2025-12-31T23:59:59Z', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/automatisch/license') |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = licenseMock(); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
| import axios from '../../../../helpers/axios-with-proxy.js'; |  | ||||||
| import logger from '../../../../helpers/logger.js'; |  | ||||||
|  |  | ||||||
| const NOTIFICATIONS_URL = |  | ||||||
|   'https://notifications.automatisch.io/notifications.json'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   let notifications = []; |  | ||||||
|  |  | ||||||
|   try { |  | ||||||
|     const response = await axios.get(NOTIFICATIONS_URL); |  | ||||||
|     notifications = response.data; |  | ||||||
|   } catch (error) { |  | ||||||
|     logger.error('Error fetching notifications API endpoint!', error); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   renderObject(response, notifications); |  | ||||||
| }; |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| import { describe, it } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/automatisch/notifications', () => { |  | ||||||
|   it('should return Automatisch notifications', async () => { |  | ||||||
|     await request(app).get('/api/v1/automatisch/notifications').expect(200); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| import appConfig from '../../../../config/app.js'; |  | ||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   renderObject(response, { version: appConfig.version }); |  | ||||||
| }; |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| import { describe, it, expect } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/automatisch/version', () => { |  | ||||||
|   it('should return Automatisch version', async () => { |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/automatisch/version') |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = { |  | ||||||
|       data: { |  | ||||||
|         version: '0.10.0', |  | ||||||
|       }, |  | ||||||
|       meta: { |  | ||||||
|         count: 1, |  | ||||||
|         currentPage: null, |  | ||||||
|         isArray: false, |  | ||||||
|         totalPages: null, |  | ||||||
|         type: 'Object', |  | ||||||
|       }, |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
| import paginateRest from '../../../../helpers/pagination-rest.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const execution = await request.currentUser.authorizedExecutions |  | ||||||
|     .clone() |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .findById(request.params.executionId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   const executionStepsQuery = execution |  | ||||||
|     .$relatedQuery('executionSteps') |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .withGraphFetched('step') |  | ||||||
|     .orderBy('created_at', 'asc'); |  | ||||||
|  |  | ||||||
|   const executionSteps = await paginateRest( |  | ||||||
|     executionStepsQuery, |  | ||||||
|     request.query.page |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   renderObject(response, executionSteps); |  | ||||||
| }; |  | ||||||
| @@ -1,153 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import { createFlow } from '../../../../../test/factories/flow.js'; |  | ||||||
| import { createStep } from '../../../../../test/factories/step.js'; |  | ||||||
| import { createExecution } from '../../../../../test/factories/execution.js'; |  | ||||||
| import { createExecutionStep } from '../../../../../test/factories/execution-step.js'; |  | ||||||
| import { createPermission } from '../../../../../test/factories/permission'; |  | ||||||
| import getExecutionStepsMock from '../../../../../test/mocks/rest/api/v1/executions/get-execution-steps'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/executions/:executionId/execution-steps', () => { |  | ||||||
|   let currentUser, currentUserRole, anotherUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     currentUserRole = await currentUser.$relatedQuery('role'); |  | ||||||
|  |  | ||||||
|     anotherUser = await createUser(); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the execution steps of current user execution', async () => { |  | ||||||
|     const currentUserFlow = await createFlow({ |  | ||||||
|       userId: currentUser.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepOne = await createStep({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|       type: 'trigger', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepTwo = await createStep({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|       type: 'action', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const currentUserExecution = await createExecution({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const currentUserExecutionStepOne = await createExecutionStep({ |  | ||||||
|       executionId: currentUserExecution.id, |  | ||||||
|       stepId: stepOne.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const currentUserExecutionStepTwo = await createExecutionStep({ |  | ||||||
|       executionId: currentUserExecution.id, |  | ||||||
|       stepId: stepTwo.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: ['isCreator'], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/executions/${currentUserExecution.id}/execution-steps`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getExecutionStepsMock( |  | ||||||
|       [currentUserExecutionStepOne, currentUserExecutionStepTwo], |  | ||||||
|       [stepOne, stepTwo] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the execution steps of another user execution', async () => { |  | ||||||
|     const anotherUserFlow = await createFlow({ |  | ||||||
|       userId: anotherUser.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepOne = await createStep({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|       type: 'trigger', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepTwo = await createStep({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|       type: 'action', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const anotherUserExecution = await createExecution({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const anotherUserExecutionStepOne = await createExecutionStep({ |  | ||||||
|       executionId: anotherUserExecution.id, |  | ||||||
|       stepId: stepOne.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const anotherUserExecutionStepTwo = await createExecutionStep({ |  | ||||||
|       executionId: anotherUserExecution.id, |  | ||||||
|       stepId: stepTwo.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/executions/${anotherUserExecution.id}/execution-steps`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getExecutionStepsMock( |  | ||||||
|       [anotherUserExecutionStepOne, anotherUserExecutionStepTwo], |  | ||||||
|       [stepOne, stepTwo] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for not existing execution step UUID', async () => { |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const notExistingExcecutionUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get(`/api/v1/executions/${notExistingExcecutionUUID}/execution-steps`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return bad request response for invalid UUID', async () => { |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/executions/invalidExecutionUUID/execution-steps') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(400); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const execution = await request.currentUser.authorizedExecutions |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       flow: { |  | ||||||
|         steps: true, |  | ||||||
|       }, |  | ||||||
|     }) |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .findById(request.params.executionId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   renderObject(response, execution); |  | ||||||
| }; |  | ||||||
| @@ -1,134 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import { createFlow } from '../../../../../test/factories/flow.js'; |  | ||||||
| import { createStep } from '../../../../../test/factories/step.js'; |  | ||||||
| import { createExecution } from '../../../../../test/factories/execution.js'; |  | ||||||
| import { createPermission } from '../../../../../test/factories/permission'; |  | ||||||
| import getExecutionMock from '../../../../../test/mocks/rest/api/v1/executions/get-execution'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/executions/:executionId', () => { |  | ||||||
|   let currentUser, currentUserRole, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     currentUserRole = await currentUser.$relatedQuery('role'); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the execution data of current user', async () => { |  | ||||||
|     const currentUserFlow = await createFlow({ |  | ||||||
|       userId: currentUser.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepOne = await createStep({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|       type: 'trigger', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepTwo = await createStep({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|       type: 'action', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const currentUserExecution = await createExecution({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: ['isCreator'], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/executions/${currentUserExecution.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getExecutionMock( |  | ||||||
|       currentUserExecution, |  | ||||||
|       currentUserFlow, |  | ||||||
|       [stepOne, stepTwo] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the execution data of another user', async () => { |  | ||||||
|     const anotherUser = await createUser(); |  | ||||||
|  |  | ||||||
|     const anotherUserFlow = await createFlow({ |  | ||||||
|       userId: anotherUser.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepOne = await createStep({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|       type: 'trigger', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepTwo = await createStep({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|       type: 'action', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const anotherUserExecution = await createExecution({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/executions/${anotherUserExecution.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getExecutionMock( |  | ||||||
|       anotherUserExecution, |  | ||||||
|       anotherUserFlow, |  | ||||||
|       [stepOne, stepTwo] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for not existing execution UUID', async () => { |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const notExistingExcecutionUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get(`/api/v1/executions/${notExistingExcecutionUUID}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return bad request response for invalid UUID', async () => { |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/executions/invalidExecutionUUID') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(400); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
| import paginateRest from '../../../../helpers/pagination-rest.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const executionsQuery = request.currentUser.authorizedExecutions |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .orderBy('created_at', 'desc') |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       flow: { |  | ||||||
|         steps: true, |  | ||||||
|       }, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|   const executions = await paginateRest(executionsQuery, request.query.page); |  | ||||||
|  |  | ||||||
|   for (const execution of executions.records) { |  | ||||||
|     const executionSteps = await execution.$relatedQuery('executionSteps'); |  | ||||||
|     const status = executionSteps.some((step) => step.status === 'failure') |  | ||||||
|       ? 'failure' |  | ||||||
|       : 'success'; |  | ||||||
|  |  | ||||||
|     execution.status = status; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   renderObject(response, executions); |  | ||||||
| }; |  | ||||||
| @@ -1,113 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import { createFlow } from '../../../../../test/factories/flow.js'; |  | ||||||
| import { createStep } from '../../../../../test/factories/step.js'; |  | ||||||
| import { createExecution } from '../../../../../test/factories/execution.js'; |  | ||||||
| import { createPermission } from '../../../../../test/factories/permission'; |  | ||||||
| import getExecutionsMock from '../../../../../test/mocks/rest/api/v1/executions/get-executions'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/executions', () => { |  | ||||||
|   let currentUser, currentUserRole, anotherUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     currentUserRole = await currentUser.$relatedQuery('role'); |  | ||||||
|  |  | ||||||
|     anotherUser = await createUser(); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the executions of current user', async () => { |  | ||||||
|     const currentUserFlow = await createFlow({ |  | ||||||
|       userId: currentUser.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepOne = await createStep({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|       type: 'trigger', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepTwo = await createStep({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|       type: 'action', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const currentUserExecutionOne = await createExecution({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const currentUserExecutionTwo = await createExecution({ |  | ||||||
|       flowId: currentUserFlow.id, |  | ||||||
|       deletedAt: new Date().toISOString(), |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: ['isCreator'], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/executions') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getExecutionsMock( |  | ||||||
|       [currentUserExecutionTwo, currentUserExecutionOne], |  | ||||||
|       currentUserFlow, |  | ||||||
|       [stepOne, stepTwo] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the executions of another user', async () => { |  | ||||||
|     const anotherUserFlow = await createFlow({ |  | ||||||
|       userId: anotherUser.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepOne = await createStep({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|       type: 'trigger', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const stepTwo = await createStep({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|       type: 'action', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const anotherUserExecutionOne = await createExecution({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const anotherUserExecutionTwo = await createExecution({ |  | ||||||
|       flowId: anotherUserFlow.id, |  | ||||||
|       deletedAt: new Date().toISOString(), |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Execution', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/executions') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getExecutionsMock( |  | ||||||
|       [anotherUserExecutionTwo, anotherUserExecutionOne], |  | ||||||
|       anotherUserFlow, |  | ||||||
|       [stepOne, stepTwo] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const flow = await request.currentUser.authorizedFlows |  | ||||||
|     .withGraphJoined({ steps: true }) |  | ||||||
|     .orderBy('steps.position', 'asc') |  | ||||||
|     .findOne({ 'flows.id': request.params.flowId }) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   renderObject(response, flow); |  | ||||||
| }; |  | ||||||
| @@ -1,102 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import Crypto from 'crypto'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import { createFlow } from '../../../../../test/factories/flow'; |  | ||||||
| import { createStep } from '../../../../../test/factories/step'; |  | ||||||
| import { createPermission } from '../../../../../test/factories/permission'; |  | ||||||
| import getFlowMock from '../../../../../test/mocks/rest/api/v1/flows/get-flow'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/flows/:flowId', () => { |  | ||||||
|   let currentUser, currentUserRole, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     currentUserRole = await currentUser.$relatedQuery('role'); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the flow data of current user', async () => { |  | ||||||
|     const currentUserflow = await createFlow({ userId: currentUser.id }); |  | ||||||
|     const triggerStep = await createStep({ flowId: currentUserflow.id }); |  | ||||||
|     const actionStep = await createStep({ flowId: currentUserflow.id }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Flow', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: ['isCreator'], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/flows/${currentUserflow.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getFlowMock(currentUserflow, [ |  | ||||||
|       triggerStep, |  | ||||||
|       actionStep, |  | ||||||
|     ]); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return the flow data of another user', async () => { |  | ||||||
|     const anotherUser = await createUser(); |  | ||||||
|     const anotherUserFlow = await createFlow({ userId: anotherUser.id }); |  | ||||||
|     const triggerStep = await createStep({ flowId: anotherUserFlow.id }); |  | ||||||
|     const actionStep = await createStep({ flowId: anotherUserFlow.id }); |  | ||||||
|  |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Flow', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get(`/api/v1/flows/${anotherUserFlow.id}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getFlowMock(anotherUserFlow, [ |  | ||||||
|       triggerStep, |  | ||||||
|       actionStep, |  | ||||||
|     ]); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return not found response for not existing flow UUID', async () => { |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Flow', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const notExistingFlowUUID = Crypto.randomUUID(); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get(`/api/v1/flows/${notExistingFlowUUID}`) |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(404); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return bad request response for invalid UUID', async () => { |  | ||||||
|     await createPermission({ |  | ||||||
|       action: 'read', |  | ||||||
|       subject: 'Flow', |  | ||||||
|       roleId: currentUserRole.id, |  | ||||||
|       conditions: [], |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     await request(app) |  | ||||||
|       .get('/api/v1/flows/invalidFlowUUID') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(400); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
| import Billing from '../../../../helpers/billing/index.ee.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const paddleInfo = Billing.paddleInfo; |  | ||||||
|  |  | ||||||
|   renderObject(response, paddleInfo); |  | ||||||
| }; |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user.js'; |  | ||||||
| import getPaddleInfoMock from '../../../../../test/mocks/rest/api/v1/payment/get-paddle-info.js'; |  | ||||||
| import appConfig from '../../../../config/app.js'; |  | ||||||
| import billing from '../../../../helpers/billing/index.ee.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/payment/paddle-info', () => { |  | ||||||
|   let user, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     user = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(user.id); |  | ||||||
|  |  | ||||||
|     vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(true); |  | ||||||
|     vi.spyOn(billing.paddleInfo, 'vendorId', 'get').mockReturnValue( |  | ||||||
|       'sampleVendorId' |  | ||||||
|     ); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return payment plans', async () => { |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/payment/paddle-info') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedResponsePayload = await getPaddleInfoMock(); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
| import Billing from '../../../../helpers/billing/index.ee.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const paymentPlans = Billing.paddlePlans; |  | ||||||
|  |  | ||||||
|   renderObject(response, paymentPlans); |  | ||||||
| }; |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user.js'; |  | ||||||
| import getPaymentPlansMock from '../../../../../test/mocks/rest/api/v1/payment/get-plans.js'; |  | ||||||
| import appConfig from '../../../../config/app.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/payment/plans', () => { |  | ||||||
|   let user, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     user = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(user.id); |  | ||||||
|  |  | ||||||
|     vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(true); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return payment plans', async () => { |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/payment/plans') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedResponsePayload = await getPaymentPlansMock(); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   renderObject(response, request.currentUser); |  | ||||||
| }; |  | ||||||
| @@ -1,44 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createPermission } from '../../../../../test/factories/permission'; |  | ||||||
| import { createRole } from '../../../../../test/factories/role'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import getCurrentUserMock from '../../../../../test/mocks/rest/api/v1/users/get-current-user'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/users/me', () => { |  | ||||||
|   let role, permissionOne, permissionTwo, currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     role = await createRole(); |  | ||||||
|  |  | ||||||
|     permissionOne = await createPermission({ |  | ||||||
|       roleId: role.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     permissionTwo = await createPermission({ |  | ||||||
|       roleId: role.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     currentUser = await createUser({ |  | ||||||
|       roleId: role.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return current user info', async () => { |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/users/me') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = getCurrentUserMock(currentUser, role, [ |  | ||||||
|       permissionOne, |  | ||||||
|       permissionTwo, |  | ||||||
|     ]); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const invoices = await request.currentUser.getInvoices(); |  | ||||||
|  |  | ||||||
|   renderObject(response, invoices); |  | ||||||
| }; |  | ||||||
| @@ -1,34 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach, vi } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user'; |  | ||||||
| import User from '../../../../models/user'; |  | ||||||
| import getInvoicesMock from '../../../../../test/mocks/rest/api/v1/users/get-invoices.ee'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/user/invoices', () => { |  | ||||||
|   let currentUser, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     currentUser = await createUser(); |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return current user invoices', async () => { |  | ||||||
|     const invoices = [ |  | ||||||
|       { id: 1, amount: 100, description: 'Invoice 1' }, |  | ||||||
|       { id: 2, amount: 200, description: 'Invoice 2' }, |  | ||||||
|     ]; |  | ||||||
|  |  | ||||||
|     vi.spyOn(User.prototype, 'getInvoices').mockResolvedValue(invoices); |  | ||||||
|  |  | ||||||
|     const response = await request(app) |  | ||||||
|       .get('/api/v1/users/invoices') |  | ||||||
|       .set('Authorization', token) |  | ||||||
|       .expect(200); |  | ||||||
|  |  | ||||||
|     const expectedPayload = await getInvoicesMock(invoices); |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedPayload); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| import { renderObject } from '../../../../helpers/renderer.js'; |  | ||||||
|  |  | ||||||
| export default async (request, response) => { |  | ||||||
|   const inTrial = await request.currentUser.inTrial(); |  | ||||||
|  |  | ||||||
|   const trialInfo = { |  | ||||||
|     inTrial, |  | ||||||
|     expireAt: request.currentUser.trialExpiryDate, |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   renderObject(response, trialInfo); |  | ||||||
| }; |  | ||||||
| @@ -1,38 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../../../app.js'; |  | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js'; |  | ||||||
| import { createUser } from '../../../../../test/factories/user.js'; |  | ||||||
| import getUserTrialMock from '../../../../../test/mocks/rest/api/v1/users/get-user-trial.js'; |  | ||||||
| import appConfig from '../../../../config/app.js'; |  | ||||||
| import { DateTime } from 'luxon'; |  | ||||||
| import User from '../../../../models/user.js'; |  | ||||||
|  |  | ||||||
| describe('GET /api/v1/users/:userId/trial', () => { |  | ||||||
|   let user, token; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     const trialExpiryDate = DateTime.now().plus({ days: 30 }).toISODate(); |  | ||||||
|     user = await createUser({ trialExpiryDate }); |  | ||||||
|     token = createAuthTokenByUserId(user.id); |  | ||||||
|  |  | ||||||
|     vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(true); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   describe('should return in trial, active subscription and expire at info', () => { |  | ||||||
|     beforeEach(async () => { |  | ||||||
|       vi.spyOn(User.prototype, 'inTrial').mockResolvedValue(false); |  | ||||||
|       vi.spyOn(User.prototype, 'hasActiveSubscription').mockResolvedValue(true); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     it('should return null', async () => { |  | ||||||
|       const response = await request(app) |  | ||||||
|         .get(`/api/v1/users/${user.id}/trial`) |  | ||||||
|         .set('Authorization', token) |  | ||||||
|         .expect(200); |  | ||||||
|  |  | ||||||
|       const expectedResponsePayload = await getUserTrialMock(user); |  | ||||||
|       expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| export default async (request, response) => { |  | ||||||
|   response.status(200).end(); |  | ||||||
| }; |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| import { describe, it } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../app.js'; |  | ||||||
|  |  | ||||||
| describe('GET /healthcheck', () => { |  | ||||||
|   it('should return 200 response with version data', async () => { |  | ||||||
|     await request(app).get('/healthcheck').expect(200); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| export async function up(knex) { |  | ||||||
|   return knex.schema.createTable('datastore', (table) => { |  | ||||||
|     table.uuid('id').primary().defaultTo(knex.raw('gen_random_uuid()')); |  | ||||||
|     table.string('key').notNullable(); |  | ||||||
|     table.string('value'); |  | ||||||
|     table.string('scope').notNullable(); |  | ||||||
|     table.uuid('scope_id').notNullable(); |  | ||||||
|     table.index(['key', 'scope', 'scope_id']); |  | ||||||
|  |  | ||||||
|     table.timestamps(true, true); |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function down(knex) { |  | ||||||
|   return knex.schema.dropTable('datastore'); |  | ||||||
| } |  | ||||||
| @@ -1,13 +1,8 @@ | |||||||
| import Step from '../../models/flow.js'; |  | ||||||
|  |  | ||||||
| const deleteStep = async (_parent, params, context) => { | const deleteStep = async (_parent, params, context) => { | ||||||
|   const conditions = context.currentUser.can('update', 'Flow'); |   context.currentUser.can('update', 'Flow'); | ||||||
|   const isCreator = conditions.isCreator; |  | ||||||
|   const allSteps = Step.query(); |  | ||||||
|   const userSteps = context.currentUser.$relatedQuery('steps'); |  | ||||||
|   const baseQuery = isCreator ? userSteps : allSteps; |  | ||||||
|  |  | ||||||
|   const step = await baseQuery |   const step = await context.currentUser | ||||||
|  |     .$relatedQuery('steps') | ||||||
|     .withGraphFetched('flow') |     .withGraphFetched('flow') | ||||||
|     .findOne({ |     .findOne({ | ||||||
|       'steps.id': params.input.id, |       'steps.id': params.input.id, | ||||||
|   | |||||||
| @@ -1,10 +1,7 @@ | |||||||
| import appConfig from '../../config/app.js'; |  | ||||||
| import User from '../../models/user.js'; | import User from '../../models/user.js'; | ||||||
| import Role from '../../models/role.js'; | import Role from '../../models/role.js'; | ||||||
|  |  | ||||||
| const registerUser = async (_parent, params) => { | const registerUser = async (_parent, params) => { | ||||||
|   if (!appConfig.isCloud) return; |  | ||||||
|  |  | ||||||
|   const { fullName, email, password } = params.input; |   const { fullName, email, password } = params.input; | ||||||
|  |  | ||||||
|   const existingUser = await User.query().findOne({ |   const existingUser = await User.query().findOne({ | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								packages/backend/src/graphql/queries/get-automatisch-info.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								packages/backend/src/graphql/queries/get-automatisch-info.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | import appConfig from '../../config/app.js'; | ||||||
|  | import { getLicense } from '../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | const getAutomatischInfo = async () => { | ||||||
|  |   const license = await getLicense(); | ||||||
|  |  | ||||||
|  |   const computedLicense = { | ||||||
|  |     id: license ? license.id : null, | ||||||
|  |     name: license ? license.name : null, | ||||||
|  |     expireAt: license ? license.expireAt : null, | ||||||
|  |     verified: license ? true : false, | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     isCloud: appConfig.isCloud, | ||||||
|  |     isMation: appConfig.isMation, | ||||||
|  |     license: computedLicense, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default getAutomatischInfo; | ||||||
| @@ -0,0 +1,190 @@ | |||||||
|  | import { vi, describe, it, expect, beforeEach } from 'vitest'; | ||||||
|  | import request from 'supertest'; | ||||||
|  | import app from '../../app'; | ||||||
|  | import * as license from '../../helpers/license.ee'; | ||||||
|  | import appConfig from '../../config/app'; | ||||||
|  |  | ||||||
|  | describe('graphQL getAutomatischInfo query', () => { | ||||||
|  |   const query = ` | ||||||
|  |     query { | ||||||
|  |       getAutomatischInfo { | ||||||
|  |         isCloud | ||||||
|  |         isMation | ||||||
|  |         license { | ||||||
|  |           id | ||||||
|  |           name | ||||||
|  |           expireAt | ||||||
|  |           verified | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  |  | ||||||
|  |   describe('and without valid license', () => { | ||||||
|  |     beforeEach(async () => { | ||||||
|  |       vi.spyOn(license, 'getLicense').mockResolvedValue(false); | ||||||
|  |  | ||||||
|  |       vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(false); | ||||||
|  |       vi.spyOn(appConfig, 'isMation', 'get').mockReturnValue(false); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('should return empty license data', async () => { | ||||||
|  |       const response = await request(app) | ||||||
|  |         .post('/graphql') | ||||||
|  |         .send({ query }) | ||||||
|  |         .expect(200); | ||||||
|  |  | ||||||
|  |       const expectedResponsePayload = { | ||||||
|  |         data: { | ||||||
|  |           getAutomatischInfo: { | ||||||
|  |             isCloud: false, | ||||||
|  |             isMation: false, | ||||||
|  |             license: { | ||||||
|  |               id: null, | ||||||
|  |               name: null, | ||||||
|  |               expireAt: null, | ||||||
|  |               verified: false, | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       expect(response.body).toEqual(expectedResponsePayload); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   describe('and with valid license', () => { | ||||||
|  |     beforeEach(async () => { | ||||||
|  |       const mockedLicense = { | ||||||
|  |         id: '123123', | ||||||
|  |         name: 'Test License', | ||||||
|  |         expireAt: '2025-08-09T10:56:54.144Z', | ||||||
|  |         verified: true, | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       vi.spyOn(license, 'getLicense').mockResolvedValue(mockedLicense); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     describe('and with cloud flag enabled', () => { | ||||||
|  |       beforeEach(async () => { | ||||||
|  |         vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(true); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       it('should return all license data', async () => { | ||||||
|  |         const response = await request(app) | ||||||
|  |           .post('/graphql') | ||||||
|  |           .send({ query }) | ||||||
|  |           .expect(200); | ||||||
|  |  | ||||||
|  |         const expectedResponsePayload = { | ||||||
|  |           data: { | ||||||
|  |             getAutomatischInfo: { | ||||||
|  |               isCloud: true, | ||||||
|  |               isMation: false, | ||||||
|  |               license: { | ||||||
|  |                 expireAt: '2025-08-09T10:56:54.144Z', | ||||||
|  |                 id: '123123', | ||||||
|  |                 name: 'Test License', | ||||||
|  |                 verified: true, | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         expect(response.body).toEqual(expectedResponsePayload); | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     describe('and with cloud flag disabled', () => { | ||||||
|  |       beforeEach(async () => { | ||||||
|  |         vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(false); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       it('should return all license data', async () => { | ||||||
|  |         const response = await request(app) | ||||||
|  |           .post('/graphql') | ||||||
|  |           .send({ query }) | ||||||
|  |           .expect(200); | ||||||
|  |  | ||||||
|  |         const expectedResponsePayload = { | ||||||
|  |           data: { | ||||||
|  |             getAutomatischInfo: { | ||||||
|  |               isCloud: false, | ||||||
|  |               isMation: false, | ||||||
|  |               license: { | ||||||
|  |                 expireAt: '2025-08-09T10:56:54.144Z', | ||||||
|  |                 id: '123123', | ||||||
|  |                 name: 'Test License', | ||||||
|  |                 verified: true, | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         expect(response.body).toEqual(expectedResponsePayload); | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     describe('and with mation flag enabled', () => { | ||||||
|  |       beforeEach(async () => { | ||||||
|  |         vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(false); | ||||||
|  |         vi.spyOn(appConfig, 'isMation', 'get').mockReturnValue(true); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       it('should return all license data', async () => { | ||||||
|  |         const response = await request(app) | ||||||
|  |           .post('/graphql') | ||||||
|  |           .send({ query }) | ||||||
|  |           .expect(200); | ||||||
|  |  | ||||||
|  |         const expectedResponsePayload = { | ||||||
|  |           data: { | ||||||
|  |             getAutomatischInfo: { | ||||||
|  |               isCloud: false, | ||||||
|  |               isMation: true, | ||||||
|  |               license: { | ||||||
|  |                 expireAt: '2025-08-09T10:56:54.144Z', | ||||||
|  |                 id: '123123', | ||||||
|  |                 name: 'Test License', | ||||||
|  |                 verified: true, | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         expect(response.body).toEqual(expectedResponsePayload); | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     describe('and with mation flag disabled', () => { | ||||||
|  |       beforeEach(async () => { | ||||||
|  |         vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(false); | ||||||
|  |         vi.spyOn(appConfig, 'isMation', 'get').mockReturnValue(false); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       it('should return all license data', async () => { | ||||||
|  |         const response = await request(app) | ||||||
|  |           .post('/graphql') | ||||||
|  |           .send({ query }) | ||||||
|  |           .expect(200); | ||||||
|  |  | ||||||
|  |         const expectedResponsePayload = { | ||||||
|  |           data: { | ||||||
|  |             getAutomatischInfo: { | ||||||
|  |               isMation: false, | ||||||
|  |               isCloud: false, | ||||||
|  |               license: { | ||||||
|  |                 expireAt: '2025-08-09T10:56:54.144Z', | ||||||
|  |                 id: '123123', | ||||||
|  |                 name: 'Test License', | ||||||
|  |                 verified: true, | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         expect(response.body).toEqual(expectedResponsePayload); | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -1,17 +1,9 @@ | |||||||
| import appConfig from '../../config/app.js'; |  | ||||||
| import { hasValidLicense } from '../../helpers/license.ee.js'; | import { hasValidLicense } from '../../helpers/license.ee.js'; | ||||||
| import Config from '../../models/config.js'; | import Config from '../../models/config.js'; | ||||||
|  |  | ||||||
| const getConfig = async (_parent, params) => { | const getConfig = async (_parent, params) => { | ||||||
|   if (!(await hasValidLicense())) return {}; |   if (!(await hasValidLicense())) return {}; | ||||||
|  |  | ||||||
|   const defaultConfig = { |  | ||||||
|     disableNotificationsPage: appConfig.disableNotificationsPage, |  | ||||||
|     disableFavicon: appConfig.disableFavicon, |  | ||||||
|     additionalDrawerLink: appConfig.additionalDrawerLink, |  | ||||||
|     additionalDrawerLinkText: appConfig.additionalDrawerLinkText, |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   const configQuery = Config.query(); |   const configQuery = Config.query(); | ||||||
|  |  | ||||||
|   if (Array.isArray(params.keys)) { |   if (Array.isArray(params.keys)) { | ||||||
| @@ -26,7 +18,7 @@ const getConfig = async (_parent, params) => { | |||||||
|     computedConfig[key] = value?.data; |     computedConfig[key] = value?.data; | ||||||
|  |  | ||||||
|     return computedConfig; |     return computedConfig; | ||||||
|   }, defaultConfig); |   }, {}); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export default getConfig; | export default getConfig; | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ import { vi, describe, it, expect, beforeEach } from 'vitest'; | |||||||
| import request from 'supertest'; | import request from 'supertest'; | ||||||
| import app from '../../app'; | import app from '../../app'; | ||||||
| import { createConfig } from '../../../test/factories/config'; | import { createConfig } from '../../../test/factories/config'; | ||||||
| import appConfig from '../../config/app'; |  | ||||||
| import * as license from '../../helpers/license.ee'; | import * as license from '../../helpers/license.ee'; | ||||||
|  |  | ||||||
| describe('graphQL getConfig query', () => { | describe('graphQL getConfig query', () => { | ||||||
| @@ -57,10 +56,6 @@ describe('graphQL getConfig query', () => { | |||||||
|               [configOne.key]: configOne.value.data, |               [configOne.key]: configOne.value.data, | ||||||
|               [configTwo.key]: configTwo.value.data, |               [configTwo.key]: configTwo.value.data, | ||||||
|               [configThree.key]: configThree.value.data, |               [configThree.key]: configThree.value.data, | ||||||
|               disableNotificationsPage: false, |  | ||||||
|               disableFavicon: false, |  | ||||||
|               additionalDrawerLink: undefined, |  | ||||||
|               additionalDrawerLinkText: undefined, |  | ||||||
|             }, |             }, | ||||||
|           }, |           }, | ||||||
|         }; |         }; | ||||||
| @@ -87,48 +82,6 @@ describe('graphQL getConfig query', () => { | |||||||
|             getConfig: { |             getConfig: { | ||||||
|               [configOne.key]: configOne.value.data, |               [configOne.key]: configOne.value.data, | ||||||
|               [configTwo.key]: configTwo.value.data, |               [configTwo.key]: configTwo.value.data, | ||||||
|               disableNotificationsPage: false, |  | ||||||
|               disableFavicon: false, |  | ||||||
|               additionalDrawerLink: undefined, |  | ||||||
|               additionalDrawerLinkText: undefined, |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and with different defaults', () => { |  | ||||||
|       beforeEach(async () => { |  | ||||||
|         vi.spyOn(appConfig, 'disableNotificationsPage', 'get').mockReturnValue( |  | ||||||
|           true |  | ||||||
|         ); |  | ||||||
|         vi.spyOn(appConfig, 'disableFavicon', 'get').mockReturnValue(true); |  | ||||||
|         vi.spyOn(appConfig, 'additionalDrawerLink', 'get').mockReturnValue( |  | ||||||
|           'https://automatisch.io' |  | ||||||
|         ); |  | ||||||
|         vi.spyOn(appConfig, 'additionalDrawerLinkText', 'get').mockReturnValue( |  | ||||||
|           'Automatisch' |  | ||||||
|         ); |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       it('should return custom config', async () => { |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .send({ query }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getConfig: { |  | ||||||
|               [configOne.key]: configOne.value.data, |  | ||||||
|               [configTwo.key]: configTwo.value.data, |  | ||||||
|               [configThree.key]: configThree.value.data, |  | ||||||
|               disableNotificationsPage: true, |  | ||||||
|               disableFavicon: true, |  | ||||||
|               additionalDrawerLink: 'https://automatisch.io', |  | ||||||
|               additionalDrawerLinkText: 'Automatisch', |  | ||||||
|             }, |             }, | ||||||
|           }, |           }, | ||||||
|         }; |         }; | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ import getAppAuthClient from './queries/get-app-auth-client.ee.js'; | |||||||
| import getAppAuthClients from './queries/get-app-auth-clients.ee.js'; | import getAppAuthClients from './queries/get-app-auth-clients.ee.js'; | ||||||
| import getAppConfig from './queries/get-app-config.ee.js'; | import getAppConfig from './queries/get-app-config.ee.js'; | ||||||
| import getApps from './queries/get-apps.js'; | import getApps from './queries/get-apps.js'; | ||||||
|  | import getAutomatischInfo from './queries/get-automatisch-info.js'; | ||||||
| import getBillingAndUsage from './queries/get-billing-and-usage.ee.js'; | import getBillingAndUsage from './queries/get-billing-and-usage.ee.js'; | ||||||
| import getConfig from './queries/get-config.ee.js'; | import getConfig from './queries/get-config.ee.js'; | ||||||
| import getConnectedApps from './queries/get-connected-apps.js'; | import getConnectedApps from './queries/get-connected-apps.js'; | ||||||
| @@ -38,6 +39,7 @@ const queryResolvers = { | |||||||
|   getAppAuthClients, |   getAppAuthClients, | ||||||
|   getAppConfig, |   getAppConfig, | ||||||
|   getApps, |   getApps, | ||||||
|  |   getAutomatischInfo, | ||||||
|   getBillingAndUsage, |   getBillingAndUsage, | ||||||
|   getConfig, |   getConfig, | ||||||
|   getConnectedApps, |   getConnectedApps, | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user