Compare commits
	
		
			254 Commits
		
	
	
		
			snackbar-o
			...
			AUT-934
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | b6b4ed5ad2 | ||
|   | b594a8e0f3 | ||
|   | e4292815cd | ||
|   | ab37250d5d | ||
|   | e5be8d3ba7 | ||
|   | 96a421fa22 | ||
|   | 12f72401b1 | ||
|   | 7391a9eddc | ||
|   | 30dee27f72 | ||
|   | 51a9939034 | ||
|   | e03c6e0ca4 | ||
|   | bece5c6488 | ||
|   | d49bb4c52d | ||
|   | 73d0eec30c | ||
|   | 5c756b16ca | ||
|   | f482c2422c | ||
|   | 2e564c863f | ||
|   | d9917a81bb | ||
|   | 61dc431f92 | ||
|   | 7d2fb8d9d7 | ||
|   | 608b79b66f | ||
|   | 009754c18b | ||
|   | 5df07c289e | ||
|   | a36d10870b | ||
|   | b549ba3e39 | ||
|   | 897c96361f | ||
|   | e7693d8aa6 | ||
|   | 1fe755f836 | ||
|   | ea1a63f7dd | ||
|   | 85134722a5 | ||
|   | 5c9d3ed134 | ||
|   | 17fb935ea0 | ||
|   | 196642a1cf | ||
|   | 009cf63d8c | ||
|   | da399aacd6 | ||
|   | 3632ee77e5 | ||
|   | 2901f337cc | ||
|   | f0bd2f335b | ||
|   | acdd026448 | ||
|   | fb0a328ab0 | ||
|   | d2a7889fc9 | ||
|   | 88c50e014d | ||
|   | f0ef12f904 | ||
|   | 1827f5413f | ||
|   | 0609f30e25 | ||
|   | d4e4d95b6d | ||
|   | d74af4931e | ||
|   | bee043d10d | ||
|   | a65e48b98a | ||
|   | ee26b54d54 | ||
|   | 855ec53dc2 | ||
|   | 3e3e48110d | ||
|   | fc04a357c8 | ||
|   | c8147370de | ||
|   | 999426be89 | ||
|   | 91458f91ef | ||
|   | 4b9ed29cc0 | ||
|   | e3bcb673fb | ||
|   | bf4776ca4f | ||
|   | 9f7f30a92a | ||
|   | 5c29fff55e | ||
|   | a0160c2573 | ||
|   | 87d3ca287d | ||
|   | 526e093689 | ||
|   | 0930c9d8d6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ec680a713d | ||
|   | 583f90d1e9 | ||
|   | 8ba95381bc | ||
|   | ec6d634b99 | ||
|   | bc082acbe7 | ||
|   | e474ba02cb | ||
|   | ea922aaf10 | ||
|   | 766e6e20d8 | ||
|   | 8e646c244e | ||
|   | 26f31a5899 | ||
|   | 5c79e374dd | ||
|   | 7c1473ea95 | ||
|   | 1fe4cc3258 | ||
|   | 042ad4cea1 | ||
|   | e4c998dbce | ||
|   | 83c8cacdac | ||
|   | f75d5d906e | ||
|   | 85b3856564 | ||
|   | 75cb2569b5 | ||
|   | 0a4ac1cece | ||
|   | a873fd14bd | ||
|   | 85b4cd4998 | ||
|   | e9bc9b1aa8 | ||
|   | e3bf599bf6 | ||
|   | 01ae96840e | ||
|   | 186160ebf4 | ||
|   | 70f5e45c1f | ||
|   | 6dc54ecabc | ||
|   | d21888c047 | ||
|   | 33f7a90042 | ||
|   | a00d3a2c5e | ||
|   | abc64d769c | ||
|   | 88754ac569 | ||
|   | e3ee05d47d | ||
|   | 3b004e7483 | ||
|   | 9aa48c20e4 | ||
|   | 1b5d3beeca | ||
|   | 00115d313e | ||
|   | 190f1a205f | ||
|   | 8ab6f0c3fe | ||
|   | 13eea263c0 | ||
|   | b52b40962e | ||
|   | 7d1fa2e40c | ||
|   | 93b2098829 | ||
|   | a2acdc6b12 | ||
|   | 38b2c1e30f | ||
|   | e07f579f3c | ||
|   | df3297b6ca | ||
|   | fc4eeed764 | ||
|   | 3596d13be1 | ||
|   | 104d49ea1c | ||
|   | 7057317446 | ||
|   | 280575df88 | ||
|   | d2cb434b7b | ||
|   | 2ecb802a2e | ||
|   | 46e706c415 | ||
|   | 3a57349d8a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 565db852e0 | ||
|   | 754c3269ec | ||
|   | a079842408 | ||
|   | 7664b58553 | ||
|   | de77488f7e | ||
|   | d808afd21b | ||
|   | b68aff76a1 | ||
|   | 6da7fe158f | ||
|   | 4dbc7fdc7d | ||
|   | ad1e1f7eca | ||
|   | 9c3f7a3823 | ||
|   | 86f4cb7701 | ||
|   | 359a90245d | ||
|   | d8d7d86359 | ||
|   | 7189b629c0 | ||
|   | 55c9b5566c | ||
|   | ab671ccbf7 | ||
|   | 316bda8c3f | ||
|   | 76f77e8a4c | ||
|   | 4a99d5eab7 | ||
|   | 473d287c6d | ||
|   | bddd9896e4 | ||
|   | 95eb115965 | ||
|   | 9a63b213b0 | ||
|   | 90b00d88f1 | ||
|   | ec87c7f21c | ||
|   | 452f45cac6 | ||
|   | c644b3d384 | ||
|   | 68160c20e8 | ||
|   | ad144206dd | ||
|   | f3d20ab769 | ||
|   | 9767ca7116 | ||
|   | 73a5b8553f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5c684cd499 | ||
|   | 479f3e3172 | ||
|   | 6a1350fd00 | ||
|   | 563784da1c | ||
|   | 347f0ed3a5 | ||
|   | 1db9f5b2c2 | ||
|   | f2a3e26188 | ||
|   | 1c0897bfb6 | ||
|   | f0793992a6 | ||
|   | 393205ba2f | ||
|   | ab49535b6c | ||
|   | 8191b48548 | ||
|   | 925dd06432 | ||
|   | 0d525e056a | ||
|   | 3aa86eebf2 | ||
|   | d7a93abec0 | ||
|   | 6448d28a18 | ||
|   | 449976483c | ||
|   | e468b762ef | ||
|   | b578e73cc4 | ||
|   | e381f95b95 | ||
|   | 5685afae63 | ||
|   | 53a473422b | ||
|   | 3f9f17f584 | ||
|   | 2c410bf318 | ||
|   | 40934a2c77 | ||
|   | ecc9379d7e | ||
|   | f8d27342dc | ||
|   | 17bd2bf2ba | ||
|   | d984a3f275 | ||
|   | 64049bd546 | ||
|   | 9218091c33 | ||
|   | 75df7d6413 | ||
|   | 29341f81e1 | ||
|   | 68c5a3dca7 | ||
|   | b1e2e370c8 | ||
|   | ba9d3afc88 | ||
|   | 9a0434be32 | ||
|   | d6923a2ff0 | ||
|   | 8f7f6dc19e | ||
|   | 70b8817643 | ||
|   | 87c25cbbfe | ||
|   | 082e905014 | ||
|   | e3e598b208 | ||
|   | 6cf92d4ea6 | ||
|   | 89ad685f3a | ||
|   | 1e868dc802 | ||
|   | 58fcfd9a34 | ||
|   | c849afbc11 | ||
|   | 2ebe71ddd0 | ||
|   | 7a8e8c1f3e | ||
|   | 5e897ad1c2 | ||
|   | ce07907f85 | ||
|   | 1e38aa7b53 | ||
|   | 3bc0c23e5a | ||
|   | f07b6d105a | ||
|   | 46491269e3 | ||
|   | 8d9c43af6a | ||
|   | c3568354aa | ||
|   | e68696ccd4 | ||
|   | 35d8b2e790 | ||
|   | 1f83573206 | ||
|   | 2887e76514 | ||
|   | 63b9943203 | ||
|   | bd5aedd83f | ||
|   | c9ff6d7bb9 | ||
|   | 5835def5d0 | ||
|   | 6a2694ce3b | ||
|   | 65ae7bce79 | ||
|   | 57ce8da0ee | ||
|   | 7484bf7403 | ||
|   | f6b2312c49 | ||
|   | 6027cb7cb0 | ||
|   | c1740aae6c | ||
|   | 22ce29e86c | ||
|   | 251d1b5b2e | ||
|   | ea64708c69 | ||
|   | 209ec27a29 | ||
|   | ceee495525 | ||
|   | 751a2347aa | ||
|   | efd96d5fdf | ||
|   | 2ee5af8bfb | ||
|   | a4c0edf493 | ||
|   | 28c8be97b6 | ||
|   | f320a44d45 | ||
|   | c0cc6cc176 | ||
|   | be62c09d06 | ||
|   | 5fe3546d2a | ||
|   | c4b2ea125c | ||
|   | 3301b038fe | ||
|   | 6a58d1e3da | ||
|   | e5670d820d | ||
|   | 1870aead73 | ||
|   | 95db6cca2c | ||
|   | 79a792ac62 | ||
|   | 6406f9eb86 | ||
|   | 8f8ec496f8 | ||
|   | bd1ad5fa56 | ||
|   | f2e22e7445 | 
							
								
								
									
										32
									
								
								.github/workflows/docs-change.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								.github/workflows/docs-change.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | |||||||
|  | name: Automatisch Docs Change | ||||||
|  | on: | ||||||
|  |   pull_request: | ||||||
|  |     paths: | ||||||
|  |       - 'packages/docs/**' | ||||||
|  | jobs: | ||||||
|  |   label: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout repository | ||||||
|  |         uses: actions/checkout@v3 | ||||||
|  |       - name: Label PR | ||||||
|  |         uses: actions/github-script@v6 | ||||||
|  |         with: | ||||||
|  |           script: | | ||||||
|  |             const { pull_request } = context.payload; | ||||||
|  |  | ||||||
|  |             const label = 'documentation-change'; | ||||||
|  |             const hasLabel = pull_request.labels.some(({ name }) => name === label); | ||||||
|  |  | ||||||
|  |             if (!hasLabel) { | ||||||
|  |               await github.rest.issues.addLabels({ | ||||||
|  |                 owner: context.repo.owner, | ||||||
|  |                 repo: context.repo.repo, | ||||||
|  |                 issue_number: pull_request.number, | ||||||
|  |                 labels: [label], | ||||||
|  |               }); | ||||||
|  |  | ||||||
|  |               console.log(`Label "${label}" added to PR #${pull_request.number}`); | ||||||
|  |             } else { | ||||||
|  |               console.log(`Label "${label}" already exists on PR #${pull_request.number}`); | ||||||
|  |             } | ||||||
							
								
								
									
										11
									
								
								packages/backend/src/apps/gmail/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/src/apps/gmail/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 49.4 512 399.42"> | ||||||
|  |   <g fill="none" fill-rule="evenodd"> | ||||||
|  |     <g fill-rule="nonzero"> | ||||||
|  |       <path fill="#4285f4" d="M34.91 448.818h81.454V251L0 163.727V413.91c0 19.287 15.622 34.91 34.91 34.91z"/> | ||||||
|  |       <path fill="#34a853" d="M395.636 448.818h81.455c19.287 0 34.909-15.622 34.909-34.909V163.727L395.636 251z"/> | ||||||
|  |       <path fill="#fbbc04" d="M395.636 99.727V251L512 163.727v-46.545c0-43.142-49.25-67.782-83.782-41.891z"/> | ||||||
|  |     </g> | ||||||
|  |     <path fill="#ea4335" d="M116.364 251V99.727L256 204.455 395.636 99.727V251L256 355.727z"/> | ||||||
|  |     <path fill="#c5221f" fill-rule="nonzero" d="M0 117.182v46.545L116.364 251V99.727L83.782 75.291C49.25 49.4 0 74.04 0 117.18z"/> | ||||||
|  |   </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 720 B | 
							
								
								
									
										23
									
								
								packages/backend/src/apps/gmail/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								packages/backend/src/apps/gmail/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | import { URLSearchParams } from 'url'; | ||||||
|  | import authScope from '../common/auth-scope.js'; | ||||||
|  |  | ||||||
|  | export default async function generateAuthUrl($) { | ||||||
|  |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|  |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|  |   ); | ||||||
|  |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|  |   const searchParams = new URLSearchParams({ | ||||||
|  |     client_id: $.auth.data.clientId, | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|  |     prompt: 'select_account', | ||||||
|  |     scope: authScope.join(' '), | ||||||
|  |     response_type: 'code', | ||||||
|  |     access_type: 'offline', | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const url = `https://accounts.google.com/o/oauth2/v2/auth?${searchParams.toString()}`; | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     url, | ||||||
|  |   }); | ||||||
|  | } | ||||||
							
								
								
									
										48
									
								
								packages/backend/src/apps/gmail/auth/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								packages/backend/src/apps/gmail/auth/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | |||||||
|  | import generateAuthUrl from './generate-auth-url.js'; | ||||||
|  | import verifyCredentials from './verify-credentials.js'; | ||||||
|  | import refreshToken from './refresh-token.js'; | ||||||
|  | import isStillVerified from './is-still-verified.js'; | ||||||
|  |  | ||||||
|  | export default { | ||||||
|  |   fields: [ | ||||||
|  |     { | ||||||
|  |       key: 'oAuthRedirectUrl', | ||||||
|  |       label: 'OAuth Redirect URL', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: true, | ||||||
|  |       value: '{WEB_APP_URL}/app/gmail/connections/add', | ||||||
|  |       placeholder: null, | ||||||
|  |       description: | ||||||
|  |         'When asked to input a redirect URL in Google Cloud, enter the URL above.', | ||||||
|  |       clickToCopy: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'clientId', | ||||||
|  |       label: 'Client ID', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: null, | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'clientSecret', | ||||||
|  |       label: 'Client Secret', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: null, | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   generateAuthUrl, | ||||||
|  |   verifyCredentials, | ||||||
|  |   isStillVerified, | ||||||
|  |   refreshToken, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | import getCurrentUser from '../common/get-current-user.js'; | ||||||
|  |  | ||||||
|  | const isStillVerified = async ($) => { | ||||||
|  |   const currentUser = await getCurrentUser($); | ||||||
|  |   return !!currentUser.resourceName; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default isStillVerified; | ||||||
							
								
								
									
										26
									
								
								packages/backend/src/apps/gmail/auth/refresh-token.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								packages/backend/src/apps/gmail/auth/refresh-token.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | import { URLSearchParams } from 'node:url'; | ||||||
|  |  | ||||||
|  | import authScope from '../common/auth-scope.js'; | ||||||
|  |  | ||||||
|  | const refreshToken = async ($) => { | ||||||
|  |   const params = new URLSearchParams({ | ||||||
|  |     client_id: $.auth.data.clientId, | ||||||
|  |     client_secret: $.auth.data.clientSecret, | ||||||
|  |     grant_type: 'refresh_token', | ||||||
|  |     refresh_token: $.auth.data.refreshToken, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     'https://oauth2.googleapis.com/token', | ||||||
|  |     params.toString() | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     scope: authScope.join(' '), | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default refreshToken; | ||||||
							
								
								
									
										43
									
								
								packages/backend/src/apps/gmail/auth/verify-credentials.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								packages/backend/src/apps/gmail/auth/verify-credentials.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | import getCurrentUser from '../common/get-current-user.js'; | ||||||
|  |  | ||||||
|  | const verifyCredentials = async ($) => { | ||||||
|  |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|  |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|  |   ); | ||||||
|  |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|  |   const { data } = await $.http.post(`https://oauth2.googleapis.com/token`, { | ||||||
|  |     client_id: $.auth.data.clientId, | ||||||
|  |     client_secret: $.auth.data.clientSecret, | ||||||
|  |     code: $.auth.data.code, | ||||||
|  |     grant_type: 'authorization_code', | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const currentUser = await getCurrentUser($); | ||||||
|  |  | ||||||
|  |   const { displayName } = currentUser.names.find( | ||||||
|  |     (name) => name.metadata.primary | ||||||
|  |   ); | ||||||
|  |   const { value: email } = currentUser.emailAddresses.find( | ||||||
|  |     (emailAddress) => emailAddress.metadata.primary | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     clientId: $.auth.data.clientId, | ||||||
|  |     clientSecret: $.auth.data.clientSecret, | ||||||
|  |     scope: $.auth.data.scope, | ||||||
|  |     idToken: data.id_token, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     refreshToken: data.refresh_token, | ||||||
|  |     resourceName: currentUser.resourceName, | ||||||
|  |     screenName: `${displayName} - ${email}`, | ||||||
|  |     userId: email, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default verifyCredentials; | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | const addAuthHeader = ($, requestConfig) => { | ||||||
|  |   if ($.auth.data?.accessToken) { | ||||||
|  |     requestConfig.headers.Authorization = `${$.auth.data.tokenType} ${$.auth.data.accessToken}`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default addAuthHeader; | ||||||
							
								
								
									
										8
									
								
								packages/backend/src/apps/gmail/common/auth-scope.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								packages/backend/src/apps/gmail/common/auth-scope.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | const authScope = [ | ||||||
|  |   'https://www.googleapis.com/auth/gmail.compose', | ||||||
|  |   'https://www.googleapis.com/auth/gmail.modify', | ||||||
|  |   'https://www.googleapis.com/auth/userinfo.email', | ||||||
|  |   'https://www.googleapis.com/auth/userinfo.profile', | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | export default authScope; | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | const getCurrentUser = async ($) => { | ||||||
|  |   const { data: currentUser } = await $.http.get( | ||||||
|  |     'https://people.googleapis.com/v1/people/me?personFields=names,emailAddresses' | ||||||
|  |   ); | ||||||
|  |   return currentUser; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default getCurrentUser; | ||||||
							
								
								
									
										3
									
								
								packages/backend/src/apps/gmail/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								packages/backend/src/apps/gmail/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | import listLabels from './list-labels/index.js'; | ||||||
|  |  | ||||||
|  | export default [listLabels]; | ||||||
| @@ -0,0 +1,22 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List labels', | ||||||
|  |   key: 'listLabels', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const labels = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const userId = $.auth.data.userId; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get(`/gmail/v1/users/${userId}/labels`); | ||||||
|  |  | ||||||
|  |     for (const label of data.labels) { | ||||||
|  |       labels.data.push({ | ||||||
|  |         value: label.id, | ||||||
|  |         name: label.name, | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return labels; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										20
									
								
								packages/backend/src/apps/gmail/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								packages/backend/src/apps/gmail/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | import defineApp from '../../helpers/define-app.js'; | ||||||
|  | import addAuthHeader from './common/add-auth-header.js'; | ||||||
|  | import auth from './auth/index.js'; | ||||||
|  | import triggers from './triggers/index.js'; | ||||||
|  | import dynamicData from './dynamic-data/index.js'; | ||||||
|  |  | ||||||
|  | export default defineApp({ | ||||||
|  |   name: 'Gmail', | ||||||
|  |   key: 'gmail', | ||||||
|  |   baseUrl: 'https://mail.google.com', | ||||||
|  |   apiBaseUrl: 'https://gmail.googleapis.com', | ||||||
|  |   iconUrl: '{BASE_URL}/apps/gmail/assets/favicon.svg', | ||||||
|  |   authDocUrl: 'https://automatisch.io/docs/apps/gmail/connection', | ||||||
|  |   primaryColor: 'ea4335', | ||||||
|  |   supportsConnections: true, | ||||||
|  |   beforeRequest: [addAuthHeader], | ||||||
|  |   auth, | ||||||
|  |   triggers, | ||||||
|  |   dynamicData, | ||||||
|  | }); | ||||||
							
								
								
									
										3
									
								
								packages/backend/src/apps/gmail/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								packages/backend/src/apps/gmail/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | import newEmails from './new-emails/index.js'; | ||||||
|  |  | ||||||
|  | export default [newEmails]; | ||||||
							
								
								
									
										68
									
								
								packages/backend/src/apps/gmail/triggers/new-emails/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								packages/backend/src/apps/gmail/triggers/new-emails/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | |||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New emails', | ||||||
|  |   key: 'newEmails', | ||||||
|  |   pollInterval: 15, | ||||||
|  |   description: | ||||||
|  |     'Triggers when a new email is received in the specified mailbox.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Label', | ||||||
|  |       key: 'labelId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         "If you don't choose a label, this Zap will trigger for all emails, including Drafts.", | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listLabels', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const userId = $.auth.data.userId; | ||||||
|  |     const labelId = $.step.parameters.labelId; | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       maxResults: 500, | ||||||
|  |       pageToken: undefined, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if (labelId) { | ||||||
|  |       params.labelIds = labelId; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get(`/gmail/v1/users/${userId}/messages`, { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |       params.pageToken = data.nextPageToken; | ||||||
|  |  | ||||||
|  |       if (!data?.messages?.length) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       for (const message of data.messages) { | ||||||
|  |         const { data: messageData } = await $.http.get( | ||||||
|  |           `/gmail/v1/users/${userId}/messages/${message.id}` | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         $.pushTriggerItem({ | ||||||
|  |           raw: messageData, | ||||||
|  |           meta: { | ||||||
|  |             internalId: messageData.id, | ||||||
|  |           }, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } while (params.pageToken); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -90,7 +90,7 @@ export default defineAction({ | |||||||
|  |  | ||||||
|   async run($) { |   async run($) { | ||||||
|     const method = $.step.parameters.method; |     const method = $.step.parameters.method; | ||||||
|     const data = $.step.parameters.data; |     const data = $.step.parameters.data || null; | ||||||
|     const url = $.step.parameters.url; |     const url = $.step.parameters.url; | ||||||
|     const headers = $.step.parameters.headers; |     const headers = $.step.parameters.headers; | ||||||
|  |  | ||||||
| @@ -108,14 +108,17 @@ export default defineAction({ | |||||||
|       return result; |       return result; | ||||||
|     }, {}); |     }, {}); | ||||||
|  |  | ||||||
|     let contentType = headersObject['content-type']; |     let expectedResponseContentType = headersObject.accept; | ||||||
|  |  | ||||||
|     // in case HEAD request is not supported by the URL |     // in case HEAD request is not supported by the URL | ||||||
|     try { |     try { | ||||||
|       const metadataResponse = await $.http.head(url, { |       const metadataResponse = await $.http.head(url, { | ||||||
|         headers: headersObject, |         headers: headersObject, | ||||||
|       }); |       }); | ||||||
|       contentType = metadataResponse.headers['content-type']; |  | ||||||
|  |       if (!expectedResponseContentType) { | ||||||
|  |         expectedResponseContentType = metadataResponse.headers['content-type']; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       throwIfFileSizeExceedsLimit(metadataResponse.headers['content-length']); |       throwIfFileSizeExceedsLimit(metadataResponse.headers['content-length']); | ||||||
|       // eslint-disable-next-line no-empty |       // eslint-disable-next-line no-empty | ||||||
| @@ -128,7 +131,7 @@ export default defineAction({ | |||||||
|       headers: headersObject, |       headers: headersObject, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     if (!isPossiblyTextBased(contentType)) { |     if (!isPossiblyTextBased(expectedResponseContentType)) { | ||||||
|       requestData.responseType = 'arraybuffer'; |       requestData.responseType = 'arraybuffer'; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -138,7 +141,7 @@ export default defineAction({ | |||||||
|  |  | ||||||
|     let responseData = response.data; |     let responseData = response.data; | ||||||
|  |  | ||||||
|     if (!isPossiblyTextBased(contentType)) { |     if (!isPossiblyTextBased(expectedResponseContentType)) { | ||||||
|       responseData = Buffer.from(responseData).toString('base64'); |       responseData = Buffer.from(responseData).toString('base64'); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -64,33 +64,18 @@ export default defineAction({ | |||||||
|       value: '1', |       value: '1', | ||||||
|       description: |       description: | ||||||
|         'The ID of the stage this deal will be added to. If omitted, the deal will be placed in the first stage of the default pipeline.', |         'The ID of the stage this deal will be added to. If omitted, the deal will be placed in the first stage of the default pipeline.', | ||||||
|       options: [ |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|           { |           { | ||||||
|           label: 'Qualified (Pipeline)', |             name: 'key', | ||||||
|           value: 1, |             value: 'listStages', | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|           label: 'Contact Made (Pipeline)', |  | ||||||
|           value: 2, |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|           label: 'Prospect Qualified (Pipeline)', |  | ||||||
|           value: 3, |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|           label: 'Needs Defined (Pipeline)', |  | ||||||
|           value: 4, |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|           label: 'Proposal Made (Pipeline)', |  | ||||||
|           value: 5, |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|           label: 'Negotiations Started (Pipeline)', |  | ||||||
|           value: 6, |  | ||||||
|           }, |           }, | ||||||
|         ], |         ], | ||||||
|       }, |       }, | ||||||
|  |     }, | ||||||
|     { |     { | ||||||
|       label: 'Owner', |       label: 'Owner', | ||||||
|       key: 'userId', |       key: 'userId', | ||||||
|   | |||||||
| @@ -1,23 +1,25 @@ | |||||||
| import listActivityTypes from './list-activity-types/index.js'; | import listActivityTypes from './list-activity-types/index.js'; | ||||||
| import listCurrencies from './list-currencies/index.js'; | import listCurrencies from './list-currencies/index.js'; | ||||||
| import listDeals from './list-deals/index.js'; | import listDeals from './list-deals/index.js'; | ||||||
| import listLeads from './list-leads/index.js'; |  | ||||||
| import listLeadLabels from './list-lead-labels/index.js'; | import listLeadLabels from './list-lead-labels/index.js'; | ||||||
| import listOrganizations from './list-organizations/index.js'; | import listLeads from './list-leads/index.js'; | ||||||
| import listOrganizationLabelField from './list-organization-label-field/index.js'; | import listOrganizationLabelField from './list-organization-label-field/index.js'; | ||||||
|  | import listOrganizations from './list-organizations/index.js'; | ||||||
| import listPersonLabelField from './list-person-label-field/index.js'; | import listPersonLabelField from './list-person-label-field/index.js'; | ||||||
| import listPersons from './list-persons/index.js'; | import listPersons from './list-persons/index.js'; | ||||||
|  | import listStages from './list-stages/index.js'; | ||||||
| import listUsers from './list-users/index.js'; | import listUsers from './list-users/index.js'; | ||||||
|  |  | ||||||
| export default [ | export default [ | ||||||
|   listActivityTypes, |   listActivityTypes, | ||||||
|   listCurrencies, |   listCurrencies, | ||||||
|   listDeals, |   listDeals, | ||||||
|   listLeads, |  | ||||||
|   listLeadLabels, |   listLeadLabels, | ||||||
|   listOrganizations, |   listLeads, | ||||||
|   listOrganizationLabelField, |   listOrganizationLabelField, | ||||||
|  |   listOrganizations, | ||||||
|   listPersonLabelField, |   listPersonLabelField, | ||||||
|   listPersons, |   listPersons, | ||||||
|  |   listStages, | ||||||
|   listUsers, |   listUsers, | ||||||
| ]; | ]; | ||||||
|   | |||||||
| @@ -0,0 +1,23 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List stages', | ||||||
|  |   key: 'listStages', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const stages = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get('/api/v1/stages'); | ||||||
|  |  | ||||||
|  |     if (data.data?.length) { | ||||||
|  |       for (const stage of data.data) { | ||||||
|  |         stages.data.push({ | ||||||
|  |           value: stage.id, | ||||||
|  |           name: stage.name, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return stages; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										3
									
								
								packages/backend/src/apps/webhook/actions/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								packages/backend/src/apps/webhook/actions/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | import respondWith from './respond-with/index.js'; | ||||||
|  |  | ||||||
|  | export default [respondWith]; | ||||||
| @@ -0,0 +1,69 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Respond with', | ||||||
|  |   key: 'respondWith', | ||||||
|  |   description: 'Respond with defined JSON body.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Status code', | ||||||
|  |       key: 'statusCode', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       variables: true, | ||||||
|  |       value: '200', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Headers', | ||||||
|  |       key: 'headers', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: 'Add or remove headers as needed', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Key', | ||||||
|  |           key: 'key', | ||||||
|  |           type: 'string', | ||||||
|  |           required: true, | ||||||
|  |           description: 'Header key', | ||||||
|  |           variables: true, | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Value', | ||||||
|  |           key: 'value', | ||||||
|  |           type: 'string', | ||||||
|  |           required: true, | ||||||
|  |           description: 'Header value', | ||||||
|  |           variables: true, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Body', | ||||||
|  |       key: 'body', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: 'The content of the response body.', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const statusCode = parseInt($.step.parameters.statusCode, 10); | ||||||
|  |     const body = $.step.parameters.body; | ||||||
|  |     const headers = $.step.parameters.headers.reduce((result, entry) => { | ||||||
|  |       return { | ||||||
|  |         ...result, | ||||||
|  |         [entry.key]: entry.value, | ||||||
|  |       }; | ||||||
|  |     }, {}); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: { | ||||||
|  |         headers, | ||||||
|  |         body, | ||||||
|  |         statusCode, | ||||||
|  |       }, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -1,4 +1,5 @@ | |||||||
| import defineApp from '../../helpers/define-app.js'; | import defineApp from '../../helpers/define-app.js'; | ||||||
|  | import actions from './actions/index.js'; | ||||||
| import triggers from './triggers/index.js'; | import triggers from './triggers/index.js'; | ||||||
|  |  | ||||||
| export default defineApp({ | export default defineApp({ | ||||||
| @@ -10,5 +11,6 @@ export default defineApp({ | |||||||
|   baseUrl: '', |   baseUrl: '', | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   primaryColor: '0059F7', |   primaryColor: '0059F7', | ||||||
|  |   actions, | ||||||
|   triggers, |   triggers, | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -7,7 +7,20 @@ export default defineTrigger({ | |||||||
|   key: 'catchRawWebhook', |   key: 'catchRawWebhook', | ||||||
|   type: 'webhook', |   type: 'webhook', | ||||||
|   showWebhookUrl: true, |   showWebhookUrl: true, | ||||||
|   description: 'Triggers when the webhook receives a request.', |   description: | ||||||
|  |     'Triggers (immediately if configured) when the webhook receives a request.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Wait until flow is done', | ||||||
|  |       key: 'workSynchronously', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Yes', value: true }, | ||||||
|  |         { label: 'No', value: false }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|   async run($) { |   async run($) { | ||||||
|     const dataItem = { |     const dataItem = { | ||||||
|   | |||||||
| @@ -0,0 +1,13 @@ | |||||||
|  | import User from '../../../../models/user.js'; | ||||||
|  | import { renderObject, renderError } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const { email, password } = request.body; | ||||||
|  |   const token = await User.authenticate(email, password); | ||||||
|  |  | ||||||
|  |   if (token) { | ||||||
|  |     return renderObject(response, { token }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   renderError(response, [{ general: ['Incorrect email or password.'] }]); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | import { describe, it, expect, beforeEach } from 'vitest'; | ||||||
|  | import request from 'supertest'; | ||||||
|  | import app from '../../../../app.js'; | ||||||
|  | import { createUser } from '../../../../../test/factories/user'; | ||||||
|  |  | ||||||
|  | describe('POST /api/v1/access-tokens', () => { | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     await createUser({ | ||||||
|  |       email: 'user@automatisch.io', | ||||||
|  |       password: 'password', | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the token data with correct credentials', async () => { | ||||||
|  |     const response = await request(app) | ||||||
|  |       .post('/api/v1/access-tokens') | ||||||
|  |       .send({ | ||||||
|  |         email: 'user@automatisch.io', | ||||||
|  |         password: 'password', | ||||||
|  |       }) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     expect(response.body.data.token.length).toBeGreaterThan(0); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return error with incorrect credentials', async () => { | ||||||
|  |     const response = await request(app) | ||||||
|  |       .post('/api/v1/access-tokens') | ||||||
|  |       .send({ | ||||||
|  |         email: 'incorrect@email.com', | ||||||
|  |         password: 'incorrectpassword', | ||||||
|  |       }) | ||||||
|  |       .expect(422); | ||||||
|  |  | ||||||
|  |     expect(response.body.errors.general).toEqual([ | ||||||
|  |       'Incorrect email or password.', | ||||||
|  |     ]); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -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); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -4,6 +4,7 @@ import AppAuthClient from '../../../../../models/app-auth-client.js'; | |||||||
| export default async (request, response) => { | export default async (request, response) => { | ||||||
|   const appAuthClient = await AppAuthClient.query() |   const appAuthClient = await AppAuthClient.query() | ||||||
|     .findById(request.params.appAuthClientId) |     .findById(request.params.appAuthClientId) | ||||||
|  |     .where({ app_key: request.params.appKey }) | ||||||
|     .throwIfNotFound(); |     .throwIfNotFound(); | ||||||
| 
 | 
 | ||||||
|   renderObject(response, appAuthClient); |   renderObject(response, appAuthClient); | ||||||
| @@ -0,0 +1,55 @@ | |||||||
|  | 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 { createRole } from '../../../../../../test/factories/role.js'; | ||||||
|  | import getAppAuthClientMock from '../../../../../../test/mocks/rest/api/v1/admin/apps/get-auth-client.js'; | ||||||
|  | import { createAppAuthClient } from '../../../../../../test/factories/app-auth-client.js'; | ||||||
|  | import * as license from '../../../../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/admin/apps/:appKey/auth-clients/:appAuthClientId', () => { | ||||||
|  |   let currentUser, adminRole, currentAppAuthClient, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
|  |  | ||||||
|  |     adminRole = await createRole({ key: 'admin' }); | ||||||
|  |     currentUser = await createUser({ roleId: adminRole.id }); | ||||||
|  |  | ||||||
|  |     currentAppAuthClient = await createAppAuthClient({ | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return specified app auth client', async () => { | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/admin/apps/deepl/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/admin/apps/deepl/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/apps/deepl/auth-clients/invalidAppAuthClientUUID') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(400); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | import { renderObject } from '../../../../../helpers/renderer.js'; | ||||||
|  | import AppAuthClient from '../../../../../models/app-auth-client.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const appAuthClients = await AppAuthClient.query() | ||||||
|  |     .where({ app_key: request.params.appKey }) | ||||||
|  |     .orderBy('created_at', 'desc'); | ||||||
|  |  | ||||||
|  |   renderObject(response, appAuthClients); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,44 @@ | |||||||
|  | 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 { createRole } from '../../../../../../test/factories/role.js'; | ||||||
|  | import getAuthClientsMock from '../../../../../../test/mocks/rest/api/v1/admin/apps/get-auth-clients.js'; | ||||||
|  | import { createAppAuthClient } from '../../../../../../test/factories/app-auth-client.js'; | ||||||
|  | import * as license from '../../../../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/admin/apps/:appKey/auth-clients', () => { | ||||||
|  |   let currentUser, adminRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
|  |  | ||||||
|  |     adminRole = await createRole({ key: 'admin' }); | ||||||
|  |     currentUser = await createUser({ roleId: adminRole.id }); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return specified app auth client info', async () => { | ||||||
|  |     const appAuthClientOne = await createAppAuthClient({ | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const appAuthClientTwo = await createAppAuthClient({ | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/admin/apps/deepl/auth-clients') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = getAuthClientsMock([ | ||||||
|  |       appAuthClientTwo, | ||||||
|  |       appAuthClientOne, | ||||||
|  |     ]); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | 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(); | ||||||
|  |  | ||||||
|  |   const roleMappings = await samlAuthProvider | ||||||
|  |     .$relatedQuery('samlAuthProvidersRoleMappings') | ||||||
|  |     .orderBy('remote_role_name', 'asc'); | ||||||
|  |  | ||||||
|  |   renderObject(response, roleMappings); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,51 @@ | |||||||
|  | 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 { createRoleMapping } from '../../../../../../test/factories/role-mapping.js'; | ||||||
|  | import getRoleMappingsMock from '../../../../../../test/mocks/rest/api/v1/admin/saml-auth-providers/get-role-mappings.ee.js'; | ||||||
|  | import * as license from '../../../../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/admin/saml-auth-providers/:samlAuthProviderId/role-mappings', () => { | ||||||
|  |   let roleMappingOne, roleMappingTwo, samlAuthProvider, currentUser, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     const role = await createRole({ key: 'admin' }); | ||||||
|  |     currentUser = await createUser({ roleId: role.id }); | ||||||
|  |  | ||||||
|  |     samlAuthProvider = await createSamlAuthProvider(); | ||||||
|  |  | ||||||
|  |     roleMappingOne = await createRoleMapping({ | ||||||
|  |       samlAuthProviderId: samlAuthProvider.id, | ||||||
|  |       remoteRoleName: 'Admin', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     roleMappingTwo = await createRoleMapping({ | ||||||
|  |       samlAuthProviderId: samlAuthProvider.id, | ||||||
|  |       remoteRoleName: 'User', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return role mappings', async () => { | ||||||
|  |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get( | ||||||
|  |         `/api/v1/admin/saml-auth-providers/${samlAuthProvider.id}/role-mappings` | ||||||
|  |       ) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getRoleMappingsMock([ | ||||||
|  |       roleMappingOne, | ||||||
|  |       roleMappingTwo, | ||||||
|  |     ]); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -6,5 +6,7 @@ export default async (request, response) => { | |||||||
|     .findById(request.params.samlAuthProviderId) |     .findById(request.params.samlAuthProviderId) | ||||||
|     .throwIfNotFound(); |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|   renderObject(response, samlAuthProvider); |   renderObject(response, samlAuthProvider, { | ||||||
|  |     serializer: 'AdminSamlAuthProvider', | ||||||
|  |   }); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -7,5 +7,7 @@ export default async (request, response) => { | |||||||
|     'desc' |     'desc' | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   renderObject(response, samlAuthProviders); |   renderObject(response, samlAuthProviders, { | ||||||
|  |     serializer: 'AdminSamlAuthProvider', | ||||||
|  |   }); | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import AppAuthClient from '../../../../models/app-auth-client.js'; | |||||||
| export default async (request, response) => { | export default async (request, response) => { | ||||||
|   const appAuthClient = await AppAuthClient.query() |   const appAuthClient = await AppAuthClient.query() | ||||||
|     .findById(request.params.appAuthClientId) |     .findById(request.params.appAuthClientId) | ||||||
|     .where({ active: true }) |     .where({ app_key: request.params.appKey, active: true }) | ||||||
|     .throwIfNotFound(); |     .throwIfNotFound(); | ||||||
| 
 | 
 | ||||||
|   renderObject(response, appAuthClient); |   renderObject(response, appAuthClient); | ||||||
| @@ -4,25 +4,27 @@ import Crypto from 'crypto'; | |||||||
| import app from '../../../../app.js'; | import app from '../../../../app.js'; | ||||||
| import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js'; | import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js'; | ||||||
| import { createUser } from '../../../../../test/factories/user.js'; | import { createUser } from '../../../../../test/factories/user.js'; | ||||||
| import getAppAuthClientMock from '../../../../../test/mocks/rest/api/v1/admin/get-app-auth-client.js'; | import getAppAuthClientMock from '../../../../../test/mocks/rest/api/v1/apps/get-auth-client.js'; | ||||||
| import { createAppAuthClient } from '../../../../../test/factories/app-auth-client.js'; | import { createAppAuthClient } from '../../../../../test/factories/app-auth-client.js'; | ||||||
| import * as license from '../../../../helpers/license.ee.js'; | import * as license from '../../../../helpers/license.ee.js'; | ||||||
| 
 | 
 | ||||||
| describe('GET /api/v1/app-auth-clients/:id', () => { | describe('GET /api/v1/apps/:appKey/auth-clients/:appAuthClientId', () => { | ||||||
|   let currentUser, currentAppAuthClient, token; |   let currentUser, currentAppAuthClient, token; | ||||||
| 
 | 
 | ||||||
|   beforeEach(async () => { |   beforeEach(async () => { | ||||||
|     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
| 
 | 
 | ||||||
|     currentUser = await createUser(); |     currentUser = await createUser(); | ||||||
|     currentAppAuthClient = await createAppAuthClient(); |     currentAppAuthClient = await createAppAuthClient({ | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
| 
 | 
 | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   it('should return specified app auth client info', async () => { |   it('should return specified app auth client', async () => { | ||||||
|     const response = await request(app) |     const response = await request(app) | ||||||
|       .get(`/api/v1/app-auth-clients/${currentAppAuthClient.id}`) |       .get(`/api/v1/apps/deepl/auth-clients/${currentAppAuthClient.id}`) | ||||||
|       .set('Authorization', token) |       .set('Authorization', token) | ||||||
|       .expect(200); |       .expect(200); | ||||||
| 
 | 
 | ||||||
| @@ -34,14 +36,14 @@ describe('GET /api/v1/app-auth-clients/:id', () => { | |||||||
|     const notExistingAppAuthClientUUID = Crypto.randomUUID(); |     const notExistingAppAuthClientUUID = Crypto.randomUUID(); | ||||||
| 
 | 
 | ||||||
|     await request(app) |     await request(app) | ||||||
|       .get(`/api/v1/app-auth-clients/${notExistingAppAuthClientUUID}`) |       .get(`/api/v1/apps/deepl/auth-clients/${notExistingAppAuthClientUUID}`) | ||||||
|       .set('Authorization', token) |       .set('Authorization', token) | ||||||
|       .expect(404); |       .expect(404); | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   it('should return bad request response for invalid UUID', async () => { |   it('should return bad request response for invalid UUID', async () => { | ||||||
|     await request(app) |     await request(app) | ||||||
|       .get('/api/v1/app-auth-clients/invalidAppAuthClientUUID') |       .get('/api/v1/apps/deepl/auth-clients/invalidAppAuthClientUUID') | ||||||
|       .set('Authorization', token) |       .set('Authorization', token) | ||||||
|       .expect(400); |       .expect(400); | ||||||
|   }); |   }); | ||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  | import AppAuthClient from '../../../../models/app-auth-client.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const appAuthClients = await AppAuthClient.query() | ||||||
|  |     .where({ app_key: request.params.appKey, active: true }) | ||||||
|  |     .orderBy('created_at', 'desc'); | ||||||
|  |  | ||||||
|  |   renderObject(response, appAuthClients); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,42 @@ | |||||||
|  | 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 getAuthClientsMock from '../../../../../test/mocks/rest/api/v1/apps/get-auth-clients.js'; | ||||||
|  | import { createAppAuthClient } from '../../../../../test/factories/app-auth-client.js'; | ||||||
|  | import * as license from '../../../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/apps/:appKey/auth-clients', () => { | ||||||
|  |   let currentUser, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
|  |  | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return specified app auth client info', async () => { | ||||||
|  |     const appAuthClientOne = await createAppAuthClient({ | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const appAuthClientTwo = await createAppAuthClient({ | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/apps/deepl/auth-clients') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = getAuthClientsMock([ | ||||||
|  |       appAuthClientTwo, | ||||||
|  |       appAuthClientOne, | ||||||
|  |     ]); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,15 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  | import AppConfig from '../../../../models/app-config.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const appConfig = await AppConfig.query() | ||||||
|  |     .withGraphFetched({ | ||||||
|  |       appAuthClients: true, | ||||||
|  |     }) | ||||||
|  |     .findOne({ | ||||||
|  |       key: request.params.appKey, | ||||||
|  |     }) | ||||||
|  |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   renderObject(response, appConfig); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,44 @@ | |||||||
|  | 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 getAppConfigMock from '../../../../../test/mocks/rest/api/v1/apps/get-config.js'; | ||||||
|  | import { createAppConfig } from '../../../../../test/factories/app-config.js'; | ||||||
|  | import * as license from '../../../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/apps/:appKey/config', () => { | ||||||
|  |   let currentUser, appConfig, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
|  |  | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |  | ||||||
|  |     appConfig = await createAppConfig({ | ||||||
|  |       key: 'deepl', | ||||||
|  |       allowCustomConnection: true, | ||||||
|  |       shared: true, | ||||||
|  |       disabled: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return specified app config info', async () => { | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/apps/${appConfig.key}/config`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = getAppConfigMock(appConfig); | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for not existing app key', async () => { | ||||||
|  |     await request(app) | ||||||
|  |       .get('/api/v1/apps/not-existing-app-key/config') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,24 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  | import App from '../../../../models/app.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const app = await App.findOneByKey(request.params.appKey); | ||||||
|  |  | ||||||
|  |   const connections = await request.currentUser.authorizedConnections | ||||||
|  |     .clone() | ||||||
|  |     .select('connections.*') | ||||||
|  |     .withGraphFetched({ | ||||||
|  |       appConfig: true, | ||||||
|  |       appAuthClient: true, | ||||||
|  |     }) | ||||||
|  |     .fullOuterJoinRelated('steps') | ||||||
|  |     .where({ | ||||||
|  |       'connections.key': app.key, | ||||||
|  |       'connections.draft': false, | ||||||
|  |     }) | ||||||
|  |     .countDistinct('steps.flow_id as flowCount') | ||||||
|  |     .groupBy('connections.id') | ||||||
|  |     .orderBy('created_at', 'desc'); | ||||||
|  |  | ||||||
|  |   renderObject(response, connections); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,101 @@ | |||||||
|  | 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.js'; | ||||||
|  | import { createUser } from '../../../../../test/factories/user.js'; | ||||||
|  | import { createConnection } from '../../../../../test/factories/connection.js'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission.js'; | ||||||
|  | import getConnectionsMock from '../../../../../test/mocks/rest/api/v1/apps/get-connections.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/apps/:appKey/connections', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the connections data of specified app for current user', async () => { | ||||||
|  |     const currentUserConnectionOne = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const currentUserConnectionTwo = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/apps/deepl/connections') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getConnectionsMock([ | ||||||
|  |       currentUserConnectionTwo, | ||||||
|  |       currentUserConnectionOne, | ||||||
|  |     ]); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the connections data of specified app for another user', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |  | ||||||
|  |     const anotherUserConnectionOne = await createConnection({ | ||||||
|  |       userId: anotherUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const anotherUserConnectionTwo = await createConnection({ | ||||||
|  |       userId: anotherUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/apps/deepl/connections') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getConnectionsMock([ | ||||||
|  |       anotherUserConnectionTwo, | ||||||
|  |       anotherUserConnectionOne, | ||||||
|  |     ]); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for invalid connection UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get('/api/v1/connections/invalid-connection-id/connections') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										23
									
								
								packages/backend/src/controllers/api/v1/apps/get-flows.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								packages/backend/src/controllers/api/v1/apps/get-flows.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  | import App from '../../../../models/app.js'; | ||||||
|  | import paginateRest from '../../../../helpers/pagination-rest.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const app = await App.findOneByKey(request.params.appKey); | ||||||
|  |  | ||||||
|  |   const flowsQuery = request.currentUser.authorizedFlows | ||||||
|  |     .clone() | ||||||
|  |     .joinRelated({ | ||||||
|  |       steps: true, | ||||||
|  |     }) | ||||||
|  |     .withGraphFetched({ | ||||||
|  |       steps: true, | ||||||
|  |     }) | ||||||
|  |     .where('steps.app_key', app.key) | ||||||
|  |     .orderBy('active', 'desc') | ||||||
|  |     .orderBy('updated_at', 'desc'); | ||||||
|  |  | ||||||
|  |   const flows = await paginateRest(flowsQuery, request.query.page); | ||||||
|  |  | ||||||
|  |   renderObject(response, flows); | ||||||
|  | }; | ||||||
							
								
								
									
										129
									
								
								packages/backend/src/controllers/api/v1/apps/get-flows.test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								packages/backend/src/controllers/api/v1/apps/get-flows.test.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | |||||||
|  | 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.js'; | ||||||
|  | import { createUser } from '../../../../../test/factories/user.js'; | ||||||
|  | import { createFlow } from '../../../../../test/factories/flow.js'; | ||||||
|  | import { createStep } from '../../../../../test/factories/step.js'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission.js'; | ||||||
|  | import getFlowsMock from '../../../../../test/mocks/rest/api/v1/flows/get-flows.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/apps/:appKey/flows', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the flows data of specified app for current user', async () => { | ||||||
|  |     const currentUserFlowOne = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowOne = await createStep({ | ||||||
|  |       flowId: currentUserFlowOne.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'webhook', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepFlowOne = await createStep({ | ||||||
|  |       flowId: currentUserFlowOne.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const currentUserFlowTwo = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: currentUserFlowTwo.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'github', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: currentUserFlowTwo.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/apps/webhook/flows') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getFlowsMock( | ||||||
|  |       [currentUserFlowOne], | ||||||
|  |       [triggerStepFlowOne, actionStepFlowOne] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the flows data of specified app for another user', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |     const anotherUserFlowOne = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowOne = await createStep({ | ||||||
|  |       flowId: anotherUserFlowOne.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'webhook', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepFlowOne = await createStep({ | ||||||
|  |       flowId: anotherUserFlowOne.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const anotherUserFlowTwo = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: anotherUserFlowTwo.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'github', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: anotherUserFlowTwo.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/apps/webhook/flows') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getFlowsMock( | ||||||
|  |       [anotherUserFlowOne], | ||||||
|  |       [triggerStepFlowOne, actionStepFlowOne] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for invalid app key', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get('/api/v1/apps/invalid-app-key/flows') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,24 @@ | |||||||
|  | import appConfig from '../../../../config/app.js'; | ||||||
|  | import Config from '../../../../models/config.js'; | ||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const defaultConfig = { | ||||||
|  |     disableNotificationsPage: appConfig.disableNotificationsPage, | ||||||
|  |     disableFavicon: appConfig.disableFavicon, | ||||||
|  |     additionalDrawerLink: appConfig.additionalDrawerLink, | ||||||
|  |     additionalDrawerLinkText: appConfig.additionalDrawerLinkText, | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   let config = await Config.query().orderBy('key', 'asc'); | ||||||
|  |  | ||||||
|  |   config = config.reduce((computedConfig, configEntry) => { | ||||||
|  |     const { key, value } = configEntry; | ||||||
|  |  | ||||||
|  |     computedConfig[key] = value?.data; | ||||||
|  |  | ||||||
|  |     return computedConfig; | ||||||
|  |   }, defaultConfig); | ||||||
|  |  | ||||||
|  |   renderObject(response, config); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,51 @@ | |||||||
|  | import { vi, expect, describe, it } from 'vitest'; | ||||||
|  | import request from 'supertest'; | ||||||
|  | import { createConfig } from '../../../../../test/factories/config.js'; | ||||||
|  | import app from '../../../../app.js'; | ||||||
|  | import configMock from '../../../../../test/mocks/rest/api/v1/automatisch/config.js'; | ||||||
|  | import * as license from '../../../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/automatisch/config', () => { | ||||||
|  |   it('should return Automatisch config', async () => { | ||||||
|  |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
|  |  | ||||||
|  |     const logoConfig = await createConfig({ | ||||||
|  |       key: 'logo.svgData', | ||||||
|  |       value: { data: '<svg>Sample</svg>' }, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const primaryDarkConfig = await createConfig({ | ||||||
|  |       key: 'palette.primary.dark', | ||||||
|  |       value: { data: '#001F52' }, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const primaryLightConfig = await createConfig({ | ||||||
|  |       key: 'palette.primary.light', | ||||||
|  |       value: { data: '#4286FF' }, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const primaryMainConfig = await createConfig({ | ||||||
|  |       key: 'palette.primary.main', | ||||||
|  |       value: { data: '#0059F7' }, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const titleConfig = await createConfig({ | ||||||
|  |       key: 'title', | ||||||
|  |       value: { data: 'Sample Title' }, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/automatisch/config') | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = configMock( | ||||||
|  |       logoConfig, | ||||||
|  |       primaryDarkConfig, | ||||||
|  |       primaryLightConfig, | ||||||
|  |       primaryMainConfig, | ||||||
|  |       titleConfig | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   let connection = await request.currentUser.authorizedConnections | ||||||
|  |     .clone() | ||||||
|  |     .findOne({ | ||||||
|  |       id: request.params.connectionId, | ||||||
|  |     }) | ||||||
|  |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   connection = await connection.testAndUpdateConnection(); | ||||||
|  |  | ||||||
|  |   renderObject(response, connection); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,123 @@ | |||||||
|  | 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.js'; | ||||||
|  | import { createUser } from '../../../../../test/factories/user.js'; | ||||||
|  | import { createConnection } from '../../../../../test/factories/connection.js'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission.js'; | ||||||
|  |  | ||||||
|  | describe('POST /api/v1/connections/:connectionId/test', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should update the connection as not verified for current user', async () => { | ||||||
|  |     const currentUserConnection = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       verified: true, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .post(`/api/v1/connections/${currentUserConnection.id}/test`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     expect(response.body.data.verified).toEqual(false); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should update the connection as not verified for another user', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |  | ||||||
|  |     const anotherUserConnection = await createConnection({ | ||||||
|  |       userId: anotherUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       verified: true, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .post(`/api/v1/connections/${anotherUserConnection.id}/test`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     expect(response.body.data.verified).toEqual(false); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for not existing connection UUID', async () => { | ||||||
|  |     const notExistingConnectionUUID = Crypto.randomUUID(); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .post(`/api/v1/connections/${notExistingConnectionUUID}/test`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return bad request response for invalid UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .post('/api/v1/connections/invalidConnectionUUID/test') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(400); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  | import paginateRest from '../../../../helpers/pagination-rest.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const flowsQuery = request.currentUser.authorizedFlows | ||||||
|  |     .clone() | ||||||
|  |     .joinRelated({ | ||||||
|  |       steps: true, | ||||||
|  |     }) | ||||||
|  |     .withGraphFetched({ | ||||||
|  |       steps: true, | ||||||
|  |     }) | ||||||
|  |     .where('steps.connection_id', request.params.connectionId) | ||||||
|  |     .orderBy('active', 'desc') | ||||||
|  |     .orderBy('updated_at', 'desc'); | ||||||
|  |  | ||||||
|  |   const flows = await paginateRest(flowsQuery, request.query.page); | ||||||
|  |  | ||||||
|  |   renderObject(response, flows); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,128 @@ | |||||||
|  | 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.js'; | ||||||
|  | import { createUser } from '../../../../../test/factories/user.js'; | ||||||
|  | import { createConnection } from '../../../../../test/factories/connection.js'; | ||||||
|  | import { createFlow } from '../../../../../test/factories/flow.js'; | ||||||
|  | import { createStep } from '../../../../../test/factories/step.js'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission.js'; | ||||||
|  | import getFlowsMock from '../../../../../test/mocks/rest/api/v1/flows/get-flows.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/connections/:connectionId/flows', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the flows data of specified connection for current user', async () => { | ||||||
|  |     const currentUserFlowOne = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const currentUserConnection = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'webhook', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowOne = await createStep({ | ||||||
|  |       flowId: currentUserFlowOne.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'webhook', | ||||||
|  |       connectionId: currentUserConnection.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepFlowOne = await createStep({ | ||||||
|  |       flowId: currentUserFlowOne.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const currentUserFlowTwo = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: currentUserFlowTwo.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'github', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: currentUserFlowTwo.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/connections/${currentUserConnection.id}/flows`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getFlowsMock( | ||||||
|  |       [currentUserFlowOne], | ||||||
|  |       [triggerStepFlowOne, actionStepFlowOne] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the flows data of specified connection for another user', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |     const anotherUserFlowOne = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const anotherUserConnection = await createConnection({ | ||||||
|  |       userId: anotherUser.id, | ||||||
|  |       key: 'webhook', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowOne = await createStep({ | ||||||
|  |       flowId: anotherUserFlowOne.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'webhook', | ||||||
|  |       connectionId: anotherUserConnection.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepFlowOne = await createStep({ | ||||||
|  |       flowId: anotherUserFlowOne.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const anotherUserFlowTwo = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: anotherUserFlowTwo.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |       appKey: 'github', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: anotherUserFlowTwo.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/connections/${anotherUserConnection.id}/flows`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getFlowsMock( | ||||||
|  |       [anotherUserFlowOne], | ||||||
|  |       [triggerStepFlowOne, actionStepFlowOne] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -2,6 +2,7 @@ import { renderObject } from '../../../../helpers/renderer.js'; | |||||||
|  |  | ||||||
| export default async (request, response) => { | export default async (request, response) => { | ||||||
|   const execution = await request.currentUser.authorizedExecutions |   const execution = await request.currentUser.authorizedExecutions | ||||||
|  |     .clone() | ||||||
|     .withGraphFetched({ |     .withGraphFetched({ | ||||||
|       flow: { |       flow: { | ||||||
|         steps: true, |         steps: true, | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ import paginateRest from '../../../../helpers/pagination-rest.js'; | |||||||
|  |  | ||||||
| export default async (request, response) => { | export default async (request, response) => { | ||||||
|   const executionsQuery = request.currentUser.authorizedExecutions |   const executionsQuery = request.currentUser.authorizedExecutions | ||||||
|  |     .clone() | ||||||
|     .withSoftDeleted() |     .withSoftDeleted() | ||||||
|     .orderBy('created_at', 'desc') |     .orderBy('created_at', 'desc') | ||||||
|     .withGraphFetched({ |     .withGraphFetched({ | ||||||
|   | |||||||
| @@ -42,9 +42,12 @@ describe('GET /api/v1/executions', () => { | |||||||
|  |  | ||||||
|     const currentUserExecutionTwo = await createExecution({ |     const currentUserExecutionTwo = await createExecution({ | ||||||
|       flowId: currentUserFlow.id, |       flowId: currentUserFlow.id, | ||||||
|       deletedAt: new Date().toISOString(), |  | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     await currentUserExecutionTwo | ||||||
|  |       .$query() | ||||||
|  |       .patchAndFetch({ deletedAt: new Date().toISOString() }); | ||||||
|  |  | ||||||
|     await createPermission({ |     await createPermission({ | ||||||
|       action: 'read', |       action: 'read', | ||||||
|       subject: 'Execution', |       subject: 'Execution', | ||||||
| @@ -87,9 +90,12 @@ describe('GET /api/v1/executions', () => { | |||||||
|  |  | ||||||
|     const anotherUserExecutionTwo = await createExecution({ |     const anotherUserExecutionTwo = await createExecution({ | ||||||
|       flowId: anotherUserFlow.id, |       flowId: anotherUserFlow.id, | ||||||
|       deletedAt: new Date().toISOString(), |  | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     await anotherUserExecutionTwo | ||||||
|  |       .$query() | ||||||
|  |       .patchAndFetch({ deletedAt: new Date().toISOString() }); | ||||||
|  |  | ||||||
|     await createPermission({ |     await createPermission({ | ||||||
|       action: 'read', |       action: 'read', | ||||||
|       subject: 'Execution', |       subject: 'Execution', | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import { renderObject } from '../../../../helpers/renderer.js'; | |||||||
|  |  | ||||||
| export default async (request, response) => { | export default async (request, response) => { | ||||||
|   const flow = await request.currentUser.authorizedFlows |   const flow = await request.currentUser.authorizedFlows | ||||||
|  |     .clone() | ||||||
|     .withGraphJoined({ steps: true }) |     .withGraphJoined({ steps: true }) | ||||||
|     .orderBy('steps.position', 'asc') |     .orderBy('steps.position', 'asc') | ||||||
|     .findOne({ 'flows.id': request.params.flowId }) |     .findOne({ 'flows.id': request.params.flowId }) | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								packages/backend/src/controllers/api/v1/flows/get-flows.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								packages/backend/src/controllers/api/v1/flows/get-flows.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  | import paginateRest from '../../../../helpers/pagination-rest.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const flowsQuery = request.currentUser.authorizedFlows | ||||||
|  |     .clone() | ||||||
|  |     .withGraphFetched({ | ||||||
|  |       steps: true, | ||||||
|  |     }) | ||||||
|  |     .where((builder) => { | ||||||
|  |       if (request.query.name) { | ||||||
|  |         builder.where('flows.name', 'ilike', `%${request.query.name}%`); | ||||||
|  |       } | ||||||
|  |     }) | ||||||
|  |     .orderBy('active', 'desc') | ||||||
|  |     .orderBy('updated_at', 'desc'); | ||||||
|  |  | ||||||
|  |   const flows = await paginateRest(flowsQuery, request.query.page); | ||||||
|  |  | ||||||
|  |   renderObject(response, flows); | ||||||
|  | }; | ||||||
							
								
								
									
										118
									
								
								packages/backend/src/controllers/api/v1/flows/get-flows.test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								packages/backend/src/controllers/api/v1/flows/get-flows.test.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | |||||||
|  | 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'; | ||||||
|  | import { createStep } from '../../../../../test/factories/step'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission'; | ||||||
|  | import getFlowsMock from '../../../../../test/mocks/rest/api/v1/flows/get-flows.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/flows', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the flows data of current user', async () => { | ||||||
|  |     const currentUserFlowOne = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowOne = await createStep({ | ||||||
|  |       flowId: currentUserFlowOne.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |     }); | ||||||
|  |     const actionStepFlowOne = await createStep({ | ||||||
|  |       flowId: currentUserFlowOne.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const currentUserFlowTwo = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowTwo = await createStep({ | ||||||
|  |       flowId: currentUserFlowTwo.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |     }); | ||||||
|  |     const actionStepFlowTwo = await createStep({ | ||||||
|  |       flowId: currentUserFlowTwo.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/flows') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getFlowsMock( | ||||||
|  |       [currentUserFlowTwo, currentUserFlowOne], | ||||||
|  |       [ | ||||||
|  |         triggerStepFlowOne, | ||||||
|  |         actionStepFlowOne, | ||||||
|  |         triggerStepFlowTwo, | ||||||
|  |         actionStepFlowTwo, | ||||||
|  |       ] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the flows data of another user', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |  | ||||||
|  |     const anotherUserFlowOne = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowOne = await createStep({ | ||||||
|  |       flowId: anotherUserFlowOne.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |     }); | ||||||
|  |     const actionStepFlowOne = await createStep({ | ||||||
|  |       flowId: anotherUserFlowOne.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const anotherUserFlowTwo = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStepFlowTwo = await createStep({ | ||||||
|  |       flowId: anotherUserFlowTwo.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |     }); | ||||||
|  |     const actionStepFlowTwo = await createStep({ | ||||||
|  |       flowId: anotherUserFlowTwo.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/flows') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getFlowsMock( | ||||||
|  |       [anotherUserFlowTwo, anotherUserFlowOne], | ||||||
|  |       [ | ||||||
|  |         triggerStepFlowOne, | ||||||
|  |         actionStepFlowOne, | ||||||
|  |         triggerStepFlowTwo, | ||||||
|  |         actionStepFlowTwo, | ||||||
|  |       ] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | 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() | ||||||
|  |     .where({ | ||||||
|  |       active: true, | ||||||
|  |     }) | ||||||
|  |     .orderBy('created_at', 'desc'); | ||||||
|  |  | ||||||
|  |   renderObject(response, samlAuthProviders); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | import { vi, describe, it, expect, beforeEach } from 'vitest'; | ||||||
|  | import request from 'supertest'; | ||||||
|  | import app from '../../../../app.js'; | ||||||
|  | import { createSamlAuthProvider } from '../../../../../test/factories/saml-auth-provider.ee.js'; | ||||||
|  | import getSamlAuthProvidersMock from '../../../../../test/mocks/rest/api/v1/saml-auth-providers/get-saml-auth-providers.js'; | ||||||
|  | import * as license from '../../../../helpers/license.ee.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/saml-auth-providers', () => { | ||||||
|  |   let samlAuthProviderOne, samlAuthProviderTwo; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     samlAuthProviderOne = await createSamlAuthProvider(); | ||||||
|  |     samlAuthProviderTwo = await createSamlAuthProvider(); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return saml auth providers', async () => { | ||||||
|  |     vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get('/api/v1/saml-auth-providers') | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getSamlAuthProvidersMock([ | ||||||
|  |       samlAuthProviderTwo, | ||||||
|  |       samlAuthProviderOne, | ||||||
|  |     ]); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const step = await request.currentUser.authorizedSteps | ||||||
|  |     .clone() | ||||||
|  |     .where('steps.id', request.params.stepId) | ||||||
|  |     .whereNotNull('steps.app_key') | ||||||
|  |     .whereNotNull('steps.connection_id') | ||||||
|  |     .first() | ||||||
|  |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   const dynamicData = await step.createDynamicData( | ||||||
|  |     request.body.dynamicDataKey, | ||||||
|  |     request.body.parameters | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   renderObject(response, dynamicData); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,244 @@ | |||||||
|  | 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 { createConnection } from '../../../../../test/factories/connection'; | ||||||
|  | import { createFlow } from '../../../../../test/factories/flow'; | ||||||
|  | import { createStep } from '../../../../../test/factories/step'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission'; | ||||||
|  | import listRepos from '../../../../apps/github/dynamic-data/list-repos/index.js'; | ||||||
|  | import HttpError from '../../../../errors/http.js'; | ||||||
|  |  | ||||||
|  | describe('POST /api/v1/steps/:stepId/dynamic-data', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   describe('should return dynamically created data', () => { | ||||||
|  |     let repositories; | ||||||
|  |  | ||||||
|  |     beforeEach(async () => { | ||||||
|  |       repositories = [ | ||||||
|  |         { | ||||||
|  |           value: 'automatisch/automatisch', | ||||||
|  |           name: 'automatisch/automatisch', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           value: 'automatisch/sample', | ||||||
|  |           name: 'automatisch/sample', | ||||||
|  |         }, | ||||||
|  |       ]; | ||||||
|  |  | ||||||
|  |       vi.spyOn(listRepos, 'run').mockImplementation(async () => { | ||||||
|  |         return { | ||||||
|  |           data: repositories, | ||||||
|  |         }; | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('of the current users step', async () => { | ||||||
|  |       const currentUserFlow = await createFlow({ userId: currentUser.id }); | ||||||
|  |       const connection = await createConnection({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |       const actionStep = await createStep({ | ||||||
|  |         flowId: currentUserFlow.id, | ||||||
|  |         connectionId: connection.id, | ||||||
|  |         type: 'action', | ||||||
|  |         appKey: 'github', | ||||||
|  |         key: 'createIssue', | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       await createPermission({ | ||||||
|  |         action: 'read', | ||||||
|  |         subject: 'Flow', | ||||||
|  |         roleId: currentUserRole.id, | ||||||
|  |         conditions: ['isCreator'], | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       await createPermission({ | ||||||
|  |         action: 'update', | ||||||
|  |         subject: 'Flow', | ||||||
|  |         roleId: currentUserRole.id, | ||||||
|  |         conditions: ['isCreator'], | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const response = await request(app) | ||||||
|  |         .post(`/api/v1/steps/${actionStep.id}/dynamic-data`) | ||||||
|  |         .set('Authorization', token) | ||||||
|  |         .send({ | ||||||
|  |           dynamicDataKey: 'listRepos', | ||||||
|  |           parameters: {}, | ||||||
|  |         }) | ||||||
|  |         .expect(200); | ||||||
|  |  | ||||||
|  |       expect(response.body.data).toEqual(repositories); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('of the another users step', async () => { | ||||||
|  |       const anotherUser = await createUser(); | ||||||
|  |       const anotherUserFlow = await createFlow({ userId: anotherUser.id }); | ||||||
|  |       const connection = await createConnection({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |       const actionStep = await createStep({ | ||||||
|  |         flowId: anotherUserFlow.id, | ||||||
|  |         connectionId: connection.id, | ||||||
|  |         type: 'action', | ||||||
|  |         appKey: 'github', | ||||||
|  |         key: 'createIssue', | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       await createPermission({ | ||||||
|  |         action: 'read', | ||||||
|  |         subject: 'Flow', | ||||||
|  |         roleId: currentUserRole.id, | ||||||
|  |         conditions: [], | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       await createPermission({ | ||||||
|  |         action: 'update', | ||||||
|  |         subject: 'Flow', | ||||||
|  |         roleId: currentUserRole.id, | ||||||
|  |         conditions: [], | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const response = await request(app) | ||||||
|  |         .post(`/api/v1/steps/${actionStep.id}/dynamic-data`) | ||||||
|  |         .set('Authorization', token) | ||||||
|  |         .send({ | ||||||
|  |           dynamicDataKey: 'listRepos', | ||||||
|  |           parameters: {}, | ||||||
|  |         }) | ||||||
|  |         .expect(200); | ||||||
|  |  | ||||||
|  |       expect(response.body.data).toEqual(repositories); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   describe('should return error for dynamically created data', () => { | ||||||
|  |     let errors; | ||||||
|  |  | ||||||
|  |     beforeEach(async () => { | ||||||
|  |       errors = { | ||||||
|  |         message: 'Not Found', | ||||||
|  |         documentation_url: | ||||||
|  |           'https://docs.github.com/rest/users/users#get-a-user', | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       vi.spyOn(listRepos, 'run').mockImplementation(async () => { | ||||||
|  |         throw new HttpError({ message: errors }); | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('of the current users step', async () => { | ||||||
|  |       const currentUserFlow = await createFlow({ userId: currentUser.id }); | ||||||
|  |       const connection = await createConnection({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |       const actionStep = await createStep({ | ||||||
|  |         flowId: currentUserFlow.id, | ||||||
|  |         connectionId: connection.id, | ||||||
|  |         type: 'action', | ||||||
|  |         appKey: 'github', | ||||||
|  |         key: 'createIssue', | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       await createPermission({ | ||||||
|  |         action: 'read', | ||||||
|  |         subject: 'Flow', | ||||||
|  |         roleId: currentUserRole.id, | ||||||
|  |         conditions: ['isCreator'], | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       await createPermission({ | ||||||
|  |         action: 'update', | ||||||
|  |         subject: 'Flow', | ||||||
|  |         roleId: currentUserRole.id, | ||||||
|  |         conditions: ['isCreator'], | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const response = await request(app) | ||||||
|  |         .post(`/api/v1/steps/${actionStep.id}/dynamic-data`) | ||||||
|  |         .set('Authorization', token) | ||||||
|  |         .send({ | ||||||
|  |           dynamicDataKey: 'listRepos', | ||||||
|  |           parameters: {}, | ||||||
|  |         }) | ||||||
|  |         .expect(200); | ||||||
|  |  | ||||||
|  |       expect(response.body.errors).toEqual(errors); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for not existing step UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const notExistingStepUUID = Crypto.randomUUID(); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/steps/${notExistingStepUUID}/dynamic-data`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for existing step UUID without app key', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const step = await createStep({ appKey: null }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/steps/${step.id}/dynamic-data`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return bad request response for invalid UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .post('/api/v1/steps/invalidStepUUID/dynamic-fields') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(400); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,17 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const step = await request.currentUser.authorizedSteps | ||||||
|  |     .clone() | ||||||
|  |     .where('steps.id', request.params.stepId) | ||||||
|  |     .whereNotNull('steps.app_key') | ||||||
|  |     .first() | ||||||
|  |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   const dynamicFields = await step.createDynamicFields( | ||||||
|  |     request.body.dynamicFieldsKey, | ||||||
|  |     request.body.parameters | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   renderObject(response, dynamicFields); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,169 @@ | |||||||
|  | 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 createDynamicFieldsMock from '../../../../../test/mocks/rest/api/v1/steps/create-dynamic-fields'; | ||||||
|  |  | ||||||
|  | describe('POST /api/v1/steps/:stepId/dynamic-fields', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return dynamically created fields of the current users step', async () => { | ||||||
|  |     const currentUserflow = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const actionStep = await createStep({ | ||||||
|  |       flowId: currentUserflow.id, | ||||||
|  |       type: 'action', | ||||||
|  |       appKey: 'slack', | ||||||
|  |       key: 'sendMessageToChannel', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .post(`/api/v1/steps/${actionStep.id}/dynamic-fields`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .send({ | ||||||
|  |         dynamicFieldsKey: 'listFieldsAfterSendAsBot', | ||||||
|  |         parameters: { | ||||||
|  |           sendAsBot: true, | ||||||
|  |         }, | ||||||
|  |       }) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await createDynamicFieldsMock(); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return dynamically created fields of the another users step', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |     const anotherUserflow = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const actionStep = await createStep({ | ||||||
|  |       flowId: anotherUserflow.id, | ||||||
|  |       type: 'action', | ||||||
|  |       appKey: 'slack', | ||||||
|  |       key: 'sendMessageToChannel', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .post(`/api/v1/steps/${actionStep.id}/dynamic-fields`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .send({ | ||||||
|  |         dynamicFieldsKey: 'listFieldsAfterSendAsBot', | ||||||
|  |         parameters: { | ||||||
|  |           sendAsBot: true, | ||||||
|  |         }, | ||||||
|  |       }) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await createDynamicFieldsMock(); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for not existing step UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const notExistingStepUUID = Crypto.randomUUID(); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/steps/${notExistingStepUUID}/dynamic-fields`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for existing step UUID without app key', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const step = await createStep({ appKey: null }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/steps/${step.id}/dynamic-fields`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return bad request response for invalid UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .post('/api/v1/steps/invalidStepUUID/dynamic-fields') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(400); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const step = await request.currentUser.authorizedSteps | ||||||
|  |     .findById(request.params.stepId) | ||||||
|  |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   const connection = await step.$relatedQuery('connection').throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   renderObject(response, connection); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,121 @@ | |||||||
|  | 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 { createConnection } from '../../../../../test/factories/connection'; | ||||||
|  | import { createFlow } from '../../../../../test/factories/flow'; | ||||||
|  | import { createStep } from '../../../../../test/factories/step'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission'; | ||||||
|  | import getConnectionMock from '../../../../../test/mocks/rest/api/v1/steps/get-connection'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/steps/:stepId/connection', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the current user connection data of specified step', async () => { | ||||||
|  |     const currentUserflow = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const currentUserConnection = await createConnection(); | ||||||
|  |     const triggerStep = await createStep({ | ||||||
|  |       flowId: currentUserflow.id, | ||||||
|  |       connectionId: currentUserConnection.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/steps/${triggerStep.id}/connection`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getConnectionMock(currentUserConnection); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the current user connection data of specified step', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |     const anotherUserFlow = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const anotherUserConnection = await createConnection(); | ||||||
|  |     const triggerStep = await createStep({ | ||||||
|  |       flowId: anotherUserFlow.id, | ||||||
|  |       connectionId: anotherUserConnection.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/steps/${triggerStep.id}/connection`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getConnectionMock(anotherUserConnection); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for not existing step without connection', async () => { | ||||||
|  |     const stepWithoutConnection = await createStep(); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/steps/${stepWithoutConnection.id}/connection`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for not existing step UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const notExistingFlowUUID = Crypto.randomUUID(); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/steps/${notExistingFlowUUID}/connection`) | ||||||
|  |       .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/steps/invalidFlowUUID/connection') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(400); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,27 @@ | |||||||
|  | import { ref } from 'objection'; | ||||||
|  | import ExecutionStep from '../../../../models/execution-step.js'; | ||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const step = await request.currentUser.authorizedSteps | ||||||
|  |     .clone() | ||||||
|  |     .findOne({ 'steps.id': request.params.stepId }) | ||||||
|  |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   const previousSteps = await request.currentUser.authorizedSteps | ||||||
|  |     .clone() | ||||||
|  |     .withGraphJoined('executionSteps') | ||||||
|  |     .where('flow_id', '=', step.flowId) | ||||||
|  |     .andWhere('position', '<', step.position) | ||||||
|  |     .andWhere( | ||||||
|  |       'executionSteps.created_at', | ||||||
|  |       '=', | ||||||
|  |       ExecutionStep.query() | ||||||
|  |         .max('created_at') | ||||||
|  |         .where('step_id', '=', ref('steps.id')) | ||||||
|  |         .andWhere('status', 'success') | ||||||
|  |     ) | ||||||
|  |     .orderBy('steps.position', 'asc'); | ||||||
|  |  | ||||||
|  |   renderObject(response, previousSteps); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,173 @@ | |||||||
|  | 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 { createExecutionStep } from '../../../../../test/factories/execution-step.js'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission'; | ||||||
|  | import getPreviousStepsMock from '../../../../../test/mocks/rest/api/v1/steps/get-previous-steps'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/steps/:stepId/previous-steps', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUser = await createUser(); | ||||||
|  |     currentUserRole = await currentUser.$relatedQuery('role'); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the previous steps of the specified step of the current user', async () => { | ||||||
|  |     const currentUserflow = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStep = await createStep({ | ||||||
|  |       flowId: currentUserflow.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepOne = await createStep({ | ||||||
|  |       flowId: currentUserflow.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepTwo = await createStep({ | ||||||
|  |       flowId: currentUserflow.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const executionStepOne = await createExecutionStep({ | ||||||
|  |       stepId: triggerStep.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const executionStepTwo = await createExecutionStep({ | ||||||
|  |       stepId: actionStepOne.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/steps/${actionStepTwo.id}/previous-steps`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getPreviousStepsMock( | ||||||
|  |       [triggerStep, actionStepOne], | ||||||
|  |       [executionStepOne, executionStepTwo] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return the previous steps of the specified step of another user', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |     const anotherUserFlow = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const triggerStep = await createStep({ | ||||||
|  |       flowId: anotherUserFlow.id, | ||||||
|  |       type: 'trigger', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepOne = await createStep({ | ||||||
|  |       flowId: anotherUserFlow.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const actionStepTwo = await createStep({ | ||||||
|  |       flowId: anotherUserFlow.id, | ||||||
|  |       type: 'action', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const executionStepOne = await createExecutionStep({ | ||||||
|  |       stepId: triggerStep.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const executionStepTwo = await createExecutionStep({ | ||||||
|  |       stepId: actionStepOne.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/steps/${actionStepTwo.id}/previous-steps`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = await getPreviousStepsMock( | ||||||
|  |       [triggerStep, actionStepOne], | ||||||
|  |       [executionStepOne, executionStepTwo] | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response for not existing step UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const notExistingFlowUUID = Crypto.randomUUID(); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/steps/${notExistingFlowUUID}/previous-steps`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return bad request response for invalid UUID', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'update', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get('/api/v1/steps/invalidFlowUUID/previous-steps') | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(400); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,7 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const apps = await request.currentUser.getApps(request.query.name); | ||||||
|  |  | ||||||
|  |   renderObject(response, apps, { serializer: 'App' }); | ||||||
|  | }; | ||||||
							
								
								
									
										210
									
								
								packages/backend/src/controllers/api/v1/users/get-apps.test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										210
									
								
								packages/backend/src/controllers/api/v1/users/get-apps.test.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,210 @@ | |||||||
|  | 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 { createRole } from '../../../../../test/factories/role'; | ||||||
|  | import { createUser } from '../../../../../test/factories/user'; | ||||||
|  | import { createPermission } from '../../../../../test/factories/permission.js'; | ||||||
|  | import { createFlow } from '../../../../../test/factories/flow.js'; | ||||||
|  | import { createStep } from '../../../../../test/factories/step.js'; | ||||||
|  | import { createConnection } from '../../../../../test/factories/connection.js'; | ||||||
|  | import getAppsMock from '../../../../../test/mocks/rest/api/v1/users/get-apps.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/users/:userId/apps', () => { | ||||||
|  |   let currentUser, currentUserRole, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     currentUserRole = await createRole(); | ||||||
|  |     currentUser = await createUser({ roleId: currentUserRole.id }); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return all apps of the current user', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowOne = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: flowOne.id, | ||||||
|  |       appKey: 'webhook', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowOneActionStepConnection = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       connectionId: flowOneActionStepConnection.id, | ||||||
|  |       flowId: flowOne.id, | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowTwo = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const flowTwoTriggerStepConnection = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'github', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       connectionId: flowTwoTriggerStepConnection.id, | ||||||
|  |       flowId: flowTwo.id, | ||||||
|  |       appKey: 'github', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: flowTwo.id, | ||||||
|  |       appKey: 'slack', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/users/${currentUser.id}/apps`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = getAppsMock(); | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return all apps of the another user', async () => { | ||||||
|  |     const anotherUser = await createUser(); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: [], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowOne = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: flowOne.id, | ||||||
|  |       appKey: 'webhook', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowOneActionStepConnection = await createConnection({ | ||||||
|  |       userId: anotherUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       connectionId: flowOneActionStepConnection.id, | ||||||
|  |       flowId: flowOne.id, | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowTwo = await createFlow({ userId: anotherUser.id }); | ||||||
|  |  | ||||||
|  |     const flowTwoTriggerStepConnection = await createConnection({ | ||||||
|  |       userId: anotherUser.id, | ||||||
|  |       key: 'github', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       connectionId: flowTwoTriggerStepConnection.id, | ||||||
|  |       flowId: flowTwo.id, | ||||||
|  |       appKey: 'github', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: flowTwo.id, | ||||||
|  |       appKey: 'slack', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/users/${currentUser.id}/apps`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = getAppsMock(); | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return specified app of the current user', async () => { | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Flow', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createPermission({ | ||||||
|  |       action: 'read', | ||||||
|  |       subject: 'Connection', | ||||||
|  |       roleId: currentUserRole.id, | ||||||
|  |       conditions: ['isCreator'], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowOne = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: flowOne.id, | ||||||
|  |       appKey: 'webhook', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowOneActionStepConnection = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'deepl', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       connectionId: flowOneActionStepConnection.id, | ||||||
|  |       flowId: flowOne.id, | ||||||
|  |       appKey: 'deepl', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const flowTwo = await createFlow({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     const flowTwoTriggerStepConnection = await createConnection({ | ||||||
|  |       userId: currentUser.id, | ||||||
|  |       key: 'github', | ||||||
|  |       draft: false, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       connectionId: flowTwoTriggerStepConnection.id, | ||||||
|  |       flowId: flowTwo.id, | ||||||
|  |       appKey: 'github', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     await createStep({ | ||||||
|  |       flowId: flowTwo.id, | ||||||
|  |       appKey: 'slack', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/users/${currentUser.id}/apps?name=deepl`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     expect(response.body.data.length).toEqual(1); | ||||||
|  |     expect(response.body.data[0].key).toEqual('deepl'); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,7 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const planAndUsage = await request.currentUser.getPlanAndUsage(); | ||||||
|  |  | ||||||
|  |   renderObject(response, planAndUsage); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,68 @@ | |||||||
|  | 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 { createSubscription } from '../../../../../test/factories/subscription.js'; | ||||||
|  | import { createUsageData } from '../../../../../test/factories/usage-data.js'; | ||||||
|  | import appConfig from '../../../../config/app.js'; | ||||||
|  | import { DateTime } from 'luxon'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/users/:userId/plan-and-usage', () => { | ||||||
|  |   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); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return free trial plan and usage data', async () => { | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/users/${user.id}/plan-and-usage`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedResponseData = { | ||||||
|  |       plan: { | ||||||
|  |         id: null, | ||||||
|  |         limit: null, | ||||||
|  |         name: 'Free Trial', | ||||||
|  |       }, | ||||||
|  |       usage: { | ||||||
|  |         task: 0, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     expect(response.body.data).toEqual(expectedResponseData); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return current plan and usage data', async () => { | ||||||
|  |     await createSubscription({ userId: user.id }); | ||||||
|  |  | ||||||
|  |     await createUsageData({ | ||||||
|  |       userId: user.id, | ||||||
|  |       consumedTaskCount: 1234, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/users/${user.id}/plan-and-usage`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedResponseData = { | ||||||
|  |       plan: { | ||||||
|  |         id: '47384', | ||||||
|  |         limit: '10,000', | ||||||
|  |         name: '10k - monthly', | ||||||
|  |       }, | ||||||
|  |       usage: { | ||||||
|  |         task: 1234, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     expect(response.body.data).toEqual(expectedResponseData); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | import { renderObject } from '../../../../helpers/renderer.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const subscription = await request.currentUser | ||||||
|  |     .$relatedQuery('currentSubscription') | ||||||
|  |     .throwIfNotFound(); | ||||||
|  |  | ||||||
|  |   renderObject(response, subscription); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,51 @@ | |||||||
|  | import { vi, describe, it, expect, beforeEach } from 'vitest'; | ||||||
|  | import request from 'supertest'; | ||||||
|  | import appConfig from '../../../../config/app.js'; | ||||||
|  | import app from '../../../../app.js'; | ||||||
|  | import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id'; | ||||||
|  | import { createRole } from '../../../../../test/factories/role'; | ||||||
|  | import { createUser } from '../../../../../test/factories/user'; | ||||||
|  | import { createSubscription } from '../../../../../test/factories/subscription.js'; | ||||||
|  | import getSubscriptionMock from '../../../../../test/mocks/rest/api/v1/users/get-subscription.js'; | ||||||
|  |  | ||||||
|  | describe('GET /api/v1/users/:userId/subscription', () => { | ||||||
|  |   let currentUser, role, subscription, token; | ||||||
|  |  | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     vi.spyOn(appConfig, 'isCloud', 'get').mockReturnValue(true); | ||||||
|  |  | ||||||
|  |     role = await createRole(); | ||||||
|  |  | ||||||
|  |     currentUser = await createUser({ | ||||||
|  |       roleId: role.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     subscription = await createSubscription({ userId: currentUser.id }); | ||||||
|  |  | ||||||
|  |     token = createAuthTokenByUserId(currentUser.id); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return subscription info of the current user', async () => { | ||||||
|  |     const response = await request(app) | ||||||
|  |       .get(`/api/v1/users/${currentUser.id}/subscription`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(200); | ||||||
|  |  | ||||||
|  |     const expectedPayload = getSubscriptionMock(subscription); | ||||||
|  |  | ||||||
|  |     expect(response.body).toEqual(expectedPayload); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   it('should return not found response if there is no current subscription', async () => { | ||||||
|  |     const userWithoutSubscription = await createUser({ | ||||||
|  |       roleId: role.id, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const token = createAuthTokenByUserId(userWithoutSubscription.id); | ||||||
|  |  | ||||||
|  |     await request(app) | ||||||
|  |       .get(`/api/v1/users/${userWithoutSubscription.id}/subscription`) | ||||||
|  |       .set('Authorization', token) | ||||||
|  |       .expect(404); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
| @@ -0,0 +1,29 @@ | |||||||
|  | import Flow from '../../models/flow.js'; | ||||||
|  | import logger from '../../helpers/logger.js'; | ||||||
|  | import handlerSync from '../../helpers/webhook-handler-sync.js'; | ||||||
|  |  | ||||||
|  | export default async (request, response) => { | ||||||
|  |   const computedRequestPayload = { | ||||||
|  |     headers: request.headers, | ||||||
|  |     body: request.body, | ||||||
|  |     query: request.query, | ||||||
|  |     params: request.params, | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   logger.debug(`Handling incoming webhook request at ${request.originalUrl}.`); | ||||||
|  |   logger.debug(JSON.stringify(computedRequestPayload, null, 2)); | ||||||
|  |  | ||||||
|  |   const flowId = request.params.flowId; | ||||||
|  |   const flow = await Flow.query().findById(flowId).throwIfNotFound(); | ||||||
|  |   const triggerStep = await flow.getTriggerStep(); | ||||||
|  |  | ||||||
|  |   if (triggerStep.appKey !== 'webhook') { | ||||||
|  |     const connection = await triggerStep.$relatedQuery('connection'); | ||||||
|  |  | ||||||
|  |     if (!(await connection.verifyWebhook(request))) { | ||||||
|  |       return response.sendStatus(401); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   await handlerSync(flowId, request, response); | ||||||
|  | }; | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | export async function up(knex) { | ||||||
|  |   await knex.schema.table('app_auth_clients', (table) => { | ||||||
|  |     table.string('app_key'); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export async function down(knex) { | ||||||
|  |   await knex.schema.table('app_auth_clients', (table) => { | ||||||
|  |     table.dropColumn('app_key'); | ||||||
|  |   }); | ||||||
|  | } | ||||||
| @@ -0,0 +1,17 @@ | |||||||
|  | export async function up(knex) { | ||||||
|  |   const appAuthClients = await knex('app_auth_clients').select('*'); | ||||||
|  |  | ||||||
|  |   for (const appAuthClient of appAuthClients) { | ||||||
|  |     const appConfig = await knex('app_configs') | ||||||
|  |       .where('id', appAuthClient.app_config_id) | ||||||
|  |       .first(); | ||||||
|  |  | ||||||
|  |     await knex('app_auth_clients') | ||||||
|  |       .where('id', appAuthClient.id) | ||||||
|  |       .update({ app_key: appConfig.key }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export async function down() { | ||||||
|  |   // void | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | export async function up(knex) { | ||||||
|  |   await knex.schema.table('app_auth_clients', (table) => { | ||||||
|  |     table.dropColumn('app_config_id'); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export async function down(knex) { | ||||||
|  |   await knex.schema.table('app_auth_clients', (table) => { | ||||||
|  |     table.uuid('app_config_id').references('id').inTable('app_configs'); | ||||||
|  |   }); | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | export async function up(knex) { | ||||||
|  |   await knex.schema.table('app_auth_clients', (table) => { | ||||||
|  |     table.string('app_key').notNullable().alter(); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export async function down(knex) { | ||||||
|  |   await knex.schema.table('app_auth_clients', (table) => { | ||||||
|  |     table.string('app_key').nullable().alter(); | ||||||
|  |   }); | ||||||
|  | } | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| import Step from '../../models/flow.js'; | import Step from '../../models/step.js'; | ||||||
|  |  | ||||||
| const deleteStep = async (_parent, params, context) => { | const deleteStep = async (_parent, params, context) => { | ||||||
|   const conditions = context.currentUser.can('update', 'Flow'); |   const conditions = context.currentUser.can('update', 'Flow'); | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| import AppAuthClient from '../../models/app-auth-client.js'; |  | ||||||
|  |  | ||||||
| const getAppAuthClient = async (_parent, params, context) => { |  | ||||||
|   let canSeeAllClients = false; |  | ||||||
|   try { |  | ||||||
|     context.currentUser.can('read', 'App'); |  | ||||||
|  |  | ||||||
|     canSeeAllClients = true; |  | ||||||
|   } catch { |  | ||||||
|     // void |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const appAuthClient = AppAuthClient.query() |  | ||||||
|     .findById(params.id) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   if (!canSeeAllClients) { |  | ||||||
|     appAuthClient.where({ active: true }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return await appAuthClient; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getAppAuthClient; |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| import AppConfig from '../../models/app-config.js'; |  | ||||||
|  |  | ||||||
| const getAppAuthClients = async (_parent, params, context) => { |  | ||||||
|   let canSeeAllClients = false; |  | ||||||
|   try { |  | ||||||
|     context.currentUser.can('read', 'App'); |  | ||||||
|  |  | ||||||
|     canSeeAllClients = true; |  | ||||||
|   } catch { |  | ||||||
|     // void |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const appConfig = await AppConfig.query() |  | ||||||
|     .findOne({ |  | ||||||
|       key: params.appKey, |  | ||||||
|     }) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   const appAuthClients = appConfig |  | ||||||
|     .$relatedQuery('appAuthClients') |  | ||||||
|     .where({ active: params.active }) |  | ||||||
|     .skipUndefined(); |  | ||||||
|  |  | ||||||
|   if (!canSeeAllClients) { |  | ||||||
|     appAuthClients.where({ |  | ||||||
|       active: true, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return await appAuthClients; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getAppAuthClients; |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| import AppConfig from '../../models/app-config.js'; |  | ||||||
|  |  | ||||||
| const getAppConfig = async (_parent, params, context) => { |  | ||||||
|   context.currentUser.can('create', 'Connection'); |  | ||||||
|  |  | ||||||
|   const appConfig = await AppConfig.query() |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       appAuthClients: true, |  | ||||||
|     }) |  | ||||||
|     .findOne({ |  | ||||||
|       key: params.key, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|   return appConfig; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getAppConfig; |  | ||||||
| @@ -1,41 +0,0 @@ | |||||||
| import App from '../../models/app.js'; |  | ||||||
| import Connection from '../../models/connection.js'; |  | ||||||
|  |  | ||||||
| const getApp = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('read', 'Connection'); |  | ||||||
|  |  | ||||||
|   const userConnections = context.currentUser.$relatedQuery('connections'); |  | ||||||
|   const allConnections = Connection.query(); |  | ||||||
|   const connectionBaseQuery = conditions.isCreator |  | ||||||
|     ? userConnections |  | ||||||
|     : allConnections; |  | ||||||
|  |  | ||||||
|   const app = await App.findOneByKey(params.key); |  | ||||||
|  |  | ||||||
|   if (context.currentUser) { |  | ||||||
|     const connections = await connectionBaseQuery |  | ||||||
|       .clone() |  | ||||||
|       .select('connections.*') |  | ||||||
|       .withGraphFetched({ |  | ||||||
|         appConfig: true, |  | ||||||
|         appAuthClient: true, |  | ||||||
|       }) |  | ||||||
|       .fullOuterJoinRelated('steps') |  | ||||||
|       .where({ |  | ||||||
|         'connections.key': params.key, |  | ||||||
|         'connections.draft': false, |  | ||||||
|       }) |  | ||||||
|       .countDistinct('steps.flow_id as flowCount') |  | ||||||
|       .groupBy('connections.id') |  | ||||||
|       .orderBy('created_at', 'desc'); |  | ||||||
|  |  | ||||||
|     return { |  | ||||||
|       ...app, |  | ||||||
|       connections, |  | ||||||
|     }; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return app; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getApp; |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| import App from '../../models/app.js'; |  | ||||||
|  |  | ||||||
| const getApps = async (_parent, params) => { |  | ||||||
|   const apps = await App.findAll(params.name); |  | ||||||
|  |  | ||||||
|   if (params.onlyWithTriggers) { |  | ||||||
|     return apps.filter((app) => app.triggers?.length); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (params.onlyWithActions) { |  | ||||||
|     return apps.filter((app) => app.actions?.length); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return apps; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getApps; |  | ||||||
| @@ -1,101 +0,0 @@ | |||||||
| import { DateTime } from 'luxon'; |  | ||||||
| import Billing from '../../helpers/billing/index.ee.js'; |  | ||||||
| import ExecutionStep from '../../models/execution-step.js'; |  | ||||||
|  |  | ||||||
| const getBillingAndUsage = async (_parent, _params, context) => { |  | ||||||
|   const persistedSubscription = await context.currentUser.$relatedQuery( |  | ||||||
|     'currentSubscription' |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   const subscription = persistedSubscription |  | ||||||
|     ? paidSubscription(persistedSubscription) |  | ||||||
|     : freeTrialSubscription(); |  | ||||||
|  |  | ||||||
|   return { |  | ||||||
|     subscription, |  | ||||||
|     usage: { |  | ||||||
|       task: executionStepCount(context), |  | ||||||
|     }, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const paidSubscription = (subscription) => { |  | ||||||
|   const currentPlan = Billing.paddlePlans.find( |  | ||||||
|     (plan) => plan.productId === subscription.paddlePlanId |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   return { |  | ||||||
|     status: subscription.status, |  | ||||||
|     monthlyQuota: { |  | ||||||
|       title: currentPlan.limit, |  | ||||||
|       action: { |  | ||||||
|         type: 'link', |  | ||||||
|         text: 'Cancel plan', |  | ||||||
|         src: subscription.cancelUrl, |  | ||||||
|       }, |  | ||||||
|     }, |  | ||||||
|     nextBillAmount: { |  | ||||||
|       title: subscription.nextBillAmount |  | ||||||
|         ? '€' + subscription.nextBillAmount |  | ||||||
|         : '---', |  | ||||||
|       action: { |  | ||||||
|         type: 'link', |  | ||||||
|         text: 'Update payment method', |  | ||||||
|         src: subscription.updateUrl, |  | ||||||
|       }, |  | ||||||
|     }, |  | ||||||
|     nextBillDate: { |  | ||||||
|       title: subscription.nextBillDate ? subscription.nextBillDate : '---', |  | ||||||
|       action: { |  | ||||||
|         type: 'text', |  | ||||||
|         text: '(monthly payment)', |  | ||||||
|       }, |  | ||||||
|     }, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const freeTrialSubscription = () => { |  | ||||||
|   return { |  | ||||||
|     status: null, |  | ||||||
|     monthlyQuota: { |  | ||||||
|       title: 'Free Trial', |  | ||||||
|       action: { |  | ||||||
|         type: 'link', |  | ||||||
|         text: 'Upgrade plan', |  | ||||||
|         src: '/settings/billing/upgrade', |  | ||||||
|       }, |  | ||||||
|     }, |  | ||||||
|     nextBillAmount: { |  | ||||||
|       title: '---', |  | ||||||
|       action: null, |  | ||||||
|     }, |  | ||||||
|     nextBillDate: { |  | ||||||
|       title: '---', |  | ||||||
|       action: null, |  | ||||||
|     }, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const executionIds = async (context) => { |  | ||||||
|   return ( |  | ||||||
|     await context.currentUser |  | ||||||
|       .$relatedQuery('executions') |  | ||||||
|       .select('executions.id') |  | ||||||
|   ).map((execution) => execution.id); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const executionStepCount = async (context) => { |  | ||||||
|   const executionStepCount = await ExecutionStep.query() |  | ||||||
|     .whereIn('execution_id', await executionIds(context)) |  | ||||||
|     .andWhere( |  | ||||||
|       'created_at', |  | ||||||
|       '>=', |  | ||||||
|       DateTime.now().minus({ days: 30 }).toISODate() |  | ||||||
|     ) |  | ||||||
|     .count() |  | ||||||
|     .first(); |  | ||||||
|  |  | ||||||
|   return executionStepCount.count; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getBillingAndUsage; |  | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| import appConfig from '../../config/app.js'; |  | ||||||
| import { hasValidLicense } from '../../helpers/license.ee.js'; |  | ||||||
| import Config from '../../models/config.js'; |  | ||||||
|  |  | ||||||
| const getConfig = async (_parent, params) => { |  | ||||||
|   if (!(await hasValidLicense())) return {}; |  | ||||||
|  |  | ||||||
|   const defaultConfig = { |  | ||||||
|     disableNotificationsPage: appConfig.disableNotificationsPage, |  | ||||||
|     disableFavicon: appConfig.disableFavicon, |  | ||||||
|     additionalDrawerLink: appConfig.additionalDrawerLink, |  | ||||||
|     additionalDrawerLinkText: appConfig.additionalDrawerLinkText, |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   const configQuery = Config.query(); |  | ||||||
|  |  | ||||||
|   if (Array.isArray(params.keys)) { |  | ||||||
|     configQuery.whereIn('key', params.keys); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const config = await configQuery.orderBy('key', 'asc'); |  | ||||||
|  |  | ||||||
|   return config.reduce((computedConfig, configEntry) => { |  | ||||||
|     const { key, value } = configEntry; |  | ||||||
|  |  | ||||||
|     computedConfig[key] = value?.data; |  | ||||||
|  |  | ||||||
|     return computedConfig; |  | ||||||
|   }, defaultConfig); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getConfig; |  | ||||||
| @@ -1,140 +0,0 @@ | |||||||
| import { vi, describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../app'; |  | ||||||
| import { createConfig } from '../../../test/factories/config'; |  | ||||||
| import appConfig from '../../config/app'; |  | ||||||
| import * as license from '../../helpers/license.ee'; |  | ||||||
|  |  | ||||||
| describe('graphQL getConfig query', () => { |  | ||||||
|   let configOne, configTwo, configThree, query; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     configOne = await createConfig({ key: 'configOne' }); |  | ||||||
|     configTwo = await createConfig({ key: 'configTwo' }); |  | ||||||
|     configThree = await createConfig({ key: 'configThree' }); |  | ||||||
|  |  | ||||||
|     query = ` |  | ||||||
|       query { |  | ||||||
|         getConfig |  | ||||||
|       } |  | ||||||
|     `; |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   describe('and without valid license', () => { |  | ||||||
|     beforeEach(async () => { |  | ||||||
|       vi.spyOn(license, 'hasValidLicense').mockResolvedValue(false); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and correct permissions', () => { |  | ||||||
|       it('should return empty config data', async () => { |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .send({ query }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { data: { getConfig: {} } }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   describe('and with valid license', () => { |  | ||||||
|     beforeEach(async () => { |  | ||||||
|       vi.spyOn(license, 'hasValidLicense').mockResolvedValue(true); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and without providing specific keys', () => { |  | ||||||
|       it('should return all config data', 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: false, |  | ||||||
|               disableFavicon: false, |  | ||||||
|               additionalDrawerLink: undefined, |  | ||||||
|               additionalDrawerLinkText: undefined, |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and with providing specific keys', () => { |  | ||||||
|       it('should return all config data', async () => { |  | ||||||
|         query = ` |  | ||||||
|           query { |  | ||||||
|             getConfig(keys: ["configOne", "configTwo"]) |  | ||||||
|           } |  | ||||||
|         `; |  | ||||||
|  |  | ||||||
|         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, |  | ||||||
|               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', |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,67 +0,0 @@ | |||||||
| import App from '../../models/app.js'; |  | ||||||
| import Flow from '../../models/flow.js'; |  | ||||||
| import Connection from '../../models/connection.js'; |  | ||||||
|  |  | ||||||
| const getConnectedApps = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('read', 'Connection'); |  | ||||||
|  |  | ||||||
|   const userConnections = context.currentUser.$relatedQuery('connections'); |  | ||||||
|   const allConnections = Connection.query(); |  | ||||||
|   const connectionBaseQuery = conditions.isCreator |  | ||||||
|     ? userConnections |  | ||||||
|     : allConnections; |  | ||||||
|  |  | ||||||
|   const userFlows = context.currentUser.$relatedQuery('flows'); |  | ||||||
|   const allFlows = Flow.query(); |  | ||||||
|   const flowBaseQuery = conditions.isCreator ? userFlows : allFlows; |  | ||||||
|  |  | ||||||
|   let apps = await App.findAll(params.name); |  | ||||||
|  |  | ||||||
|   const connections = await connectionBaseQuery |  | ||||||
|     .clone() |  | ||||||
|     .select('connections.key') |  | ||||||
|     .where({ draft: false }) |  | ||||||
|     .count('connections.id as count') |  | ||||||
|     .groupBy('connections.key'); |  | ||||||
|  |  | ||||||
|   const flows = await flowBaseQuery |  | ||||||
|     .clone() |  | ||||||
|     .withGraphJoined('steps') |  | ||||||
|     .orderBy('created_at', 'desc'); |  | ||||||
|  |  | ||||||
|   const duplicatedUsedApps = flows |  | ||||||
|     .map((flow) => flow.steps.map((step) => step.appKey)) |  | ||||||
|     .flat() |  | ||||||
|     .filter(Boolean); |  | ||||||
|  |  | ||||||
|   const connectionKeys = connections.map((connection) => connection.key); |  | ||||||
|   const usedApps = [...new Set([...duplicatedUsedApps, ...connectionKeys])]; |  | ||||||
|  |  | ||||||
|   apps = apps |  | ||||||
|     .filter((app) => { |  | ||||||
|       return usedApps.includes(app.key); |  | ||||||
|     }) |  | ||||||
|     .map((app) => { |  | ||||||
|       const connection = connections.find( |  | ||||||
|         (connection) => connection.key === app.key |  | ||||||
|       ); |  | ||||||
|  |  | ||||||
|       app.connectionCount = connection?.count || 0; |  | ||||||
|       app.flowCount = 0; |  | ||||||
|  |  | ||||||
|       flows.forEach((flow) => { |  | ||||||
|         const usedFlow = flow.steps.find((step) => step.appKey === app.key); |  | ||||||
|  |  | ||||||
|         if (usedFlow) { |  | ||||||
|           app.flowCount += 1; |  | ||||||
|         } |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       return app; |  | ||||||
|     }) |  | ||||||
|     .sort((appA, appB) => appA.name.localeCompare(appB.name)); |  | ||||||
|  |  | ||||||
|   return apps; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getConnectedApps; |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| const getCurrentUser = async (_parent, _params, context) => { |  | ||||||
|   return context.currentUser; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getCurrentUser; |  | ||||||
| @@ -1,79 +0,0 @@ | |||||||
| import { 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'; |  | ||||||
|  |  | ||||||
| describe('graphQL getCurrentUser query', () => { |  | ||||||
|   let role, currentUser, token, requestObject; |  | ||||||
|  |  | ||||||
|   beforeEach(async () => { |  | ||||||
|     role = await createRole({ |  | ||||||
|       key: 'sample', |  | ||||||
|       name: 'sample', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     currentUser = await createUser({ |  | ||||||
|       roleId: role.id, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|     requestObject = request(app).post('/graphql').set('Authorization', token); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should return user data', async () => { |  | ||||||
|     const query = ` |  | ||||||
|       query { |  | ||||||
|         getCurrentUser { |  | ||||||
|           id |  | ||||||
|           email |  | ||||||
|           fullName |  | ||||||
|           email |  | ||||||
|           createdAt |  | ||||||
|           updatedAt |  | ||||||
|           role { |  | ||||||
|             id |  | ||||||
|             name |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     `; |  | ||||||
|  |  | ||||||
|     const response = await requestObject.send({ query }).expect(200); |  | ||||||
|  |  | ||||||
|     const expectedResponsePayload = { |  | ||||||
|       data: { |  | ||||||
|         getCurrentUser: { |  | ||||||
|           createdAt: currentUser.createdAt.getTime().toString(), |  | ||||||
|           email: currentUser.email, |  | ||||||
|           fullName: currentUser.fullName, |  | ||||||
|           id: currentUser.id, |  | ||||||
|           role: { id: role.id, name: role.name }, |  | ||||||
|           updatedAt: currentUser.updatedAt.getTime().toString(), |  | ||||||
|         }, |  | ||||||
|       }, |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('should not return user password', async () => { |  | ||||||
|     const query = ` |  | ||||||
|       query { |  | ||||||
|         getCurrentUser { |  | ||||||
|           id |  | ||||||
|           email |  | ||||||
|           password |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     `; |  | ||||||
|  |  | ||||||
|     const response = await requestObject.send({ query }).expect(400); |  | ||||||
|  |  | ||||||
|     expect(response.body.errors).toBeDefined(); |  | ||||||
|     expect(response.body.errors[0].message).toEqual( |  | ||||||
|       'Cannot query field "password" on type "User".' |  | ||||||
|     ); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,65 +0,0 @@ | |||||||
| import App from '../../models/app.js'; |  | ||||||
| import Step from '../../models/step.js'; |  | ||||||
| import ExecutionStep from '../../models/execution-step.js'; |  | ||||||
| import globalVariable from '../../helpers/global-variable.js'; |  | ||||||
| import computeParameters from '../../helpers/compute-parameters.js'; |  | ||||||
|  |  | ||||||
| const getDynamicData = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('update', 'Flow'); |  | ||||||
|   const userSteps = context.currentUser.$relatedQuery('steps'); |  | ||||||
|   const allSteps = Step.query(); |  | ||||||
|   const stepBaseQuery = conditions.isCreator ? userSteps : allSteps; |  | ||||||
|  |  | ||||||
|   const step = await stepBaseQuery |  | ||||||
|     .clone() |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       connection: true, |  | ||||||
|       flow: true, |  | ||||||
|     }) |  | ||||||
|     .findById(params.stepId); |  | ||||||
|  |  | ||||||
|   if (!step) return null; |  | ||||||
|  |  | ||||||
|   const connection = step.connection; |  | ||||||
|  |  | ||||||
|   if (!connection || !step.appKey) return null; |  | ||||||
|  |  | ||||||
|   const flow = step.flow; |  | ||||||
|   const app = await App.findOneByKey(step.appKey); |  | ||||||
|   const $ = await globalVariable({ connection, app, flow, step }); |  | ||||||
|  |  | ||||||
|   const command = app.dynamicData.find((data) => data.key === params.key); |  | ||||||
|  |  | ||||||
|   // apply run-time parameters that're not persisted yet |  | ||||||
|   for (const parameterKey in params.parameters) { |  | ||||||
|     const parameterValue = params.parameters[parameterKey]; |  | ||||||
|     $.step.parameters[parameterKey] = parameterValue; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const lastExecution = await flow.$relatedQuery('lastExecution'); |  | ||||||
|   const lastExecutionId = lastExecution?.id; |  | ||||||
|  |  | ||||||
|   const priorExecutionSteps = lastExecutionId |  | ||||||
|     ? await ExecutionStep.query().where({ |  | ||||||
|         execution_id: lastExecutionId, |  | ||||||
|       }) |  | ||||||
|     : []; |  | ||||||
|  |  | ||||||
|   // compute variables in parameters |  | ||||||
|   const computedParameters = computeParameters( |  | ||||||
|     $.step.parameters, |  | ||||||
|     priorExecutionSteps |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   $.step.parameters = computedParameters; |  | ||||||
|  |  | ||||||
|   const fetchedData = await command.run($); |  | ||||||
|  |  | ||||||
|   if (fetchedData.error) { |  | ||||||
|     throw new Error(JSON.stringify(fetchedData.error)); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return fetchedData.data; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getDynamicData; |  | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| import App from '../../models/app.js'; |  | ||||||
| import Step from '../../models/step.js'; |  | ||||||
| import globalVariable from '../../helpers/global-variable.js'; |  | ||||||
|  |  | ||||||
| const getDynamicFields = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('update', 'Flow'); |  | ||||||
|   const userSteps = context.currentUser.$relatedQuery('steps'); |  | ||||||
|   const allSteps = Step.query(); |  | ||||||
|   const stepBaseQuery = conditions.isCreator ? userSteps : allSteps; |  | ||||||
|  |  | ||||||
|   const step = await stepBaseQuery |  | ||||||
|     .clone() |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       connection: true, |  | ||||||
|       flow: true, |  | ||||||
|     }) |  | ||||||
|     .findById(params.stepId); |  | ||||||
|  |  | ||||||
|   if (!step) return null; |  | ||||||
|  |  | ||||||
|   const connection = step.connection; |  | ||||||
|  |  | ||||||
|   if (!step.appKey) return null; |  | ||||||
|  |  | ||||||
|   const app = await App.findOneByKey(step.appKey); |  | ||||||
|   const $ = await globalVariable({ connection, app, flow: step.flow, step }); |  | ||||||
|  |  | ||||||
|   const command = app.dynamicFields.find((data) => data.key === params.key); |  | ||||||
|  |  | ||||||
|   for (const parameterKey in params.parameters) { |  | ||||||
|     const parameterValue = params.parameters[parameterKey]; |  | ||||||
|     $.step.parameters[parameterKey] = parameterValue; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const additionalFields = (await command.run($)) || []; |  | ||||||
|  |  | ||||||
|   return additionalFields; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getDynamicFields; |  | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| import paginate from '../../helpers/pagination.js'; |  | ||||||
| import Execution from '../../models/execution.js'; |  | ||||||
|  |  | ||||||
| const getExecutionSteps = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('read', 'Execution'); |  | ||||||
|   const userExecutions = context.currentUser.$relatedQuery('executions'); |  | ||||||
|   const allExecutions = Execution.query(); |  | ||||||
|   const executionBaseQuery = conditions.isCreator |  | ||||||
|     ? userExecutions |  | ||||||
|     : allExecutions; |  | ||||||
|  |  | ||||||
|   const execution = await executionBaseQuery |  | ||||||
|     .clone() |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .findById(params.executionId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   const executionSteps = execution |  | ||||||
|     .$relatedQuery('executionSteps') |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .withGraphFetched('step') |  | ||||||
|     .orderBy('created_at', 'asc'); |  | ||||||
|  |  | ||||||
|   return paginate(executionSteps, params.limit, params.offset); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getExecutionSteps; |  | ||||||
| @@ -1,25 +0,0 @@ | |||||||
| import Execution from '../../models/execution.js'; |  | ||||||
|  |  | ||||||
| const getExecution = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('read', 'Execution'); |  | ||||||
|   const userExecutions = context.currentUser.$relatedQuery('executions'); |  | ||||||
|   const allExecutions = Execution.query(); |  | ||||||
|   const executionBaseQuery = conditions.isCreator |  | ||||||
|     ? userExecutions |  | ||||||
|     : allExecutions; |  | ||||||
|  |  | ||||||
|   const execution = await executionBaseQuery |  | ||||||
|     .clone() |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       flow: { |  | ||||||
|         steps: true, |  | ||||||
|       }, |  | ||||||
|     }) |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .findById(params.executionId) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   return execution; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getExecution; |  | ||||||
| @@ -1,70 +0,0 @@ | |||||||
| import { raw } from 'objection'; |  | ||||||
| import { DateTime } from 'luxon'; |  | ||||||
| import Execution from '../../models/execution.js'; |  | ||||||
| import paginate from '../../helpers/pagination.js'; |  | ||||||
|  |  | ||||||
| const getExecutions = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('read', 'Execution'); |  | ||||||
|  |  | ||||||
|   const filters = params.filters; |  | ||||||
|  |  | ||||||
|   const userExecutions = context.currentUser.$relatedQuery('executions'); |  | ||||||
|   const allExecutions = Execution.query(); |  | ||||||
|   const executionBaseQuery = conditions.isCreator |  | ||||||
|     ? userExecutions |  | ||||||
|     : allExecutions; |  | ||||||
|  |  | ||||||
|   const selectStatusStatement = ` |  | ||||||
|     case |  | ||||||
|       when count(*) filter (where execution_steps.status = 'failure') > 0 |  | ||||||
|         then 'failure' |  | ||||||
|       else 'success' |  | ||||||
|     end |  | ||||||
|     as status |  | ||||||
|   `; |  | ||||||
|  |  | ||||||
|   const executions = executionBaseQuery |  | ||||||
|     .clone() |  | ||||||
|     .joinRelated('executionSteps as execution_steps') |  | ||||||
|     .select('executions.*', raw(selectStatusStatement)) |  | ||||||
|     .groupBy('executions.id') |  | ||||||
|     .orderBy('created_at', 'desc'); |  | ||||||
|  |  | ||||||
|   const computedExecutions = Execution.query() |  | ||||||
|     .with('executions', executions) |  | ||||||
|     .withSoftDeleted() |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       flow: { |  | ||||||
|         steps: true, |  | ||||||
|       }, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|   if (filters?.flowId) { |  | ||||||
|     computedExecutions.where('executions.flow_id', filters.flowId); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (filters?.status) { |  | ||||||
|     computedExecutions.where('executions.status', filters.status); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (filters?.createdAt) { |  | ||||||
|     const createdAtFilter = filters.createdAt; |  | ||||||
|     if (createdAtFilter.from) { |  | ||||||
|       const isoFromDateTime = DateTime.fromMillis( |  | ||||||
|         parseInt(createdAtFilter.from, 10) |  | ||||||
|       ).toISO(); |  | ||||||
|       computedExecutions.where('executions.created_at', '>=', isoFromDateTime); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (createdAtFilter.to) { |  | ||||||
|       const isoToDateTime = DateTime.fromMillis( |  | ||||||
|         parseInt(createdAtFilter.to, 10) |  | ||||||
|       ).toISO(); |  | ||||||
|       computedExecutions.where('executions.created_at', '<=', isoToDateTime); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return paginate(computedExecutions, params.limit, params.offset); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getExecutions; |  | ||||||
| @@ -1,472 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../app'; |  | ||||||
| import appConfig from '../../config/app'; |  | ||||||
| import createAuthTokenByUserId from '../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createRole } from '../../../test/factories/role'; |  | ||||||
| import { createPermission } from '../../../test/factories/permission'; |  | ||||||
| import { createUser } from '../../../test/factories/user'; |  | ||||||
| import { createFlow } from '../../../test/factories/flow'; |  | ||||||
| import { createStep } from '../../../test/factories/step'; |  | ||||||
| import { createExecution } from '../../../test/factories/execution'; |  | ||||||
| import { createExecutionStep } from '../../../test/factories/execution-step'; |  | ||||||
|  |  | ||||||
| describe('graphQL getExecutions query', () => { |  | ||||||
|   const query = ` |  | ||||||
|     query { |  | ||||||
|       getExecutions(limit: 10, offset: 0) { |  | ||||||
|         pageInfo { |  | ||||||
|           currentPage |  | ||||||
|           totalPages |  | ||||||
|         } |  | ||||||
|         edges { |  | ||||||
|           node { |  | ||||||
|             id |  | ||||||
|             testRun |  | ||||||
|             createdAt |  | ||||||
|             updatedAt |  | ||||||
|             status |  | ||||||
|             flow { |  | ||||||
|               id |  | ||||||
|               name |  | ||||||
|               active |  | ||||||
|               steps { |  | ||||||
|                 iconUrl |  | ||||||
|               } |  | ||||||
|             } |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   `; |  | ||||||
|  |  | ||||||
|   describe('and without correct permissions', () => { |  | ||||||
|     it('should throw not authorized error', async () => { |  | ||||||
|       const userWithoutPermissions = await createUser(); |  | ||||||
|       const token = createAuthTokenByUserId(userWithoutPermissions.id); |  | ||||||
|  |  | ||||||
|       const response = await request(app) |  | ||||||
|         .post('/graphql') |  | ||||||
|         .set('Authorization', token) |  | ||||||
|         .send({ query }) |  | ||||||
|         .expect(200); |  | ||||||
|  |  | ||||||
|       expect(response.body.errors).toBeDefined(); |  | ||||||
|       expect(response.body.errors[0].message).toEqual('Not authorized!'); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   describe('and with correct permission', () => { |  | ||||||
|     let role, |  | ||||||
|       currentUser, |  | ||||||
|       anotherUser, |  | ||||||
|       token, |  | ||||||
|       flowOne, |  | ||||||
|       stepOneForFlowOne, |  | ||||||
|       stepTwoForFlowOne, |  | ||||||
|       executionOne, |  | ||||||
|       flowTwo, |  | ||||||
|       stepOneForFlowTwo, |  | ||||||
|       stepTwoForFlowTwo, |  | ||||||
|       executionTwo, |  | ||||||
|       flowThree, |  | ||||||
|       stepOneForFlowThree, |  | ||||||
|       stepTwoForFlowThree, |  | ||||||
|       executionThree, |  | ||||||
|       expectedResponseForExecutionOne, |  | ||||||
|       expectedResponseForExecutionTwo, |  | ||||||
|       expectedResponseForExecutionThree; |  | ||||||
|  |  | ||||||
|     beforeEach(async () => { |  | ||||||
|       role = await createRole({ |  | ||||||
|         key: 'sample', |  | ||||||
|         name: 'sample', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       currentUser = await createUser({ |  | ||||||
|         roleId: role.id, |  | ||||||
|         fullName: 'Current User', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       anotherUser = await createUser(); |  | ||||||
|  |  | ||||||
|       token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|  |  | ||||||
|       flowOne = await createFlow({ |  | ||||||
|         userId: currentUser.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       stepOneForFlowOne = await createStep({ |  | ||||||
|         flowId: flowOne.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       stepTwoForFlowOne = await createStep({ |  | ||||||
|         flowId: flowOne.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       executionOne = await createExecution({ |  | ||||||
|         flowId: flowOne.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       await createExecutionStep({ |  | ||||||
|         executionId: executionOne.id, |  | ||||||
|         stepId: stepOneForFlowOne.id, |  | ||||||
|         status: 'success', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       await createExecutionStep({ |  | ||||||
|         executionId: executionOne.id, |  | ||||||
|         stepId: stepTwoForFlowOne.id, |  | ||||||
|         status: 'success', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       flowTwo = await createFlow({ |  | ||||||
|         userId: currentUser.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       stepOneForFlowTwo = await createStep({ |  | ||||||
|         flowId: flowTwo.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       stepTwoForFlowTwo = await createStep({ |  | ||||||
|         flowId: flowTwo.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       executionTwo = await createExecution({ |  | ||||||
|         flowId: flowTwo.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       await createExecutionStep({ |  | ||||||
|         executionId: executionTwo.id, |  | ||||||
|         stepId: stepOneForFlowTwo.id, |  | ||||||
|         status: 'success', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       await createExecutionStep({ |  | ||||||
|         executionId: executionTwo.id, |  | ||||||
|         stepId: stepTwoForFlowTwo.id, |  | ||||||
|         status: 'failure', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       flowThree = await createFlow({ |  | ||||||
|         userId: anotherUser.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       stepOneForFlowThree = await createStep({ |  | ||||||
|         flowId: flowThree.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       stepTwoForFlowThree = await createStep({ |  | ||||||
|         flowId: flowThree.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       executionThree = await createExecution({ |  | ||||||
|         flowId: flowThree.id, |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       await createExecutionStep({ |  | ||||||
|         executionId: executionThree.id, |  | ||||||
|         stepId: stepOneForFlowThree.id, |  | ||||||
|         status: 'success', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       await createExecutionStep({ |  | ||||||
|         executionId: executionThree.id, |  | ||||||
|         stepId: stepTwoForFlowThree.id, |  | ||||||
|         status: 'failure', |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       expectedResponseForExecutionOne = { |  | ||||||
|         node: { |  | ||||||
|           createdAt: executionOne.createdAt.getTime().toString(), |  | ||||||
|           flow: { |  | ||||||
|             active: flowOne.active, |  | ||||||
|             id: flowOne.id, |  | ||||||
|             name: flowOne.name, |  | ||||||
|             steps: [ |  | ||||||
|               { |  | ||||||
|                 iconUrl: `${appConfig.baseUrl}/apps/${stepOneForFlowOne.appKey}/assets/favicon.svg`, |  | ||||||
|               }, |  | ||||||
|               { |  | ||||||
|                 iconUrl: `${appConfig.baseUrl}/apps/${stepTwoForFlowOne.appKey}/assets/favicon.svg`, |  | ||||||
|               }, |  | ||||||
|             ], |  | ||||||
|           }, |  | ||||||
|           id: executionOne.id, |  | ||||||
|           status: 'success', |  | ||||||
|           testRun: executionOne.testRun, |  | ||||||
|           updatedAt: executionOne.updatedAt.getTime().toString(), |  | ||||||
|         }, |  | ||||||
|       }; |  | ||||||
|  |  | ||||||
|       expectedResponseForExecutionTwo = { |  | ||||||
|         node: { |  | ||||||
|           createdAt: executionTwo.createdAt.getTime().toString(), |  | ||||||
|           flow: { |  | ||||||
|             active: flowTwo.active, |  | ||||||
|             id: flowTwo.id, |  | ||||||
|             name: flowTwo.name, |  | ||||||
|             steps: [ |  | ||||||
|               { |  | ||||||
|                 iconUrl: `${appConfig.baseUrl}/apps/${stepTwoForFlowTwo.appKey}/assets/favicon.svg`, |  | ||||||
|               }, |  | ||||||
|               { |  | ||||||
|                 iconUrl: `${appConfig.baseUrl}/apps/${stepTwoForFlowTwo.appKey}/assets/favicon.svg`, |  | ||||||
|               }, |  | ||||||
|             ], |  | ||||||
|           }, |  | ||||||
|           id: executionTwo.id, |  | ||||||
|           status: 'failure', |  | ||||||
|           testRun: executionTwo.testRun, |  | ||||||
|           updatedAt: executionTwo.updatedAt.getTime().toString(), |  | ||||||
|         }, |  | ||||||
|       }; |  | ||||||
|  |  | ||||||
|       expectedResponseForExecutionThree = { |  | ||||||
|         node: { |  | ||||||
|           createdAt: executionThree.createdAt.getTime().toString(), |  | ||||||
|           flow: { |  | ||||||
|             active: flowThree.active, |  | ||||||
|             id: flowThree.id, |  | ||||||
|             name: flowThree.name, |  | ||||||
|             steps: [ |  | ||||||
|               { |  | ||||||
|                 iconUrl: `${appConfig.baseUrl}/apps/${stepOneForFlowThree.appKey}/assets/favicon.svg`, |  | ||||||
|               }, |  | ||||||
|               { |  | ||||||
|                 iconUrl: `${appConfig.baseUrl}/apps/${stepTwoForFlowThree.appKey}/assets/favicon.svg`, |  | ||||||
|               }, |  | ||||||
|             ], |  | ||||||
|           }, |  | ||||||
|           id: executionThree.id, |  | ||||||
|           status: 'failure', |  | ||||||
|           testRun: executionThree.testRun, |  | ||||||
|           updatedAt: executionThree.updatedAt.getTime().toString(), |  | ||||||
|         }, |  | ||||||
|       }; |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and with isCreator condition', () => { |  | ||||||
|       beforeEach(async () => { |  | ||||||
|         await createPermission({ |  | ||||||
|           action: 'read', |  | ||||||
|           subject: 'Execution', |  | ||||||
|           roleId: role.id, |  | ||||||
|           conditions: ['isCreator'], |  | ||||||
|         }); |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       it('should return executions data of the current user', async () => { |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .set('Authorization', token) |  | ||||||
|           .send({ query }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getExecutions: { |  | ||||||
|               edges: [ |  | ||||||
|                 expectedResponseForExecutionTwo, |  | ||||||
|                 expectedResponseForExecutionOne, |  | ||||||
|               ], |  | ||||||
|               pageInfo: { currentPage: 1, totalPages: 1 }, |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and without isCreator condition', () => { |  | ||||||
|       beforeEach(async () => { |  | ||||||
|         await createPermission({ |  | ||||||
|           action: 'read', |  | ||||||
|           subject: 'Execution', |  | ||||||
|           roleId: role.id, |  | ||||||
|           conditions: [], |  | ||||||
|         }); |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       it('should return executions data of all users', async () => { |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .set('Authorization', token) |  | ||||||
|           .send({ query }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getExecutions: { |  | ||||||
|               edges: [ |  | ||||||
|                 expectedResponseForExecutionThree, |  | ||||||
|                 expectedResponseForExecutionTwo, |  | ||||||
|                 expectedResponseForExecutionOne, |  | ||||||
|               ], |  | ||||||
|               pageInfo: { currentPage: 1, totalPages: 1 }, |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and with filters', () => { |  | ||||||
|       beforeEach(async () => { |  | ||||||
|         await createPermission({ |  | ||||||
|           action: 'read', |  | ||||||
|           subject: 'Execution', |  | ||||||
|           roleId: role.id, |  | ||||||
|           conditions: [], |  | ||||||
|         }); |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       it('should return executions data for the specified flow', async () => { |  | ||||||
|         const query = ` |  | ||||||
|             query { |  | ||||||
|               getExecutions(limit: 10, offset: 0, filters: { flowId: "${flowOne.id}" }) { |  | ||||||
|                 pageInfo { |  | ||||||
|                   currentPage |  | ||||||
|                   totalPages |  | ||||||
|                 } |  | ||||||
|                 edges { |  | ||||||
|                   node { |  | ||||||
|                     id |  | ||||||
|                     testRun |  | ||||||
|                     createdAt |  | ||||||
|                     updatedAt |  | ||||||
|                     status |  | ||||||
|                     flow { |  | ||||||
|                       id |  | ||||||
|                       name |  | ||||||
|                       active |  | ||||||
|                       steps { |  | ||||||
|                         iconUrl |  | ||||||
|                       } |  | ||||||
|                     } |  | ||||||
|                   } |  | ||||||
|                 } |  | ||||||
|               } |  | ||||||
|             } |  | ||||||
|           `; |  | ||||||
|  |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .set('Authorization', token) |  | ||||||
|           .send({ query }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getExecutions: { |  | ||||||
|               edges: [expectedResponseForExecutionOne], |  | ||||||
|               pageInfo: { currentPage: 1, totalPages: 1 }, |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       it('should return only executions data with success status', async () => { |  | ||||||
|         const query = ` |  | ||||||
|             query { |  | ||||||
|               getExecutions(limit: 10, offset: 0, filters: { status: "success" }) { |  | ||||||
|                 pageInfo { |  | ||||||
|                   currentPage |  | ||||||
|                   totalPages |  | ||||||
|                 } |  | ||||||
|                 edges { |  | ||||||
|                   node { |  | ||||||
|                     id |  | ||||||
|                     testRun |  | ||||||
|                     createdAt |  | ||||||
|                     updatedAt |  | ||||||
|                     status |  | ||||||
|                     flow { |  | ||||||
|                       id |  | ||||||
|                       name |  | ||||||
|                       active |  | ||||||
|                       steps { |  | ||||||
|                         iconUrl |  | ||||||
|                       } |  | ||||||
|                     } |  | ||||||
|                   } |  | ||||||
|                 } |  | ||||||
|               } |  | ||||||
|             } |  | ||||||
|           `; |  | ||||||
|  |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .set('Authorization', token) |  | ||||||
|           .send({ query }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getExecutions: { |  | ||||||
|               edges: [expectedResponseForExecutionOne], |  | ||||||
|               pageInfo: { currentPage: 1, totalPages: 1 }, |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       it('should return only executions data within date range', async () => { |  | ||||||
|         const createdAtFrom = executionOne.createdAt.getTime().toString(); |  | ||||||
|  |  | ||||||
|         const createdAtTo = executionOne.createdAt.getTime().toString(); |  | ||||||
|  |  | ||||||
|         const query = ` |  | ||||||
|             query { |  | ||||||
|               getExecutions(limit: 10, offset: 0, filters: { createdAt: { from: "${createdAtFrom}", to: "${createdAtTo}" }}) { |  | ||||||
|                 pageInfo { |  | ||||||
|                   currentPage |  | ||||||
|                   totalPages |  | ||||||
|                 } |  | ||||||
|                 edges { |  | ||||||
|                   node { |  | ||||||
|                     id |  | ||||||
|                     testRun |  | ||||||
|                     createdAt |  | ||||||
|                     updatedAt |  | ||||||
|                     status |  | ||||||
|                     flow { |  | ||||||
|                       id |  | ||||||
|                       name |  | ||||||
|                       active |  | ||||||
|                       steps { |  | ||||||
|                         iconUrl |  | ||||||
|                       } |  | ||||||
|                     } |  | ||||||
|                   } |  | ||||||
|                 } |  | ||||||
|               } |  | ||||||
|             } |  | ||||||
|           `; |  | ||||||
|  |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .set('Authorization', token) |  | ||||||
|           .send({ query }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getExecutions: { |  | ||||||
|               edges: [expectedResponseForExecutionOne], |  | ||||||
|               pageInfo: { currentPage: 1, totalPages: 1 }, |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| import Flow from '../../models/flow.js'; |  | ||||||
|  |  | ||||||
| const getFlow = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('read', 'Flow'); |  | ||||||
|   const userFlows = context.currentUser.$relatedQuery('flows'); |  | ||||||
|   const allFlows = Flow.query(); |  | ||||||
|   const baseQuery = conditions.isCreator ? userFlows : allFlows; |  | ||||||
|  |  | ||||||
|   const flow = await baseQuery |  | ||||||
|     .clone() |  | ||||||
|     .withGraphJoined('[steps.[connection]]') |  | ||||||
|     .orderBy('steps.position', 'asc') |  | ||||||
|     .findOne({ 'flows.id': params.id }) |  | ||||||
|     .throwIfNotFound(); |  | ||||||
|  |  | ||||||
|   return flow; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getFlow; |  | ||||||
| @@ -1,240 +0,0 @@ | |||||||
| import { describe, it, expect, beforeEach } from 'vitest'; |  | ||||||
| import request from 'supertest'; |  | ||||||
| import app from '../../app'; |  | ||||||
| import appConfig from '../../config/app'; |  | ||||||
| import createAuthTokenByUserId from '../../helpers/create-auth-token-by-user-id'; |  | ||||||
| import { createRole } from '../../../test/factories/role'; |  | ||||||
| import { createPermission } from '../../../test/factories/permission'; |  | ||||||
| import { createUser } from '../../../test/factories/user'; |  | ||||||
| import { createFlow } from '../../../test/factories/flow'; |  | ||||||
| import { createStep } from '../../../test/factories/step'; |  | ||||||
| import { createConnection } from '../../../test/factories/connection'; |  | ||||||
|  |  | ||||||
| describe('graphQL getFlow query', () => { |  | ||||||
|   const query = (flowId) => { |  | ||||||
|     return ` |  | ||||||
|       query { |  | ||||||
|         getFlow(id: "${flowId}") { |  | ||||||
|           id |  | ||||||
|           name |  | ||||||
|           active |  | ||||||
|           status |  | ||||||
|           steps { |  | ||||||
|             id |  | ||||||
|             type |  | ||||||
|             key |  | ||||||
|             appKey |  | ||||||
|             iconUrl |  | ||||||
|             webhookUrl |  | ||||||
|             status |  | ||||||
|             position |  | ||||||
|             connection { |  | ||||||
|               id |  | ||||||
|               verified |  | ||||||
|               createdAt |  | ||||||
|             } |  | ||||||
|             parameters |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     `; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   describe('and without permissions', () => { |  | ||||||
|     it('should throw not authorized error', async () => { |  | ||||||
|       const userWithoutPermissions = await createUser(); |  | ||||||
|       const token = createAuthTokenByUserId(userWithoutPermissions.id); |  | ||||||
|       const flow = await createFlow(); |  | ||||||
|  |  | ||||||
|       const response = await request(app) |  | ||||||
|         .post('/graphql') |  | ||||||
|         .set('Authorization', token) |  | ||||||
|         .send({ query: query(flow.id) }) |  | ||||||
|         .expect(200); |  | ||||||
|  |  | ||||||
|       expect(response.body.errors).toBeDefined(); |  | ||||||
|       expect(response.body.errors[0].message).toEqual('Not authorized!'); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   describe('and with correct permission', () => { |  | ||||||
|     let currentUser, currentUserRole, currentUserFlow; |  | ||||||
|  |  | ||||||
|     beforeEach(async () => { |  | ||||||
|       currentUserRole = await createRole(); |  | ||||||
|       currentUser = await createUser({ roleId: currentUserRole.id }); |  | ||||||
|       currentUserFlow = await createFlow({ userId: currentUser.id }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and with isCreator condition', () => { |  | ||||||
|       it('should return executions data of the current user', async () => { |  | ||||||
|         await createPermission({ |  | ||||||
|           action: 'read', |  | ||||||
|           subject: 'Flow', |  | ||||||
|           roleId: currentUserRole.id, |  | ||||||
|           conditions: ['isCreator'], |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const triggerStep = await createStep({ |  | ||||||
|           flowId: currentUserFlow.id, |  | ||||||
|           type: 'trigger', |  | ||||||
|           key: 'catchRawWebhook', |  | ||||||
|           webhookPath: `/webhooks/flows/${currentUserFlow.id}`, |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const actionConnection = await createConnection({ |  | ||||||
|           userId: currentUser.id, |  | ||||||
|           formattedData: { |  | ||||||
|             screenName: 'Test', |  | ||||||
|             authenticationKey: 'test key', |  | ||||||
|           }, |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const actionStep = await createStep({ |  | ||||||
|           flowId: currentUserFlow.id, |  | ||||||
|           type: 'action', |  | ||||||
|           connectionId: actionConnection.id, |  | ||||||
|           key: 'translateText', |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|  |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .set('Authorization', token) |  | ||||||
|           .send({ query: query(currentUserFlow.id) }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getFlow: { |  | ||||||
|               active: currentUserFlow.active, |  | ||||||
|               id: currentUserFlow.id, |  | ||||||
|               name: currentUserFlow.name, |  | ||||||
|               status: 'draft', |  | ||||||
|               steps: [ |  | ||||||
|                 { |  | ||||||
|                   appKey: triggerStep.appKey, |  | ||||||
|                   connection: null, |  | ||||||
|                   iconUrl: `${appConfig.baseUrl}/apps/${triggerStep.appKey}/assets/favicon.svg`, |  | ||||||
|                   id: triggerStep.id, |  | ||||||
|                   key: 'catchRawWebhook', |  | ||||||
|                   parameters: {}, |  | ||||||
|                   position: 1, |  | ||||||
|                   status: triggerStep.status, |  | ||||||
|                   type: 'trigger', |  | ||||||
|                   webhookUrl: `${appConfig.baseUrl}/webhooks/flows/${currentUserFlow.id}`, |  | ||||||
|                 }, |  | ||||||
|                 { |  | ||||||
|                   appKey: actionStep.appKey, |  | ||||||
|                   connection: { |  | ||||||
|                     createdAt: actionConnection.createdAt.getTime().toString(), |  | ||||||
|                     id: actionConnection.id, |  | ||||||
|                     verified: actionConnection.verified, |  | ||||||
|                   }, |  | ||||||
|                   iconUrl: `${appConfig.baseUrl}/apps/${actionStep.appKey}/assets/favicon.svg`, |  | ||||||
|                   id: actionStep.id, |  | ||||||
|                   key: 'translateText', |  | ||||||
|                   parameters: {}, |  | ||||||
|                   position: 1, |  | ||||||
|                   status: actionStep.status, |  | ||||||
|                   type: 'action', |  | ||||||
|                   webhookUrl: 'http://localhost:3000/null', |  | ||||||
|                 }, |  | ||||||
|               ], |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     describe('and without isCreator condition', () => { |  | ||||||
|       it('should return executions data of all users', async () => { |  | ||||||
|         await createPermission({ |  | ||||||
|           action: 'read', |  | ||||||
|           subject: 'Flow', |  | ||||||
|           roleId: currentUserRole.id, |  | ||||||
|           conditions: [], |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const anotherUser = await createUser(); |  | ||||||
|         const anotherUserFlow = await createFlow({ userId: anotherUser.id }); |  | ||||||
|  |  | ||||||
|         const triggerStep = await createStep({ |  | ||||||
|           flowId: anotherUserFlow.id, |  | ||||||
|           type: 'trigger', |  | ||||||
|           key: 'catchRawWebhook', |  | ||||||
|           webhookPath: `/webhooks/flows/${anotherUserFlow.id}`, |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const actionConnection = await createConnection({ |  | ||||||
|           userId: anotherUser.id, |  | ||||||
|           formattedData: { |  | ||||||
|             screenName: 'Test', |  | ||||||
|             authenticationKey: 'test key', |  | ||||||
|           }, |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const actionStep = await createStep({ |  | ||||||
|           flowId: anotherUserFlow.id, |  | ||||||
|           type: 'action', |  | ||||||
|           connectionId: actionConnection.id, |  | ||||||
|           key: 'translateText', |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         const token = createAuthTokenByUserId(currentUser.id); |  | ||||||
|  |  | ||||||
|         const response = await request(app) |  | ||||||
|           .post('/graphql') |  | ||||||
|           .set('Authorization', token) |  | ||||||
|           .send({ query: query(anotherUserFlow.id) }) |  | ||||||
|           .expect(200); |  | ||||||
|  |  | ||||||
|         const expectedResponsePayload = { |  | ||||||
|           data: { |  | ||||||
|             getFlow: { |  | ||||||
|               active: anotherUserFlow.active, |  | ||||||
|               id: anotherUserFlow.id, |  | ||||||
|               name: anotherUserFlow.name, |  | ||||||
|               status: 'draft', |  | ||||||
|               steps: [ |  | ||||||
|                 { |  | ||||||
|                   appKey: triggerStep.appKey, |  | ||||||
|                   connection: null, |  | ||||||
|                   iconUrl: `${appConfig.baseUrl}/apps/${triggerStep.appKey}/assets/favicon.svg`, |  | ||||||
|                   id: triggerStep.id, |  | ||||||
|                   key: 'catchRawWebhook', |  | ||||||
|                   parameters: {}, |  | ||||||
|                   position: 1, |  | ||||||
|                   status: triggerStep.status, |  | ||||||
|                   type: 'trigger', |  | ||||||
|                   webhookUrl: `${appConfig.baseUrl}/webhooks/flows/${anotherUserFlow.id}`, |  | ||||||
|                 }, |  | ||||||
|                 { |  | ||||||
|                   appKey: actionStep.appKey, |  | ||||||
|                   connection: { |  | ||||||
|                     createdAt: actionConnection.createdAt.getTime().toString(), |  | ||||||
|                     id: actionConnection.id, |  | ||||||
|                     verified: actionConnection.verified, |  | ||||||
|                   }, |  | ||||||
|                   iconUrl: `${appConfig.baseUrl}/apps/${actionStep.appKey}/assets/favicon.svg`, |  | ||||||
|                   id: actionStep.id, |  | ||||||
|                   key: 'translateText', |  | ||||||
|                   parameters: {}, |  | ||||||
|                   position: 1, |  | ||||||
|                   status: actionStep.status, |  | ||||||
|                   type: 'action', |  | ||||||
|                   webhookUrl: 'http://localhost:3000/null', |  | ||||||
|                 }, |  | ||||||
|               ], |  | ||||||
|             }, |  | ||||||
|           }, |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         expect(response.body).toEqual(expectedResponsePayload); |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| import Flow from '../../models/flow.js'; |  | ||||||
| import paginate from '../../helpers/pagination.js'; |  | ||||||
|  |  | ||||||
| const getFlows = async (_parent, params, context) => { |  | ||||||
|   const conditions = context.currentUser.can('read', 'Flow'); |  | ||||||
|   const userFlows = context.currentUser.$relatedQuery('flows'); |  | ||||||
|   const allFlows = Flow.query(); |  | ||||||
|   const baseQuery = conditions.isCreator ? userFlows : allFlows; |  | ||||||
|  |  | ||||||
|   const flowsQuery = baseQuery |  | ||||||
|     .clone() |  | ||||||
|     .joinRelated({ |  | ||||||
|       steps: true, |  | ||||||
|     }) |  | ||||||
|     .withGraphFetched({ |  | ||||||
|       steps: { |  | ||||||
|         connection: true, |  | ||||||
|       }, |  | ||||||
|     }) |  | ||||||
|     .where((builder) => { |  | ||||||
|       if (params.connectionId) { |  | ||||||
|         builder.where('steps.connection_id', params.connectionId); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (params.name) { |  | ||||||
|         builder.where('flows.name', 'ilike', `%${params.name}%`); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (params.appKey) { |  | ||||||
|         builder.where('steps.app_key', params.appKey); |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|     .groupBy('flows.id') |  | ||||||
|     .orderBy('active', 'desc') |  | ||||||
|     .orderBy('updated_at', 'desc'); |  | ||||||
|  |  | ||||||
|   return paginate(flowsQuery, params.limit, params.offset); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getFlows; |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| import Billing from '../../helpers/billing/index.ee.js'; |  | ||||||
|  |  | ||||||
| const getInvoices = async (_parent, _params, context) => { |  | ||||||
|   const subscription = await context.currentUser.$relatedQuery( |  | ||||||
|     'currentSubscription' |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   if (!subscription) { |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const invoices = await Billing.paddleClient.getInvoices( |  | ||||||
|     Number(subscription.paddleSubscriptionId) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   return invoices; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getInvoices; |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| import axios from '../../helpers/axios-with-proxy.js'; |  | ||||||
|  |  | ||||||
| const NOTIFICATIONS_URL = |  | ||||||
|   'https://notifications.automatisch.io/notifications.json'; |  | ||||||
|  |  | ||||||
| const getNotifications = async () => { |  | ||||||
|   try { |  | ||||||
|     const { data: notifications = [] } = await axios.get(NOTIFICATIONS_URL); |  | ||||||
|  |  | ||||||
|     return notifications; |  | ||||||
|   } catch (err) { |  | ||||||
|     return []; |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getNotifications; |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user