Compare commits
	
		
			431 Commits
		
	
	
		
			snackbar-o
			...
			AUT-1020
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | d4ddde5279 | ||
|   | 176d056aed | ||
|   | 0526ec5b06 | ||
|   | 061a9c6947 | ||
|   | ab307cdee0 | ||
|   | 9d5dac1701 | ||
|   | 751eb41e72 | ||
|   | f08dc25711 | ||
|   | 737eb31776 | ||
|   | d6abf283bc | ||
|   | bac4ab5aa4 | ||
|   | b5839390fd | ||
|   | d19271dae1 | ||
|   | ef5a09314e | ||
|   | ba52e298eb | ||
|   | b3c3998189 | ||
|   | 782f9b5c04 | ||
|   | 3079d8c605 | ||
|   | c5202d7b3e | ||
|   | fbae83f4de | ||
|   | 5b7b8c934f | ||
|   | b70223e824 | ||
|   | 9900bbbc8d | ||
|   | fc7f1ddd69 | ||
|   | 991b2f4bf7 | ||
|   | bb251b16a9 | ||
|   | 1b34a48a61 | ||
|   | 8c83b715fe | ||
|   | c122708b0b | ||
|   | 258d920ff2 | ||
|   | 6e5c0cc0c7 | ||
|   | 9dc82290b5 | ||
|   | bb73f90374 | ||
|   | 8a0720b0e3 | ||
|   | 88468c4f89 | ||
|   | d19a45592f | ||
|   | 21a921d25d | ||
|   | 3b2946aac5 | ||
|   | 196d555e8c | ||
|   | 28f39b5c7e | ||
|   | ec8ac17f4a | ||
|   | c45573349a | ||
|   | d36c9d43f6 | ||
|   | b06c744392 | ||
|   | 9548c93b4c | ||
|   | 4144944ab2 | ||
|   | 46b85519c1 | ||
|   | 5a83fc33ec | ||
|   | c80791267f | ||
|   | b30f97db3e | ||
|   | 717c81fa2b | ||
|   | ae188bc563 | ||
|   | fc4561221d | ||
|   | 5aeb4f8809 | ||
|   | c6c900bc39 | ||
|   | c18ab67a25 | ||
|   | 55ae1470d0 | ||
|   | a1136fdfb2 | ||
|   | dfe56d3aa2 | ||
|   | 3da5e13ecd | ||
|   | 40d0fe0db6 | ||
|   | 029fd2d0b0 | ||
|   | 12905ad733 | ||
|   | d257f59a3e | ||
|   | 1dc9646894 | ||
|   | c9281b4605 | ||
|   | e9d2ae5d67 | ||
|   | 0f8e05610b | ||
|   | 9ea2196e51 | ||
|   | 97327a9033 | ||
|   | 8bf11ba7d1 | ||
|   | 523833b015 | ||
|   | e25a651d26 | ||
|   | 92a8c1483d | ||
|   | 8da3448e9c | ||
|   | f2385d8916 | ||
|   | 17a8daa526 | ||
|   | 51e254f127 | ||
|   | feccf571cd | ||
|   | 9f8ce44c1b | ||
|   | a7cfe7f23b | ||
|   | 8eaf775ef2 | ||
|   | 7b4179a87f | ||
|   | 43281bcbfe | ||
|   | 8a35d47caf | ||
|   | 759e8b6c42 | ||
|   | a8b01244af | ||
|   | 6ffb16ac67 | ||
|   | 5b66cc6c8b | ||
|   | 9a96258265 | ||
|   | a6cc1566c7 | ||
|   | d8d6227125 | ||
|   | fbfa67e471 | ||
|   | ab897ada5a | ||
|   | 3bcd3f3cb7 | ||
|   | acbede8631 | ||
|   | fa8c7571d7 | ||
|   | a58575c5a1 | ||
|   | 51f7009a80 | ||
|   | ba24c77f06 | ||
|   | 6b712c9a90 | ||
|   | 033b15a158 | ||
|   | e398bb84d4 | ||
|   | 05e902ab0c | ||
|   | 36eee61cb5 | ||
|   | 2409ce6fce | ||
|   | 999d61e520 | ||
|   | 08d2418190 | ||
|   | 6c07faeaaf | ||
|   | 3f5cfbf5a2 | ||
|   | 28f7707c75 | ||
|   | dc8358f6ce | ||
|   | ffcda04677 | ||
|   | 64ef655abb | ||
|   | 67887b1220 | ||
|   | 07803d1263 | ||
|   | f5ff7f7e13 | ||
|   | 641d062b82 | ||
|   | bb28a06ee9 | ||
|   | 81338c60f2 | ||
|   | f47fa5d272 | ||
|   | 29e9e012a5 | ||
|   | f89ddb5847 | ||
|   | c721f063ef | ||
|   | a709565336 | ||
|   | 91e484aef1 | ||
|   | 6ba94dcc8e | ||
|   | 788530be45 | ||
|   | 7ed392e854 | ||
|   | 3932e554da | ||
|   | 1a21624618 | ||
|   | 9f292ff018 | ||
|   | dbb24b3a9b | ||
|   | 35b2639837 | ||
|   | 35951199cd | ||
|   | 79af909c51 | ||
|   | 3482aa7b76 | ||
|   | 5dbc1f59ef | ||
|   | 2166a3220e | ||
|   | 24a7d1ef10 | ||
|   | 18ffbb7317 | ||
|   | 363874de6a | ||
|   | 68d1719b11 | ||
|   | 1a75d81268 | ||
|   | 2163be4227 | ||
|   | b54afcd922 | ||
|   | 0a86641a0f | ||
|   | 18464c746a | ||
|   | ba92cddae1 | ||
|   | 2a4f8ed45f | ||
|   | 135a0028be | ||
|   | 4da6e8372f | ||
|   | 6a7cdf2570 | ||
|   | 73c929f25e | ||
|   | 754c2d41c2 | ||
|   | 7201e48111 | ||
|   | 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 | ||
|   | c413ab030b | ||
|   | 0f39007f92 | ||
|   | ab811daba7 | ||
|   | 1428bf8ffa | ||
|   | 53f7f38e23 | ||
|   | ce6ad9e0d3 | ||
|   | 17a0c6123a | ||
|   | 10edc5a8ad | ||
|   | bdf07d6bd5 | ||
|   | 10ff65e71a | ||
|   | 9395097fda | ||
|   | 47eb0e00e3 | ||
|   | 57d5f34ac5 | ||
|   | 06b040412a | ||
|   | cab040c74a | ||
|   | 21706a7d15 | ||
|   | 22e4b8aaeb | ||
|   | bd43a6021a | ||
|   | a90b58b6db | ||
|   | 92a9b096ec | ||
|   | 7a6aa99840 | ||
|   | 5657f0d793 | ||
|   | 798529007e | 
| @@ -36,7 +36,6 @@ services: | |||||||
|   keycloak: |   keycloak: | ||||||
|     image: quay.io/keycloak/keycloak:21.1 |     image: quay.io/keycloak/keycloak:21.1 | ||||||
|     restart: always |     restart: always | ||||||
|     container_name: keycloak |  | ||||||
|     environment: |     environment: | ||||||
|       - KEYCLOAK_ADMIN=admin |       - KEYCLOAK_ADMIN=admin | ||||||
|       - KEYCLOAK_ADMIN_PASSWORD=admin |       - KEYCLOAK_ADMIN_PASSWORD=admin | ||||||
|   | |||||||
| @@ -4,5 +4,9 @@ | |||||||
| **/.devcontainer | **/.devcontainer | ||||||
| **/.github | **/.github | ||||||
| **/.vscode | **/.vscode | ||||||
|  | **/.env | ||||||
|  | **/.env.test | ||||||
|  | **/.env.production | ||||||
|  | **/yarn-error.log | ||||||
| packages/docs | packages/docs | ||||||
| packages/e2e-test | packages/e2e-test | ||||||
|   | |||||||
							
								
								
									
										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}`); | ||||||
|  |             } | ||||||
| @@ -1,14 +1,25 @@ | |||||||
| # syntax=docker/dockerfile:1 | # syntax=docker/dockerfile:1 | ||||||
| FROM node:18-alpine | FROM node:18-alpine | ||||||
| WORKDIR /automatisch |  | ||||||
|  | ENV PORT 3000 | ||||||
|  |  | ||||||
|  | RUN \ | ||||||
|  |   apk --no-cache add --virtual build-dependencies python3 build-base git | ||||||
|  |  | ||||||
|  | WORKDIR /automatisch | ||||||
|  |  | ||||||
|  | # copy the app, note .dockerignore | ||||||
|  | COPY . /automatisch | ||||||
|  |  | ||||||
|  | RUN yarn | ||||||
|  |  | ||||||
|  | RUN cd packages/web && yarn build | ||||||
|  |  | ||||||
| RUN \ | RUN \ | ||||||
|   apk --no-cache add --virtual build-dependencies python3 build-base && \ |  | ||||||
|   yarn global add @automatisch/cli@0.10.0 --network-timeout 1000000 && \ |  | ||||||
|   rm -rf /usr/local/share/.cache/ && \ |   rm -rf /usr/local/share/.cache/ && \ | ||||||
|   apk del build-dependencies |   apk del build-dependencies | ||||||
|  |  | ||||||
| COPY ./entrypoint.sh /entrypoint.sh | COPY ./docker/entrypoint.sh /entrypoint.sh | ||||||
|  |  | ||||||
| EXPOSE 3000 | EXPOSE 3000 | ||||||
| ENTRYPOINT ["sh", "/entrypoint.sh"] | ENTRYPOINT ["sh", "/entrypoint.sh"] | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| # syntax=docker/dockerfile:1 |  | ||||||
| FROM node:18-alpine |  | ||||||
|  |  | ||||||
| ENV PORT 3000 |  | ||||||
|  |  | ||||||
| RUN \ |  | ||||||
|   apk --no-cache add --virtual build-dependencies python3 build-base git |  | ||||||
|  |  | ||||||
| RUN git clone https://github.com/automatisch/automatisch.git |  | ||||||
|  |  | ||||||
| WORKDIR /automatisch |  | ||||||
|  |  | ||||||
| RUN yarn install |  | ||||||
|  |  | ||||||
| RUN if [ "$WORKER" != "true" ]; then cd packages/web && yarn build; fi |  | ||||||
|  |  | ||||||
| RUN \ |  | ||||||
|   rm -rf /usr/local/share/.cache/ && \ |  | ||||||
|   apk del build-dependencies |  | ||||||
|  |  | ||||||
| COPY ./docker/entrypoint-cloud.sh /entrypoint-cloud.sh |  | ||||||
|  |  | ||||||
| EXPOSE 3000 |  | ||||||
| ENTRYPOINT ["sh", "/entrypoint-cloud.sh"] |  | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| # syntax=docker/dockerfile:1 | # syntax=docker/dockerfile:1 | ||||||
| FROM automatischio/automatisch:0.10.0 | FROM automatischio/automatisch:latest | ||||||
| WORKDIR /automatisch | WORKDIR /automatisch | ||||||
|  |  | ||||||
| RUN apk add --no-cache openssl dos2unix | RUN apk add --no-cache openssl dos2unix | ||||||
|   | |||||||
| @@ -1,13 +0,0 @@ | |||||||
| #!/bin/sh |  | ||||||
|  |  | ||||||
| set -e |  | ||||||
|  |  | ||||||
| cd packages/backend |  | ||||||
|  |  | ||||||
| if [ -n "$WORKER" ]; then |  | ||||||
|   yarn start:worker |  | ||||||
| else |  | ||||||
|   yarn db:migrate |  | ||||||
|   yarn db:seed:user |  | ||||||
|   yarn start |  | ||||||
| fi |  | ||||||
| @@ -2,8 +2,12 @@ | |||||||
|  |  | ||||||
| set -e | set -e | ||||||
|  |  | ||||||
|  | cd packages/backend | ||||||
|  |  | ||||||
| if [ -n "$WORKER" ]; then | if [ -n "$WORKER" ]; then | ||||||
|   automatisch start-worker |   yarn start:worker | ||||||
| else | else | ||||||
|   automatisch start |   yarn db:migrate | ||||||
|  |   yarn db:seed:user | ||||||
|  |   yarn start | ||||||
| fi | fi | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import appConfig from '../../src/config/app.js'; | |||||||
| import logger from '../../src/helpers/logger.js'; | import logger from '../../src/helpers/logger.js'; | ||||||
| import client from './client.js'; | import client from './client.js'; | ||||||
| import User from '../../src/models/user.js'; | import User from '../../src/models/user.js'; | ||||||
|  | import Config from '../../src/models/config.js'; | ||||||
| import Role from '../../src/models/role.js'; | import Role from '../../src/models/role.js'; | ||||||
| import '../../src/config/orm.js'; | import '../../src/config/orm.js'; | ||||||
| import process from 'process'; | import process from 'process'; | ||||||
| @@ -21,6 +22,14 @@ export async function createUser( | |||||||
|   email = 'user@automatisch.io', |   email = 'user@automatisch.io', | ||||||
|   password = 'sample' |   password = 'sample' | ||||||
| ) { | ) { | ||||||
|  |   if (appConfig.disableSeedUser) { | ||||||
|  |     logger.info('Seed user is disabled.'); | ||||||
|  |  | ||||||
|  |     process.exit(0); | ||||||
|  |  | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const UNIQUE_VIOLATION_CODE = '23505'; |   const UNIQUE_VIOLATION_CODE = '23505'; | ||||||
|  |  | ||||||
|   const role = await fetchAdminRole(); |   const role = await fetchAdminRole(); | ||||||
| @@ -37,6 +46,8 @@ export async function createUser( | |||||||
|     if (userCount === 0) { |     if (userCount === 0) { | ||||||
|       const user = await User.query().insertAndFetch(userParams); |       const user = await User.query().insertAndFetch(userParams); | ||||||
|       logger.info(`User has been saved: ${user.email}`); |       logger.info(`User has been saved: ${user.email}`); | ||||||
|  |  | ||||||
|  |       await Config.markInstallationCompleted(); | ||||||
|     } else { |     } else { | ||||||
|       logger.info('No need to seed a user.'); |       logger.info('No need to seed a user.'); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ | |||||||
|     "accounting": "^0.4.1", |     "accounting": "^0.4.1", | ||||||
|     "ajv-formats": "^2.1.1", |     "ajv-formats": "^2.1.1", | ||||||
|     "axios": "1.6.0", |     "axios": "1.6.0", | ||||||
|     "bcrypt": "^5.0.1", |     "bcrypt": "^5.1.0", | ||||||
|     "bullmq": "^3.0.0", |     "bullmq": "^3.0.0", | ||||||
|     "cors": "^2.8.5", |     "cors": "^2.8.5", | ||||||
|     "crypto-js": "^4.1.1", |     "crypto-js": "^4.1.1", | ||||||
| @@ -67,6 +67,7 @@ | |||||||
|     "pluralize": "^8.0.0", |     "pluralize": "^8.0.0", | ||||||
|     "raw-body": "^2.5.2", |     "raw-body": "^2.5.2", | ||||||
|     "showdown": "^2.1.0", |     "showdown": "^2.1.0", | ||||||
|  |     "uuid": "^9.0.1", | ||||||
|     "winston": "^3.7.1", |     "winston": "^3.7.1", | ||||||
|     "xmlrpc": "^1.3.2" |     "xmlrpc": "^1.3.2" | ||||||
|   }, |   }, | ||||||
| @@ -95,6 +96,7 @@ | |||||||
|     "url": "https://github.com/automatisch/automatisch/issues" |     "url": "https://github.com/automatisch/automatisch/issues" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|  |     "node-gyp": "^10.1.0", | ||||||
|     "nodemon": "^2.0.13", |     "nodemon": "^2.0.13", | ||||||
|     "supertest": "^6.3.3", |     "supertest": "^6.3.3", | ||||||
|     "vitest": "^1.1.3" |     "vitest": "^1.1.3" | ||||||
|   | |||||||
| @@ -0,0 +1,92 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Create record', | ||||||
|  |   key: 'createRecord', | ||||||
|  |   description: 'Creates a new record with fields that automatically populate.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Base', | ||||||
|  |       key: 'baseId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Base in which to create the record.', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listBases', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Table', | ||||||
|  |       key: 'tableId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       dependsOn: ['parameters.baseId'], | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listTables', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.baseId', | ||||||
|  |             value: '{parameters.baseId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |       additionalFields: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicFields', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listFields', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.baseId', | ||||||
|  |             value: '{parameters.baseId}', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.tableId', | ||||||
|  |             value: '{parameters.tableId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { baseId, tableId, ...rest } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const fields = Object.entries(rest).reduce((result, [key, value]) => { | ||||||
|  |       if (Array.isArray(value)) { | ||||||
|  |         result[key] = value.map((item) => item.value); | ||||||
|  |       } else if (value !== '') { | ||||||
|  |         result[key] = value; | ||||||
|  |       } | ||||||
|  |       return result; | ||||||
|  |     }, {}); | ||||||
|  |  | ||||||
|  |     const body = { | ||||||
|  |       typecast: true, | ||||||
|  |       fields, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.post(`/v0/${baseId}/${tableId}`, body); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: data, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
							
								
								
									
										174
									
								
								packages/backend/src/apps/airtable/actions/find-record/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								packages/backend/src/apps/airtable/actions/find-record/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  | import { URLSearchParams } from 'url'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Find record', | ||||||
|  |   key: 'findRecord', | ||||||
|  |   description: | ||||||
|  |     "Finds a record using simple field search or use Airtable's formula syntax to find a matching record.", | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Base', | ||||||
|  |       key: 'baseId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Base in which to create the record.', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listBases', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Table', | ||||||
|  |       key: 'tableId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       dependsOn: ['parameters.baseId'], | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listTables', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.baseId', | ||||||
|  |             value: '{parameters.baseId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Search by field', | ||||||
|  |       key: 'tableField', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       dependsOn: ['parameters.baseId', 'parameters.tableId'], | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listTableFields', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.baseId', | ||||||
|  |             value: '{parameters.baseId}', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.tableId', | ||||||
|  |             value: '{parameters.tableId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Search Value', | ||||||
|  |       key: 'searchValue', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         'The value of unique identifier for the record. For date values, please use the ISO format (e.g., "YYYY-MM-DD").', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Search for exact match?', | ||||||
|  |       key: 'exactMatch', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Yes', value: 'true' }, | ||||||
|  |         { label: 'No', value: 'false' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Search Formula', | ||||||
|  |       key: 'searchFormula', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       variables: true, | ||||||
|  |       description: | ||||||
|  |         'Instead, you have the option to use an Airtable search formula for locating records according to sophisticated criteria and across various fields.', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Limit to View', | ||||||
|  |       key: 'limitToView', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       dependsOn: ['parameters.baseId', 'parameters.tableId'], | ||||||
|  |       description: | ||||||
|  |         'You have the choice to restrict the search to a particular view ID if desired.', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listTableViews', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.baseId', | ||||||
|  |             value: '{parameters.baseId}', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.tableId', | ||||||
|  |             value: '{parameters.tableId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { | ||||||
|  |       baseId, | ||||||
|  |       tableId, | ||||||
|  |       tableField, | ||||||
|  |       searchValue, | ||||||
|  |       exactMatch, | ||||||
|  |       searchFormula, | ||||||
|  |       limitToView, | ||||||
|  |     } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     let filterByFormula; | ||||||
|  |  | ||||||
|  |     if (tableField && searchValue) { | ||||||
|  |       filterByFormula = | ||||||
|  |         exactMatch === 'true' | ||||||
|  |           ? `{${tableField}} = '${searchValue}'` | ||||||
|  |           : `LOWER({${tableField}}) = LOWER('${searchValue}')`; | ||||||
|  |     } else { | ||||||
|  |       filterByFormula = searchFormula; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const body = new URLSearchParams({ | ||||||
|  |       filterByFormula, | ||||||
|  |       view: limitToView, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.post( | ||||||
|  |       `/v0/${baseId}/${tableId}/listRecords`, | ||||||
|  |       body | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: data, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
							
								
								
									
										4
									
								
								packages/backend/src/apps/airtable/actions/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								packages/backend/src/apps/airtable/actions/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | import createRecord from './create-record/index.js'; | ||||||
|  | import findRecord from './find-record/index.js'; | ||||||
|  |  | ||||||
|  | export default [createRecord, findRecord]; | ||||||
							
								
								
									
										9
									
								
								packages/backend/src/apps/airtable/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								packages/backend/src/apps/airtable/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <svg width="256px" height="215px" viewBox="0 0 256 215" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" preserveAspectRatio="xMidYMid"> | ||||||
|  | 	<g> | ||||||
|  | 		<path d="M114.25873,2.70101695 L18.8604023,42.1756384 C13.5552723,44.3711638 13.6102328,51.9065311 18.9486282,54.0225085 L114.746142,92.0117514 C123.163769,95.3498757 132.537419,95.3498757 140.9536,92.0117514 L236.75256,54.0225085 C242.08951,51.9065311 242.145916,44.3711638 236.83934,42.1756384 L141.442459,2.70101695 C132.738459,-0.900338983 122.961284,-0.900338983 114.25873,2.70101695" fill="#FFBF00"></path> | ||||||
|  | 		<path d="M136.349071,112.756863 L136.349071,207.659101 C136.349071,212.173089 140.900664,215.263892 145.096461,213.600615 L251.844122,172.166219 C254.281184,171.200072 255.879376,168.845451 255.879376,166.224705 L255.879376,71.3224678 C255.879376,66.8084791 251.327783,63.7176768 247.131986,65.3809537 L140.384325,106.815349 C137.94871,107.781496 136.349071,110.136118 136.349071,112.756863" fill="#26B5F8"></path> | ||||||
|  | 		<path d="M111.422771,117.65355 L79.742409,132.949912 L76.5257763,134.504714 L9.65047684,166.548104 C5.4112904,168.593211 0.000578531073,165.503855 0.000578531073,160.794612 L0.000578531073,71.7210757 C0.000578531073,70.0173017 0.874160452,68.5463864 2.04568588,67.4384994 C2.53454463,66.9481944 3.08848814,66.5446689 3.66412655,66.2250305 C5.26231864,65.2661153 7.54173107,65.0101153 9.47981017,65.7766689 L110.890522,105.957098 C116.045234,108.002206 116.450206,115.225166 111.422771,117.65355" fill="#ED3049"></path> | ||||||
|  | 		<path d="M111.422771,117.65355 L79.742409,132.949912 L2.04568588,67.4384994 C2.53454463,66.9481944 3.08848814,66.5446689 3.66412655,66.2250305 C5.26231864,65.2661153 7.54173107,65.0101153 9.47981017,65.7766689 L110.890522,105.957098 C116.045234,108.002206 116.450206,115.225166 111.422771,117.65355" fill-opacity="0.25" fill="#000000"></path> | ||||||
|  | 	</g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.9 KiB | 
							
								
								
									
										38
									
								
								packages/backend/src/apps/airtable/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								packages/backend/src/apps/airtable/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | |||||||
|  | import crypto from 'crypto'; | ||||||
|  | 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 state = crypto.randomBytes(100).toString('base64url'); | ||||||
|  |   const codeVerifier = crypto.randomBytes(96).toString('base64url'); | ||||||
|  |   const codeChallenge = crypto | ||||||
|  |     .createHash('sha256') | ||||||
|  |     .update(codeVerifier) | ||||||
|  |     .digest('base64') | ||||||
|  |     .replace(/=/g, '') | ||||||
|  |     .replace(/\+/g, '-') | ||||||
|  |     .replace(/\//g, '_'); | ||||||
|  |  | ||||||
|  |   const searchParams = new URLSearchParams({ | ||||||
|  |     client_id: $.auth.data.clientId, | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|  |     response_type: 'code', | ||||||
|  |     scope: authScope.join(' '), | ||||||
|  |     state, | ||||||
|  |     code_challenge: codeChallenge, | ||||||
|  |     code_challenge_method: 'S256', | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const url = `https://airtable.com/oauth2/v1/authorize?${searchParams.toString()}`; | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     url, | ||||||
|  |     originalCodeChallenge: codeChallenge, | ||||||
|  |     originalState: state, | ||||||
|  |     codeVerifier, | ||||||
|  |   }); | ||||||
|  | } | ||||||
							
								
								
									
										48
									
								
								packages/backend/src/apps/airtable/auth/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								packages/backend/src/apps/airtable/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/airtable/connections/add', | ||||||
|  |       placeholder: null, | ||||||
|  |       description: | ||||||
|  |         'When asked to input a redirect URL in Airtable, 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.id; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default isStillVerified; | ||||||
							
								
								
									
										40
									
								
								packages/backend/src/apps/airtable/auth/refresh-token.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								packages/backend/src/apps/airtable/auth/refresh-token.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | import { URLSearchParams } from 'node:url'; | ||||||
|  |  | ||||||
|  | import authScope from '../common/auth-scope.js'; | ||||||
|  |  | ||||||
|  | const refreshToken = async ($) => { | ||||||
|  |   const params = new URLSearchParams({ | ||||||
|  |     client_id: $.auth.data.clientId, | ||||||
|  |     grant_type: 'refresh_token', | ||||||
|  |     refresh_token: $.auth.data.refreshToken, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const basicAuthToken = Buffer.from( | ||||||
|  |     $.auth.data.clientId + ':' + $.auth.data.clientSecret | ||||||
|  |   ).toString('base64'); | ||||||
|  |  | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     'https://airtable.com/oauth2/v1/token', | ||||||
|  |     params.toString(), | ||||||
|  |     { | ||||||
|  |       headers: { | ||||||
|  |         'Content-Type': 'application/x-www-form-urlencoded', | ||||||
|  |         Authorization: `Basic ${basicAuthToken}`, | ||||||
|  |       }, | ||||||
|  |       additionalProperties: { | ||||||
|  |         skipAddingAuthHeader: true, | ||||||
|  |       }, | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     refreshToken: data.refresh_token, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     refreshExpiresIn: data.refresh_expires_in, | ||||||
|  |     scope: authScope.join(' '), | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default refreshToken; | ||||||
| @@ -0,0 +1,56 @@ | |||||||
|  | import getCurrentUser from '../common/get-current-user.js'; | ||||||
|  |  | ||||||
|  | const verifyCredentials = async ($) => { | ||||||
|  |   if ($.auth.data.originalState !== $.auth.data.state) { | ||||||
|  |     throw new Error("The 'state' parameter does not match."); | ||||||
|  |   } | ||||||
|  |   if ($.auth.data.originalCodeChallenge !== $.auth.data.code_challenge) { | ||||||
|  |     throw new Error("The 'code challenge' parameter does not match."); | ||||||
|  |   } | ||||||
|  |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|  |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|  |   ); | ||||||
|  |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|  |   const basicAuthToken = Buffer.from( | ||||||
|  |     $.auth.data.clientId + ':' + $.auth.data.clientSecret | ||||||
|  |   ).toString('base64'); | ||||||
|  |  | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     'https://airtable.com/oauth2/v1/token', | ||||||
|  |     { | ||||||
|  |       code: $.auth.data.code, | ||||||
|  |       client_id: $.auth.data.clientId, | ||||||
|  |       redirect_uri: redirectUri, | ||||||
|  |       grant_type: 'authorization_code', | ||||||
|  |       code_verifier: $.auth.data.codeVerifier, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       headers: { | ||||||
|  |         'Content-Type': 'application/x-www-form-urlencoded', | ||||||
|  |         Authorization: `Basic ${basicAuthToken}`, | ||||||
|  |       }, | ||||||
|  |       additionalProperties: { | ||||||
|  |         skipAddingAuthHeader: true, | ||||||
|  |       }, | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const currentUser = await getCurrentUser($); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     clientId: $.auth.data.clientId, | ||||||
|  |     clientSecret: $.auth.data.clientSecret, | ||||||
|  |     scope: $.auth.data.scope, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     refreshExpiresIn: data.refresh_expires_in, | ||||||
|  |     refreshToken: data.refresh_token, | ||||||
|  |     screenName: currentUser.email, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default verifyCredentials; | ||||||
							
								
								
									
										12
									
								
								packages/backend/src/apps/airtable/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								packages/backend/src/apps/airtable/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | const addAuthHeader = ($, requestConfig) => { | ||||||
|  |   if ( | ||||||
|  |     !requestConfig.additionalProperties?.skipAddingAuthHeader && | ||||||
|  |     $.auth.data?.accessToken | ||||||
|  |   ) { | ||||||
|  |     requestConfig.headers.Authorization = `${$.auth.data.tokenType} ${$.auth.data.accessToken}`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default addAuthHeader; | ||||||
							
								
								
									
										12
									
								
								packages/backend/src/apps/airtable/common/auth-scope.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								packages/backend/src/apps/airtable/common/auth-scope.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | const authScope = [ | ||||||
|  |   'data.records:read', | ||||||
|  |   'data.records:write', | ||||||
|  |   'data.recordComments:read', | ||||||
|  |   'data.recordComments:write', | ||||||
|  |   'schema.bases:read', | ||||||
|  |   'schema.bases:write', | ||||||
|  |   'user.email:read', | ||||||
|  |   'webhook:manage', | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | export default authScope; | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | const getCurrentUser = async ($) => { | ||||||
|  |   const { data: currentUser } = await $.http.get('/v0/meta/whoami'); | ||||||
|  |   return currentUser; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default getCurrentUser; | ||||||
							
								
								
									
										6
									
								
								packages/backend/src/apps/airtable/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								packages/backend/src/apps/airtable/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | import listBases from './list-bases/index.js'; | ||||||
|  | import listTableFields from './list-table-fields/index.js'; | ||||||
|  | import listTableViews from './list-table-views/index.js'; | ||||||
|  | import listTables from './list-tables/index.js'; | ||||||
|  |  | ||||||
|  | export default [listBases, listTableFields, listTableViews, listTables]; | ||||||
| @@ -0,0 +1,28 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List bases', | ||||||
|  |   key: 'listBases', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const bases = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const params = {}; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get('/v0/meta/bases', { params }); | ||||||
|  |       params.offset = data.offset; | ||||||
|  |  | ||||||
|  |       if (data?.bases) { | ||||||
|  |         for (const base of data.bases) { | ||||||
|  |           bases.data.push({ | ||||||
|  |             value: base.id, | ||||||
|  |             name: base.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return bases; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List table fields', | ||||||
|  |   key: 'listTableFields', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const tableFields = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const { baseId, tableId } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     if (!baseId) { | ||||||
|  |       return tableFields; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = {}; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get(`/v0/meta/bases/${baseId}/tables`, { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |       params.offset = data.offset; | ||||||
|  |  | ||||||
|  |       if (data?.tables) { | ||||||
|  |         for (const table of data.tables) { | ||||||
|  |           if (table.id === tableId) { | ||||||
|  |             table.fields.forEach((field) => { | ||||||
|  |               tableFields.data.push({ | ||||||
|  |                 value: field.name, | ||||||
|  |                 name: field.name, | ||||||
|  |               }); | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return tableFields; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List table views', | ||||||
|  |   key: 'listTableViews', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const tableViews = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const { baseId, tableId } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     if (!baseId) { | ||||||
|  |       return tableViews; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = {}; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get(`/v0/meta/bases/${baseId}/tables`, { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |       params.offset = data.offset; | ||||||
|  |  | ||||||
|  |       if (data?.tables) { | ||||||
|  |         for (const table of data.tables) { | ||||||
|  |           if (table.id === tableId) { | ||||||
|  |             table.views.forEach((view) => { | ||||||
|  |               tableViews.data.push({ | ||||||
|  |                 value: view.id, | ||||||
|  |                 name: view.name, | ||||||
|  |               }); | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return tableViews; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,35 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List tables', | ||||||
|  |   key: 'listTables', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const tables = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const baseId = $.step.parameters.baseId; | ||||||
|  |  | ||||||
|  |     if (!baseId) { | ||||||
|  |       return tables; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = {}; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get(`/v0/meta/bases/${baseId}/tables`, { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |       params.offset = data.offset; | ||||||
|  |  | ||||||
|  |       if (data?.tables) { | ||||||
|  |         for (const table of data.tables) { | ||||||
|  |           tables.data.push({ | ||||||
|  |             value: table.id, | ||||||
|  |             name: table.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return tables; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,3 @@ | |||||||
|  | import listFields from './list-fields/index.js'; | ||||||
|  |  | ||||||
|  | export default [listFields]; | ||||||
| @@ -0,0 +1,86 @@ | |||||||
|  | const hasValue = (value) => value !== null && value !== undefined; | ||||||
|  |  | ||||||
|  | export default { | ||||||
|  |   name: 'List fields', | ||||||
|  |   key: 'listFields', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const options = []; | ||||||
|  |     const { baseId, tableId } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     if (!hasValue(baseId) || !hasValue(tableId)) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get(`/v0/meta/bases/${baseId}/tables`); | ||||||
|  |  | ||||||
|  |     const selectedTable = data.tables.find((table) => table.id === tableId); | ||||||
|  |  | ||||||
|  |     if (!selectedTable) return; | ||||||
|  |  | ||||||
|  |     selectedTable.fields.forEach((field) => { | ||||||
|  |       if (field.type === 'singleSelect') { | ||||||
|  |         options.push({ | ||||||
|  |           label: field.name, | ||||||
|  |           key: field.name, | ||||||
|  |           type: 'dropdown', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |           options: field.options.choices.map((choice) => ({ | ||||||
|  |             label: choice.name, | ||||||
|  |             value: choice.id, | ||||||
|  |           })), | ||||||
|  |         }); | ||||||
|  |       } else if (field.type === 'multipleSelects') { | ||||||
|  |         options.push({ | ||||||
|  |           label: field.name, | ||||||
|  |           key: field.name, | ||||||
|  |           type: 'dynamic', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |           fields: [ | ||||||
|  |             { | ||||||
|  |               label: 'Value', | ||||||
|  |               key: 'value', | ||||||
|  |               type: 'dropdown', | ||||||
|  |               required: false, | ||||||
|  |               variables: true, | ||||||
|  |               options: field.options.choices.map((choice) => ({ | ||||||
|  |                 label: choice.name, | ||||||
|  |                 value: choice.id, | ||||||
|  |               })), | ||||||
|  |             }, | ||||||
|  |           ], | ||||||
|  |         }); | ||||||
|  |       } else if (field.type === 'checkbox') { | ||||||
|  |         options.push({ | ||||||
|  |           label: field.name, | ||||||
|  |           key: field.name, | ||||||
|  |           type: 'dropdown', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |           options: [ | ||||||
|  |             { | ||||||
|  |               label: 'Yes', | ||||||
|  |               value: 'true', | ||||||
|  |             }, | ||||||
|  |             { | ||||||
|  |               label: 'No', | ||||||
|  |               value: 'false', | ||||||
|  |             }, | ||||||
|  |           ], | ||||||
|  |         }); | ||||||
|  |       } else { | ||||||
|  |         options.push({ | ||||||
|  |           label: field.name, | ||||||
|  |           key: field.name, | ||||||
|  |           type: 'string', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     return options; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										22
									
								
								packages/backend/src/apps/airtable/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								packages/backend/src/apps/airtable/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | import defineApp from '../../helpers/define-app.js'; | ||||||
|  | import addAuthHeader from './common/add-auth-header.js'; | ||||||
|  | import auth from './auth/index.js'; | ||||||
|  | import actions from './actions/index.js'; | ||||||
|  | import dynamicData from './dynamic-data/index.js'; | ||||||
|  | import dynamicFields from './dynamic-fields/index.js'; | ||||||
|  |  | ||||||
|  | export default defineApp({ | ||||||
|  |   name: 'Airtable', | ||||||
|  |   key: 'airtable', | ||||||
|  |   baseUrl: 'https://airtable.com', | ||||||
|  |   apiBaseUrl: 'https://api.airtable.com', | ||||||
|  |   iconUrl: '{BASE_URL}/apps/airtable/assets/favicon.svg', | ||||||
|  |   authDocUrl: '{DOCS_URL}/apps/airtable/connection', | ||||||
|  |   primaryColor: 'FFBF00', | ||||||
|  |   supportsConnections: true, | ||||||
|  |   beforeRequest: [addAuthHeader], | ||||||
|  |   auth, | ||||||
|  |   actions, | ||||||
|  |   dynamicData, | ||||||
|  |   dynamicFields, | ||||||
|  | }); | ||||||
							
								
								
									
										1
									
								
								packages/backend/src/apps/appwrite/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								packages/backend/src/apps/appwrite/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <svg xmlns:xlink="http://www.w3.org/1999/xlink" xmlns="http://www.w3.org/2000/svg" width="132" height="24" fill="none" viewBox="0 0 132 24"><path fill="#19191C" d="M38.557 19.495c2.16 0 3.25-1.113 3.725-1.87h.214c.094.805.664 1.562 1.779 1.562h2.111V16.82h-.545c-.38 0-.57-.213-.57-.545V6.776h-2.8v1.516h-.213c-.545-.758-1.684-1.824-3.772-1.824-3.321 0-5.789 2.748-5.789 6.514s2.515 6.513 5.86 6.513m.498-2.7c-1.969 0-3.51-1.445-3.51-3.79 0-2.297 1.494-3.86 3.487-3.86 1.898 0 3.487 1.397 3.487 3.86 0 2.108-1.352 3.79-3.463 3.79M48.04 24h2.799v-6.376h.213c.522.758 1.637 1.871 3.844 1.871 3.321 0 5.741-2.795 5.741-6.513 0-3.743-2.586-6.514-5.931-6.514-2.135 0-3.18 1.16-3.678 1.8h-.213V6.776h-2.776V24m6.263-7.134c-1.922 0-3.512-1.42-3.512-3.884 0-2.108 1.353-3.885 3.464-3.885 1.97 0 3.511 1.54 3.511 3.885 0 2.297-1.494 3.884-3.463 3.884M62.082 24h2.8v-6.376h.213c.522.758 1.637 1.871 3.843 1.871 3.321 0 5.51-2.795 5.51-6.513 0-3.743-2.355-6.514-5.7-6.514-2.135 0-3.179 1.16-3.677 1.8h-.214V6.776h-2.775zm6.263-7.134c-1.922 0-3.511-1.42-3.511-3.884 0-2.108 1.352-3.885 3.463-3.885 1.97 0 3.512 1.54 3.512 3.885 0 2.297-1.495 3.884-3.464 3.884m9.805 2.61h3.961l2.254-9.735h.143l2.253 9.735H90.7l3.153-12.412h-2.821l-2.254 9.759h-.214l-2.253-9.759h-3.725l-2.278 9.759h-.213l-2.23-9.759h-2.99l3.274 12.412m17.123 0h2.8V13.34c0-2.345 1.09-3.79 3.131-3.79h1.233V6.756h-.925c-1.59 0-2.8 1.09-3.274 2.132h-.19V7.064h-2.775zm21.057 0h2.183v-2.487h-2.159c-.854 0-1.21-.38-1.21-1.256V9.528h3.511V7.064h-3.511V3.582h-2.657v3.482h-2.325v2.464h2.159v6.229c0 2.63 1.589 3.719 4.009 3.719m9.693.019c2.586 0 4.864-1.279 5.67-3.86l-2.562-.616c-.451 1.373-1.755 2.084-3.131 2.084-2.041 0-3.393-1.326-3.417-3.41h9.419v-.782c0-3.695-2.301-6.443-6.097-6.443-3.346 0-6.216 2.63-6.216 6.537 0 3.79 2.538 6.49 6.334 6.49m-3.416-7.84c.166-1.492 1.518-2.747 3.298-2.747 1.708 0 3.108 1.066 3.25 2.747h-6.548"/><path fill="#19191C" fill-rule="evenodd" d="M108.916 19.476h-2.8V9.528h-2.182V7.064h4.982z" clip-rule="evenodd"/><path fill="#19191C" d="M107.309 5.342c1.02 0 1.779-.758 1.779-1.753 0-.971-.759-1.73-1.779-1.73-1.021 0-1.78.759-1.78 1.73 0 .995.759 1.753 1.78 1.753"/><path fill="#FD366E" d="M24.443 16.432v5.478H10.752c-3.989 0-7.472-2.203-9.335-5.478A11.041 11.041 0 0 1 0 11.695v-1.48a10.97 10.97 0 0 1 .381-2.247C1.661 3.368 5.82 0 10.751 0c4.934 0 9.092 3.37 10.371 7.967h-5.854c-.96-1.499-2.624-2.49-4.516-2.49s-3.555.991-4.516 2.49a5.47 5.47 0 0 0-.67 1.494 5.562 5.562 0 0 0-.202 1.494 5.5 5.5 0 0 0 1.69 3.983 5.32 5.32 0 0 0 3.698 1.494h13.69"/><path fill="#FD366E" d="M24.443 9.46v5.478h-9.994a5.5 5.5 0 0 0 1.691-3.983 5.56 5.56 0 0 0-.203-1.494h8.506"/></svg> | ||||||
| After Width: | Height: | Size: 2.6 KiB | 
							
								
								
									
										65
									
								
								packages/backend/src/apps/appwrite/auth/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								packages/backend/src/apps/appwrite/auth/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | |||||||
|  | import verifyCredentials from './verify-credentials.js'; | ||||||
|  | import isStillVerified from './is-still-verified.js'; | ||||||
|  |  | ||||||
|  | export default { | ||||||
|  |   fields: [ | ||||||
|  |     { | ||||||
|  |       key: 'screenName', | ||||||
|  |       label: 'Screen Name', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: | ||||||
|  |         'Screen name of your connection to be used on Automatisch UI.', | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'projectId', | ||||||
|  |       label: 'Project ID', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: 'Project ID of your Appwrite project.', | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'apiKey', | ||||||
|  |       label: 'API Key', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: 'API key of your Appwrite project.', | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'instanceUrl', | ||||||
|  |       label: 'Appwrite instance URL', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       readOnly: false, | ||||||
|  |       placeholder: '', | ||||||
|  |       description: '', | ||||||
|  |       clickToCopy: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'host', | ||||||
|  |       label: 'Host Name', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: 'Host name of your Appwrite project.', | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   verifyCredentials, | ||||||
|  |   isStillVerified, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | import verifyCredentials from './verify-credentials.js'; | ||||||
|  |  | ||||||
|  | const isStillVerified = async ($) => { | ||||||
|  |   await verifyCredentials($); | ||||||
|  |   return true; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default isStillVerified; | ||||||
| @@ -0,0 +1,5 @@ | |||||||
|  | const verifyCredentials = async ($) => { | ||||||
|  |   await $.http.get('/v1/users'); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default verifyCredentials; | ||||||
							
								
								
									
										16
									
								
								packages/backend/src/apps/appwrite/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								packages/backend/src/apps/appwrite/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | const addAuthHeader = ($, requestConfig) => { | ||||||
|  |   requestConfig.headers['Content-Type'] = 'application/json'; | ||||||
|  |  | ||||||
|  |   if ($.auth.data?.apiKey && $.auth.data?.projectId) { | ||||||
|  |     requestConfig.headers['X-Appwrite-Project'] = $.auth.data.projectId; | ||||||
|  |     requestConfig.headers['X-Appwrite-Key'] = $.auth.data.apiKey; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if ($.auth.data?.host) { | ||||||
|  |     requestConfig.headers['Host'] = $.auth.data.host; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default addAuthHeader; | ||||||
							
								
								
									
										13
									
								
								packages/backend/src/apps/appwrite/common/set-base-url.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/backend/src/apps/appwrite/common/set-base-url.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | const setBaseUrl = ($, requestConfig) => { | ||||||
|  |   const instanceUrl = $.auth.data.instanceUrl; | ||||||
|  |  | ||||||
|  |   if (instanceUrl) { | ||||||
|  |     requestConfig.baseURL = instanceUrl; | ||||||
|  |   } else if ($.app.apiBaseUrl) { | ||||||
|  |     requestConfig.baseURL = $.app.apiBaseUrl; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default setBaseUrl; | ||||||
							
								
								
									
										4
									
								
								packages/backend/src/apps/appwrite/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								packages/backend/src/apps/appwrite/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | import listCollections from './list-collections/index.js'; | ||||||
|  | import listDatabases from './list-databases/index.js'; | ||||||
|  |  | ||||||
|  | export default [listCollections, listDatabases]; | ||||||
| @@ -0,0 +1,44 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List collections', | ||||||
|  |   key: 'listCollections', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const collections = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const databaseId = $.step.parameters.databaseId; | ||||||
|  |  | ||||||
|  |     if (!databaseId) { | ||||||
|  |       return collections; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       queries: [ | ||||||
|  |         JSON.stringify({ | ||||||
|  |           method: 'orderAsc', | ||||||
|  |           attribute: 'name', | ||||||
|  |         }), | ||||||
|  |         JSON.stringify({ | ||||||
|  |           method: 'limit', | ||||||
|  |           values: [100], | ||||||
|  |         }), | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get( | ||||||
|  |       `/v1/databases/${databaseId}/collections`, | ||||||
|  |       { params } | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     if (data?.collections) { | ||||||
|  |       for (const collection of data.collections) { | ||||||
|  |         collections.data.push({ | ||||||
|  |           value: collection.$id, | ||||||
|  |           name: collection.name, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return collections; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,36 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List databases', | ||||||
|  |   key: 'listDatabases', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const databases = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       queries: [ | ||||||
|  |         JSON.stringify({ | ||||||
|  |           method: 'orderAsc', | ||||||
|  |           attribute: 'name', | ||||||
|  |         }), | ||||||
|  |         JSON.stringify({ | ||||||
|  |           method: 'limit', | ||||||
|  |           values: [100], | ||||||
|  |         }), | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get('/v1/databases', { params }); | ||||||
|  |  | ||||||
|  |     if (data?.databases) { | ||||||
|  |       for (const database of data.databases) { | ||||||
|  |         databases.data.push({ | ||||||
|  |           value: database.$id, | ||||||
|  |           name: database.name, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return databases; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										21
									
								
								packages/backend/src/apps/appwrite/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								packages/backend/src/apps/appwrite/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | import defineApp from '../../helpers/define-app.js'; | ||||||
|  | import addAuthHeader from './common/add-auth-header.js'; | ||||||
|  | import setBaseUrl from './common/set-base-url.js'; | ||||||
|  | import auth from './auth/index.js'; | ||||||
|  | import triggers from './triggers/index.js'; | ||||||
|  | import dynamicData from './dynamic-data/index.js'; | ||||||
|  |  | ||||||
|  | export default defineApp({ | ||||||
|  |   name: 'Appwrite', | ||||||
|  |   key: 'appwrite', | ||||||
|  |   baseUrl: 'https://appwrite.io', | ||||||
|  |   apiBaseUrl: 'https://cloud.appwrite.io', | ||||||
|  |   iconUrl: '{BASE_URL}/apps/appwrite/assets/favicon.svg', | ||||||
|  |   authDocUrl: '{DOCS_URL}/apps/appwrite/connection', | ||||||
|  |   primaryColor: 'FD366E', | ||||||
|  |   supportsConnections: true, | ||||||
|  |   beforeRequest: [setBaseUrl, addAuthHeader], | ||||||
|  |   auth, | ||||||
|  |   triggers, | ||||||
|  |   dynamicData, | ||||||
|  | }); | ||||||
							
								
								
									
										3
									
								
								packages/backend/src/apps/appwrite/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								packages/backend/src/apps/appwrite/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | import newDocuments from './new-documents/index.js'; | ||||||
|  |  | ||||||
|  | export default [newDocuments]; | ||||||
| @@ -0,0 +1,104 @@ | |||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New documents', | ||||||
|  |   key: 'newDocuments', | ||||||
|  |   pollInterval: 15, | ||||||
|  |   description: 'Triggers when a new document is created.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Database', | ||||||
|  |       key: 'databaseId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listDatabases', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Collection', | ||||||
|  |       key: 'collectionId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       dependsOn: ['parameters.databaseId'], | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listCollections', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.databaseId', | ||||||
|  |             value: '{parameters.databaseId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { databaseId, collectionId } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const limit = 1; | ||||||
|  |     let lastDocumentId = undefined; | ||||||
|  |     let offset = 0; | ||||||
|  |     let documentCount = 0; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const params = { | ||||||
|  |         queries: [ | ||||||
|  |           JSON.stringify({ | ||||||
|  |             method: 'orderDesc', | ||||||
|  |             attribute: '$createdAt', | ||||||
|  |           }), | ||||||
|  |           JSON.stringify({ | ||||||
|  |             method: 'limit', | ||||||
|  |             values: [limit], | ||||||
|  |           }), | ||||||
|  |           // An invalid cursor shouldn't be sent. | ||||||
|  |           lastDocumentId && | ||||||
|  |             JSON.stringify({ | ||||||
|  |               method: 'cursorAfter', | ||||||
|  |               values: [lastDocumentId], | ||||||
|  |             }), | ||||||
|  |         ].filter(Boolean), | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       const { data } = await $.http.get( | ||||||
|  |         `/v1/databases/${databaseId}/collections/${collectionId}/documents`, | ||||||
|  |         { params } | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       const documents = data?.documents; | ||||||
|  |       documentCount = documents?.length; | ||||||
|  |       offset = offset + limit; | ||||||
|  |       lastDocumentId = documents[documentCount - 1]?.$id; | ||||||
|  |  | ||||||
|  |       if (!documentCount) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       for (const document of documents) { | ||||||
|  |         $.pushTriggerItem({ | ||||||
|  |           raw: document, | ||||||
|  |           meta: { | ||||||
|  |             internalId: document.$id, | ||||||
|  |           }, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } while (documentCount === limit); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
							
								
								
									
										216
									
								
								packages/backend/src/apps/asana/actions/create-project/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										216
									
								
								packages/backend/src/apps/asana/actions/create-project/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,216 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  | import omitBy from 'lodash/omitBy.js'; | ||||||
|  | import isEmpty from 'lodash/isEmpty.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Create project', | ||||||
|  |   key: 'createProject', | ||||||
|  |   description: 'Creates a new project.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Workspace', | ||||||
|  |       key: 'workspaceId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listWorkspaces', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Team', | ||||||
|  |       key: 'teamId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listTeams', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.workspaceId', | ||||||
|  |             value: '{parameters.workspaceId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Due date', | ||||||
|  |       key: 'dueDate', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: 'Example due on: 2019-09-15', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Name', | ||||||
|  |       key: 'name', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Notes', | ||||||
|  |       key: 'notes', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: 'You can format the notes using html.', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Is the notes rich text?', | ||||||
|  |       key: 'richText', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { | ||||||
|  |           label: 'No', | ||||||
|  |           value: 'false', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Yes', | ||||||
|  |           value: 'true', | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Default View', | ||||||
|  |       key: 'defaultView', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { | ||||||
|  |           label: 'List', | ||||||
|  |           value: 'list', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Board', | ||||||
|  |           value: 'board', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Calendar', | ||||||
|  |           value: 'calendar', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Timeline', | ||||||
|  |           value: 'timeline', | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Owner', | ||||||
|  |       key: 'ownerId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       dependsOn: ['parameters.workspaceId'], | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listUsers', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.workspaceId', | ||||||
|  |             value: '{parameters.workspaceId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Followers', | ||||||
|  |       key: 'followerIds', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Follower', | ||||||
|  |           key: 'followerId', | ||||||
|  |           type: 'dropdown', | ||||||
|  |           required: false, | ||||||
|  |           dependsOn: ['parameters.workspaceId'], | ||||||
|  |           description: '', | ||||||
|  |           variables: true, | ||||||
|  |           source: { | ||||||
|  |             type: 'query', | ||||||
|  |             name: 'getDynamicData', | ||||||
|  |             arguments: [ | ||||||
|  |               { | ||||||
|  |                 name: 'key', | ||||||
|  |                 value: 'listUsers', | ||||||
|  |               }, | ||||||
|  |               { | ||||||
|  |                 name: 'parameters.workspaceId', | ||||||
|  |                 value: '{parameters.workspaceId}', | ||||||
|  |               }, | ||||||
|  |             ], | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { | ||||||
|  |       workspaceId, | ||||||
|  |       teamId, | ||||||
|  |       dueDate, | ||||||
|  |       name, | ||||||
|  |       notes, | ||||||
|  |       richText, | ||||||
|  |       defaultView, | ||||||
|  |       ownerId, | ||||||
|  |       followerIds, | ||||||
|  |     } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const allFollowers = followerIds | ||||||
|  |       .map((followerId) => followerId.followerId) | ||||||
|  |       .filter(Boolean); | ||||||
|  |  | ||||||
|  |     const data = { | ||||||
|  |       workspace: workspaceId, | ||||||
|  |       team: teamId, | ||||||
|  |       due_on: dueDate, | ||||||
|  |       name, | ||||||
|  |       default_view: defaultView, | ||||||
|  |       owner: ownerId, | ||||||
|  |       followers: allFollowers, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if (richText === 'true') { | ||||||
|  |       data.html_notes = notes; | ||||||
|  |     } else { | ||||||
|  |       data.notes = notes; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const filteredData = omitBy(data, isEmpty); | ||||||
|  |  | ||||||
|  |     const response = await $.http.post('/1.0/projects', { data: filteredData }); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: response.data, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
							
								
								
									
										312
									
								
								packages/backend/src/apps/asana/actions/create-task/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										312
									
								
								packages/backend/src/apps/asana/actions/create-task/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,312 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  | import omitBy from 'lodash/omitBy.js'; | ||||||
|  | import isEmpty from 'lodash/isEmpty.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Create task', | ||||||
|  |   key: 'createTask', | ||||||
|  |   description: 'Creates a new task.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Workspace', | ||||||
|  |       key: 'workspaceId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listWorkspaces', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Project', | ||||||
|  |       key: 'projectId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listProjects', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.workspaceId', | ||||||
|  |             value: '{parameters.workspaceId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Section', | ||||||
|  |       key: 'sectionId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listSections', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.projectId', | ||||||
|  |             value: '{parameters.projectId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Due date type', | ||||||
|  |       key: 'dueDateType', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: "If not filled in, 'Date & time' will be assumed.", | ||||||
|  |       options: [ | ||||||
|  |         { | ||||||
|  |           label: 'Date & time', | ||||||
|  |           value: 'at', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Date only', | ||||||
|  |           value: 'on', | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Due date (date & time)', | ||||||
|  |       key: 'dueDate', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Example due at: 2019-09-15T02:06:58.147Z, example due on: 2019-09-15', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Name', | ||||||
|  |       key: 'name', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Description', | ||||||
|  |       key: 'description', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: 'You can format the description using html.', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Is the description rich text?', | ||||||
|  |       key: 'richText', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { | ||||||
|  |           label: 'No', | ||||||
|  |           value: 'false', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Yes', | ||||||
|  |           value: 'true', | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mark Task as complete?', | ||||||
|  |       key: 'taskCompleted', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { | ||||||
|  |           label: 'No', | ||||||
|  |           value: 'false', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Yes', | ||||||
|  |           value: 'true', | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mark Task as liked?', | ||||||
|  |       key: 'taskLiked', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { | ||||||
|  |           label: 'No', | ||||||
|  |           value: 'false', | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           label: 'Yes', | ||||||
|  |           value: 'true', | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Assignee', | ||||||
|  |       key: 'assigneeId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       dependsOn: ['parameters.workspaceId'], | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listUsers', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.workspaceId', | ||||||
|  |             value: '{parameters.workspaceId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Followers', | ||||||
|  |       key: 'followerIds', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Follower', | ||||||
|  |           key: 'followerId', | ||||||
|  |           type: 'dropdown', | ||||||
|  |           required: false, | ||||||
|  |           dependsOn: ['parameters.workspaceId'], | ||||||
|  |           description: '', | ||||||
|  |           variables: true, | ||||||
|  |           source: { | ||||||
|  |             type: 'query', | ||||||
|  |             name: 'getDynamicData', | ||||||
|  |             arguments: [ | ||||||
|  |               { | ||||||
|  |                 name: 'key', | ||||||
|  |                 value: 'listUsers', | ||||||
|  |               }, | ||||||
|  |               { | ||||||
|  |                 name: 'parameters.workspaceId', | ||||||
|  |                 value: '{parameters.workspaceId}', | ||||||
|  |               }, | ||||||
|  |             ], | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Tags', | ||||||
|  |       key: 'tagIds', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Tag', | ||||||
|  |           key: 'tagId', | ||||||
|  |           type: 'dropdown', | ||||||
|  |           required: false, | ||||||
|  |           dependsOn: ['parameters.workspaceId'], | ||||||
|  |           description: '', | ||||||
|  |           variables: true, | ||||||
|  |           source: { | ||||||
|  |             type: 'query', | ||||||
|  |             name: 'getDynamicData', | ||||||
|  |             arguments: [ | ||||||
|  |               { | ||||||
|  |                 name: 'key', | ||||||
|  |                 value: 'listTags', | ||||||
|  |               }, | ||||||
|  |               { | ||||||
|  |                 name: 'parameters.workspaceId', | ||||||
|  |                 value: '{parameters.workspaceId}', | ||||||
|  |               }, | ||||||
|  |             ], | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { | ||||||
|  |       workspaceId, | ||||||
|  |       projectId, | ||||||
|  |       sectionId, | ||||||
|  |       dueDateType, | ||||||
|  |       dueDate, | ||||||
|  |       name, | ||||||
|  |       description, | ||||||
|  |       richText, | ||||||
|  |       taskCompleted, | ||||||
|  |       taskLiked, | ||||||
|  |       assigneeId, | ||||||
|  |       followerIds, | ||||||
|  |       tagIds, | ||||||
|  |     } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const allFollowers = followerIds | ||||||
|  |       .map((followerId) => followerId.followerId) | ||||||
|  |       .filter(Boolean); | ||||||
|  |  | ||||||
|  |     const allTags = tagIds.map((tagId) => tagId.tagId).filter(Boolean); | ||||||
|  |  | ||||||
|  |     const data = { | ||||||
|  |       name, | ||||||
|  |       completed: taskCompleted, | ||||||
|  |       liked: taskLiked, | ||||||
|  |       assignee: assigneeId, | ||||||
|  |       assignee_section: sectionId, | ||||||
|  |       followers: allFollowers, | ||||||
|  |       projects: projectId, | ||||||
|  |       tags: allTags, | ||||||
|  |       workspace: workspaceId, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if (richText === 'true') { | ||||||
|  |       data.html_notes = description; | ||||||
|  |     } else { | ||||||
|  |       data.notes = description; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (dueDateType === 'on') { | ||||||
|  |       data.due_on = dueDate; | ||||||
|  |     } else { | ||||||
|  |       data.due_at = dueDate; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const filteredData = omitBy(data, isEmpty); | ||||||
|  |  | ||||||
|  |     const response = await $.http.post('/1.0/tasks', { data: filteredData }); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: response.data, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,49 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Find project', | ||||||
|  |   key: 'findProject', | ||||||
|  |   description: 'Finds an existing project.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Workspace', | ||||||
|  |       key: 'workspaceId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listWorkspaces', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Name', | ||||||
|  |       key: 'name', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { workspaceId, name } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get( | ||||||
|  |       `/1.0/workspaces/${workspaceId}/projects` | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     const project = data.data.find((project) => project.name === name); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: project, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,69 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Find task in a project', | ||||||
|  |   key: 'findTaskInProject', | ||||||
|  |   description: 'Finds an existing task within a project.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Workspace', | ||||||
|  |       key: 'workspaceId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listWorkspaces', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Project', | ||||||
|  |       key: 'projectId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listProjects', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'parameters.workspaceId', | ||||||
|  |             value: '{parameters.workspaceId}', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Task Name', | ||||||
|  |       key: 'taskName', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { projectId, taskName } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get(`/1.0/projects/${projectId}/tasks`); | ||||||
|  |  | ||||||
|  |     const task = data.data.find((task) => task.name === taskName); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: task, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
							
								
								
									
										6
									
								
								packages/backend/src/apps/asana/actions/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								packages/backend/src/apps/asana/actions/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | import createProject from './create-project/index.js'; | ||||||
|  | import createTask from './create-task/index.js'; | ||||||
|  | import findProject from './find-project/index.js'; | ||||||
|  | import findTaskInProject from './find-task-in-project/index.js'; | ||||||
|  |  | ||||||
|  | export default [createProject, createTask, findProject, findTaskInProject]; | ||||||
							
								
								
									
										8
									
								
								packages/backend/src/apps/asana/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								packages/backend/src/apps/asana/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | <svg width="555" height="110" viewBox="0 0 555 110" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  | <path fill-rule="evenodd" clip-rule="evenodd" d="M541.011 83.513C541.347 87.339 544.407 92.209 549.709 92.209H552.811C554.014 92.209 555 91.2232 555 90.0197V21.7948H554.986C554.923 20.6454 553.974 19.7248 552.811 19.7248H543.199C542.036 19.7248 541.087 20.6454 541.023 21.7948H541.011V27.3385C535.122 20.0789 525.836 17.0657 516.525 17.0657C495.359 17.0657 478.202 34.2365 478.202 55.419C478.202 76.6027 495.359 93.7741 516.525 93.7741V93.7759C525.836 93.7759 535.983 90.1606 541.01 83.5042L541.011 83.513V83.513ZM516.562 80.3509C503.101 80.3509 492.188 69.1896 492.188 55.4192C492.188 41.6511 503.101 30.4892 516.562 30.4892C530.022 30.4892 540.934 41.6511 540.934 55.4192C540.934 69.1896 530.022 80.3509 516.562 80.3509V80.3509Z" fill="#690031"/> | ||||||
|  | <path fill-rule="evenodd" clip-rule="evenodd" d="M466.05 85.8589L466.045 50.5554H466.046C466.046 30.655 453.501 17.2299 433.497 17.2299C423.947 17.2299 416.119 22.7561 413.355 27.5032C412.757 23.7913 410.788 19.8896 404.681 19.8896H401.569C400.365 19.8896 399.382 20.876 399.382 22.0795V83.6835C399.382 83.6853 399.382 83.69 399.382 83.6929V90.3102H399.394C399.457 91.4579 400.408 92.3796 401.57 92.3796H411.182C411.33 92.3796 411.474 92.3621 411.613 92.3347C411.677 92.3225 411.736 92.2975 411.798 92.28C411.869 92.2579 411.944 92.241 412.012 92.213C412.097 92.1775 412.175 92.1298 412.255 92.0855C412.294 92.0617 412.334 92.0448 412.372 92.0197C412.468 91.958 412.556 91.8835 412.641 91.8072C412.655 91.7932 412.672 91.7839 412.686 91.7711C412.781 91.6785 412.868 91.5766 412.946 91.4707C412.946 91.4689 412.946 91.4689 412.946 91.4689C413.187 91.1382 413.333 90.7399 413.357 90.3102H413.369V50.0081C413.369 39.3201 422.028 30.655 432.709 30.655C443.389 30.655 452.047 39.3201 452.047 50.0081L452.056 83.6952L452.058 83.6835C452.058 83.7132 452.063 83.7441 452.063 83.7761V90.3102H452.076C452.139 91.4579 453.089 92.3796 454.251 92.3796H463.864C464.012 92.3796 464.156 92.3621 464.295 92.3347C464.352 92.3243 464.404 92.3015 464.46 92.2858C464.538 92.2631 464.619 92.2433 464.695 92.213C464.773 92.1804 464.845 92.135 464.92 92.0931C464.965 92.0675 465.013 92.0489 465.056 92.0197C465.145 91.9615 465.226 91.8911 465.306 91.8212C465.326 91.8026 465.349 91.7886 465.368 91.7694C465.459 91.6814 465.54 91.5865 465.615 91.487C465.62 91.4794 465.626 91.4736 465.632 91.466C465.868 91.1382 466.013 90.7428 466.038 90.3161C466.038 90.3131 466.039 90.3102 466.039 90.3102H466.052V85.86L466.05 85.8589" fill="#690031"/> | ||||||
|  | <path fill-rule="evenodd" clip-rule="evenodd" d="M365.94 83.5127C366.276 87.3387 369.336 92.2088 374.638 92.2088H377.74C378.943 92.2088 379.927 91.223 379.927 90.0195V21.7945H379.915C379.852 20.6452 378.901 19.7246 377.74 19.7246H368.128C366.965 19.7246 366.016 20.6452 365.951 21.7945H365.94V27.3382C360.05 20.0786 350.764 17.0654 341.453 17.0654C320.288 17.0654 303.131 34.2362 303.131 55.4188C303.131 76.6025 320.288 93.7739 341.453 93.7739V93.7756C350.764 93.7756 360.912 90.1604 365.939 83.504L365.94 83.5127V83.5127ZM341.49 80.3506C328.03 80.3506 317.117 69.1893 317.117 55.4189C317.117 41.6509 328.03 30.489 341.49 30.489C354.952 30.489 365.862 41.6509 365.862 55.4189C365.862 69.1893 354.952 80.3506 341.49 80.3506V80.3506Z" fill="#690031"/> | ||||||
|  | <path fill-rule="evenodd" clip-rule="evenodd" d="M246.284 73.7415C252.702 78.1905 259.706 80.3513 266.437 80.3513C272.85 80.3513 279.479 77.0242 279.479 71.2337C279.479 63.5024 265.033 62.2995 255.957 59.2124C246.88 56.1252 239.061 49.7437 239.061 39.4092C239.061 23.5956 253.14 17.0645 266.281 17.0645C274.607 17.0645 283.198 19.8121 288.767 23.7482C290.686 25.2027 289.517 26.8726 289.517 26.8726L284.201 34.4716C283.603 35.3276 282.559 36.067 281.059 35.1407C279.559 34.2149 274.298 30.4884 266.281 30.4884C258.263 30.4884 253.434 34.1939 253.434 38.7868C253.434 44.2943 259.711 46.0266 267.063 47.9038C279.875 51.36 293.852 55.5144 293.852 71.2337C293.852 85.1665 280.829 93.777 266.437 93.777C255.53 93.777 246.244 90.6654 238.456 84.9459C236.834 83.3208 237.967 81.8121 237.967 81.8121L243.257 74.2515C244.334 72.8378 245.691 73.331 246.284 73.7415" fill="#690031"/> | ||||||
|  | <path fill-rule="evenodd" clip-rule="evenodd" d="M209.331 83.5127C209.668 87.3387 212.728 92.2088 218.03 92.2088H221.132C222.334 92.2088 223.32 91.223 223.32 90.0195V21.7945H223.307C223.244 20.6452 222.294 19.7246 221.132 19.7246H211.519C210.357 19.7246 209.408 20.6452 209.343 21.7945H209.331V27.3382C203.442 20.0786 194.156 17.0654 184.845 17.0654C163.68 17.0654 146.522 34.2362 146.522 55.4188C146.522 76.6025 163.68 93.7739 184.845 93.7739V93.7756C194.156 93.7756 204.304 90.1604 209.33 83.504L209.331 83.5127V83.5127ZM184.883 80.3506C171.422 80.3506 160.509 69.1893 160.509 55.4189C160.509 41.6509 171.422 30.489 184.883 30.489C198.343 30.489 209.255 41.6509 209.255 55.4189C209.255 69.1893 198.343 80.3506 184.883 80.3506V80.3506Z" fill="#690031"/> | ||||||
|  | <path fill-rule="evenodd" clip-rule="evenodd" d="M92.794 58.0274C78.5507 58.0274 67.0041 69.5741 67.0041 83.8185C67.0041 98.0618 78.5507 109.608 92.794 109.608C107.037 109.608 118.584 98.0618 118.584 83.8185C118.584 69.5741 107.037 58.0274 92.794 58.0274V58.0274ZM25.7899 58.0298C11.5466 58.0298 0 69.5741 0 83.8186C0 98.0618 11.5466 109.608 25.7899 109.608C40.0338 109.608 51.581 98.0618 51.581 83.8186C51.581 69.5741 40.0338 58.0298 25.7899 58.0298V58.0298ZM85.0815 25.7894C85.0815 40.0338 73.5354 51.5816 59.2921 51.5816C45.0483 51.5816 33.5022 40.0338 33.5022 25.7894C33.5022 11.5478 45.0483 0 59.2921 0C73.5354 0 85.0815 11.5478 85.0815 25.7894V25.7894Z" fill="#FF584A"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 5.5 KiB | 
							
								
								
									
										25
									
								
								packages/backend/src/apps/asana/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								packages/backend/src/apps/asana/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | import { URLSearchParams } from 'url'; | ||||||
|  | import crypto from 'crypto'; | ||||||
|  |  | ||||||
|  | export default async function generateAuthUrl($) { | ||||||
|  |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|  |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|  |   ); | ||||||
|  |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|  |   const state = crypto.randomBytes(100).toString('base64url'); | ||||||
|  |  | ||||||
|  |   const searchParams = new URLSearchParams({ | ||||||
|  |     client_id: $.auth.data.clientId, | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|  |     response_type: 'code', | ||||||
|  |     //scope: authScope.join(' '), | ||||||
|  |     state, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const url = `https://app.asana.com/-/oauth_authorize?${searchParams.toString()}`; | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     url, | ||||||
|  |     originalState: state, | ||||||
|  |   }); | ||||||
|  | } | ||||||
							
								
								
									
										48
									
								
								packages/backend/src/apps/asana/auth/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								packages/backend/src/apps/asana/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/asana/connections/add', | ||||||
|  |       placeholder: null, | ||||||
|  |       description: | ||||||
|  |         'When asked to input a redirect URL in Asana, 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.data.email; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default isStillVerified; | ||||||
							
								
								
									
										37
									
								
								packages/backend/src/apps/asana/auth/refresh-token.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								packages/backend/src/apps/asana/auth/refresh-token.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | import { URLSearchParams } from 'node:url'; | ||||||
|  |  | ||||||
|  | const refreshToken = async ($) => { | ||||||
|  |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|  |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|  |   ); | ||||||
|  |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|  |  | ||||||
|  |   const params = new URLSearchParams({ | ||||||
|  |     client_id: $.auth.data.clientId, | ||||||
|  |     client_secret: $.auth.data.clientSecret, | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|  |     grant_type: 'refresh_token', | ||||||
|  |     refresh_token: $.auth.data.refreshToken, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     'https://app.asana.com/-/oauth_token', | ||||||
|  |     params.toString(), | ||||||
|  |     { | ||||||
|  |       headers: { | ||||||
|  |         'Content-Type': 'application/x-www-form-urlencoded', | ||||||
|  |       }, | ||||||
|  |       additionalProperties: { | ||||||
|  |         skipAddingAuthHeader: true, | ||||||
|  |       }, | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default refreshToken; | ||||||
							
								
								
									
										39
									
								
								packages/backend/src/apps/asana/auth/verify-credentials.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								packages/backend/src/apps/asana/auth/verify-credentials.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | const verifyCredentials = async ($) => { | ||||||
|  |   if ($.auth.data.originalState !== $.auth.data.state) { | ||||||
|  |     throw new Error("The 'state' parameter does not match."); | ||||||
|  |   } | ||||||
|  |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|  |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|  |   ); | ||||||
|  |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     'https://app.asana.com/-/oauth_token', | ||||||
|  |     { | ||||||
|  |       client_id: $.auth.data.clientId, | ||||||
|  |       client_secret: $.auth.data.clientSecret, | ||||||
|  |       code: $.auth.data.code, | ||||||
|  |       grant_type: 'authorization_code', | ||||||
|  |       redirect_uri: redirectUri, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       headers: { | ||||||
|  |         'Content-Type': 'application/x-www-form-urlencoded', | ||||||
|  |       }, | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |     clientId: $.auth.data.clientId, | ||||||
|  |     clientSecret: $.auth.data.clientSecret, | ||||||
|  |     scope: $.auth.data.scope, | ||||||
|  |     id: data.data.id, | ||||||
|  |     gid: data.data.gid, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     refreshToken: data.refresh_token, | ||||||
|  |     screenName: `${data.data.name} - ${data.data.email}`, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default verifyCredentials; | ||||||
							
								
								
									
										12
									
								
								packages/backend/src/apps/asana/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								packages/backend/src/apps/asana/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | const addAuthHeader = ($, requestConfig) => { | ||||||
|  |   if (requestConfig.additionalProperties?.skipAddingAuthHeader) | ||||||
|  |     return requestConfig; | ||||||
|  |  | ||||||
|  |   if ($.auth.data?.accessToken) { | ||||||
|  |     requestConfig.headers.Authorization = `${$.auth.data.tokenType} ${$.auth.data.accessToken}`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default addAuthHeader; | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | const getCurrentUser = async ($) => { | ||||||
|  |   const { data: currentUser } = await $.http.get( | ||||||
|  |     `/1.0/users/${$.auth.data.gid}` | ||||||
|  |   ); | ||||||
|  |   return currentUser; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default getCurrentUser; | ||||||
							
								
								
									
										15
									
								
								packages/backend/src/apps/asana/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								packages/backend/src/apps/asana/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | import listProjects from './list-projects/index.js'; | ||||||
|  | import listSections from './list-sections/index.js'; | ||||||
|  | import listTags from './list-tags/index.js'; | ||||||
|  | import listTeams from './list-teams/index.js'; | ||||||
|  | import listUsers from './list-users/index.js'; | ||||||
|  | import listWorkspaces from './list-workspaces/index.js'; | ||||||
|  |  | ||||||
|  | export default [ | ||||||
|  |   listProjects, | ||||||
|  |   listSections, | ||||||
|  |   listTags, | ||||||
|  |   listTeams, | ||||||
|  |   listUsers, | ||||||
|  |   listWorkspaces, | ||||||
|  | ]; | ||||||
| @@ -0,0 +1,42 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List projects', | ||||||
|  |   key: 'listProjects', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const projects = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const workspaceId = $.step.parameters.workspaceId; | ||||||
|  |  | ||||||
|  |     if (!workspaceId) { | ||||||
|  |       return projects; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       offset: undefined, | ||||||
|  |       workspace: workspaceId, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { | ||||||
|  |         data: { data, next_page }, | ||||||
|  |       } = await $.http.get('/1.0/projects', { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       params.offset = next_page?.offset; | ||||||
|  |  | ||||||
|  |       if (data) { | ||||||
|  |         for (const project of data) { | ||||||
|  |           projects.data.push({ | ||||||
|  |             value: project.gid, | ||||||
|  |             name: project.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return projects; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,41 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List sections', | ||||||
|  |   key: 'listSections', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const sections = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const projectId = $.step.parameters.projectId; | ||||||
|  |  | ||||||
|  |     if (!projectId) { | ||||||
|  |       return sections; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       offset: undefined, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { | ||||||
|  |         data: { data, next_page }, | ||||||
|  |       } = await $.http.get(`/1.0/projects/${projectId}/sections`, { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       params.offset = next_page?.offset; | ||||||
|  |  | ||||||
|  |       if (data) { | ||||||
|  |         for (const section of data) { | ||||||
|  |           sections.data.push({ | ||||||
|  |             value: section.gid, | ||||||
|  |             name: section.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return sections; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,42 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List tags', | ||||||
|  |   key: 'listTags', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const tags = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const workspaceId = $.step.parameters.workspaceId; | ||||||
|  |  | ||||||
|  |     if (!workspaceId) { | ||||||
|  |       return workspaceId; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       offset: undefined, | ||||||
|  |       workspace: workspaceId, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { | ||||||
|  |         data: { data, next_page }, | ||||||
|  |       } = await $.http.get('/1.0/tags', { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       params.offset = next_page?.offset; | ||||||
|  |  | ||||||
|  |       if (data) { | ||||||
|  |         for (const tag of data) { | ||||||
|  |           tags.data.push({ | ||||||
|  |             value: tag.gid, | ||||||
|  |             name: tag.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return tags; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,42 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List teams', | ||||||
|  |   key: 'listTeams', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const teams = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const workspaceId = $.step.parameters.workspaceId; | ||||||
|  |  | ||||||
|  |     if (!workspaceId) { | ||||||
|  |       return workspaceId; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       offset: undefined, | ||||||
|  |       workspace: workspaceId, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { | ||||||
|  |         data: { data, next_page }, | ||||||
|  |       } = await $.http.get('/1.0/teams', { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       params.offset = next_page?.offset; | ||||||
|  |  | ||||||
|  |       if (data) { | ||||||
|  |         for (const team of data) { | ||||||
|  |           teams.data.push({ | ||||||
|  |             value: team.gid, | ||||||
|  |             name: team.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return teams; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,42 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List users', | ||||||
|  |   key: 'listUsers', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const users = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |     const workspaceId = $.step.parameters.workspaceId; | ||||||
|  |  | ||||||
|  |     if (!workspaceId) { | ||||||
|  |       return workspaceId; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       offset: undefined, | ||||||
|  |       workspace: workspaceId, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { | ||||||
|  |         data: { data, next_page }, | ||||||
|  |       } = await $.http.get('/1.0/users', { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       params.offset = next_page?.offset; | ||||||
|  |  | ||||||
|  |       if (data) { | ||||||
|  |         for (const user of data) { | ||||||
|  |           users.data.push({ | ||||||
|  |             value: user.gid, | ||||||
|  |             name: user.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return users; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,34 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List workspaces', | ||||||
|  |   key: 'listWorkspaces', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const workspaces = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       offset: undefined, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { | ||||||
|  |         data: { data, next_page }, | ||||||
|  |       } = await $.http.get('/1.0/workspaces', { params }); | ||||||
|  |  | ||||||
|  |       params.offset = next_page?.offset; | ||||||
|  |  | ||||||
|  |       if (data) { | ||||||
|  |         for (const workspace of data) { | ||||||
|  |           workspaces.data.push({ | ||||||
|  |             value: workspace.gid, | ||||||
|  |             name: workspace.name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |  | ||||||
|  |     return workspaces; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										22
									
								
								packages/backend/src/apps/asana/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								packages/backend/src/apps/asana/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | import defineApp from '../../helpers/define-app.js'; | ||||||
|  | import addAuthHeader from './common/add-auth-header.js'; | ||||||
|  | import auth from './auth/index.js'; | ||||||
|  | import dynamicData from './dynamic-data/index.js'; | ||||||
|  | import triggers from './triggers/index.js'; | ||||||
|  | import actions from './actions/index.js'; | ||||||
|  |  | ||||||
|  | export default defineApp({ | ||||||
|  |   name: 'Asana', | ||||||
|  |   key: 'asana', | ||||||
|  |   baseUrl: 'https://asana.com', | ||||||
|  |   apiBaseUrl: 'https://app.asana.com/api', | ||||||
|  |   iconUrl: '{BASE_URL}/apps/asana/assets/favicon.svg', | ||||||
|  |   authDocUrl: '{DOCS_URL}/apps/asana/connection', | ||||||
|  |   primaryColor: '690031', | ||||||
|  |   supportsConnections: true, | ||||||
|  |   beforeRequest: [addAuthHeader], | ||||||
|  |   auth, | ||||||
|  |   dynamicData, | ||||||
|  |   triggers, | ||||||
|  |   actions, | ||||||
|  | }); | ||||||
							
								
								
									
										3
									
								
								packages/backend/src/apps/asana/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								packages/backend/src/apps/asana/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | import newProjects from './new-projects/index.js'; | ||||||
|  |  | ||||||
|  | export default [newProjects]; | ||||||
| @@ -0,0 +1,59 @@ | |||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New projects', | ||||||
|  |   key: 'newProjects', | ||||||
|  |   pollInterval: 15, | ||||||
|  |   description: 'Triggers when a new project is created.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Workspace', | ||||||
|  |       key: 'workspaceId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listWorkspaces', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const workspaceId = $.step.parameters.workspaceId; | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       offset: undefined, | ||||||
|  |       workspace: workspaceId, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       const { | ||||||
|  |         data: { data, next_page }, | ||||||
|  |       } = await $.http.get('/1.0/projects', { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       params.offset = next_page?.offset; | ||||||
|  |  | ||||||
|  |       if (data) { | ||||||
|  |         for (const project of data) { | ||||||
|  |           $.pushTriggerItem({ | ||||||
|  |             raw: project, | ||||||
|  |             meta: { | ||||||
|  |               internalId: project.gid, | ||||||
|  |             }, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (params.offset); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -11,7 +11,7 @@ export default defineApp({ | |||||||
|     'https://azure.microsoft.com/en-us/products/ai-services/openai-service', |     'https://azure.microsoft.com/en-us/products/ai-services/openai-service', | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   iconUrl: '{BASE_URL}/apps/azure-openai/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/azure-openai/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/azure-openai/connection', |   authDocUrl: '{DOCS_URL}/apps/azure-openai/connection', | ||||||
|   primaryColor: '000000', |   primaryColor: '000000', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   beforeRequest: [setBaseUrl, addAuthHeader], |   beforeRequest: [setBaseUrl, addAuthHeader], | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ export default defineApp({ | |||||||
|   name: 'Carbone', |   name: 'Carbone', | ||||||
|   key: 'carbone', |   key: 'carbone', | ||||||
|   iconUrl: '{BASE_URL}/apps/carbone/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/carbone/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/carbone/connection', |   authDocUrl: '{DOCS_URL}/apps/carbone/connection', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   baseUrl: 'https://carbone.io', |   baseUrl: 'https://carbone.io', | ||||||
|   apiBaseUrl: 'https://api.carbone.io', |   apiBaseUrl: 'https://api.carbone.io', | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ export default defineApp({ | |||||||
|   name: 'Datastore', |   name: 'Datastore', | ||||||
|   key: 'datastore', |   key: 'datastore', | ||||||
|   iconUrl: '{BASE_URL}/apps/datastore/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/datastore/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/datastore/connection', |   authDocUrl: '{DOCS_URL}/apps/datastore/connection', | ||||||
|   supportsConnections: false, |   supportsConnections: false, | ||||||
|   baseUrl: '', |   baseUrl: '', | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ export default defineApp({ | |||||||
|   name: 'DeepL', |   name: 'DeepL', | ||||||
|   key: 'deepl', |   key: 'deepl', | ||||||
|   iconUrl: '{BASE_URL}/apps/deepl/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/deepl/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/deepl/connection', |   authDocUrl: '{DOCS_URL}/apps/deepl/connection', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   baseUrl: 'https://deepl.com', |   baseUrl: 'https://deepl.com', | ||||||
|   apiBaseUrl: 'https://api.deepl.com', |   apiBaseUrl: 'https://api.deepl.com', | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ export default defineApp({ | |||||||
|   name: 'Delay', |   name: 'Delay', | ||||||
|   key: 'delay', |   key: 'delay', | ||||||
|   iconUrl: '{BASE_URL}/apps/delay/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/delay/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/delay/connection', |   authDocUrl: '{DOCS_URL}/apps/delay/connection', | ||||||
|   supportsConnections: false, |   supportsConnections: false, | ||||||
|   baseUrl: '', |   baseUrl: '', | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ export default defineApp({ | |||||||
|   name: 'Discord', |   name: 'Discord', | ||||||
|   key: 'discord', |   key: 'discord', | ||||||
|   iconUrl: '{BASE_URL}/apps/discord/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/discord/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/discord/connection', |   authDocUrl: '{DOCS_URL}/apps/discord/connection', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   baseUrl: 'https://discord.com', |   baseUrl: 'https://discord.com', | ||||||
|   apiBaseUrl: 'https://discord.com/api', |   apiBaseUrl: 'https://discord.com/api', | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								packages/backend/src/apps/disqus/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								packages/backend/src/apps/disqus/assets/favicon.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||||
|  | <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||||
|  | 	 width="200px" height="200px" viewBox="0 0 200 200" enable-background="new 0 0 200 200" xml:space="preserve"> | ||||||
|  | <g id="background"> | ||||||
|  | 	<rect fill="#2E9FFF" width="200" height="200"/> | ||||||
|  | </g> | ||||||
|  | <g id="Layer_2"> | ||||||
|  | </g> | ||||||
|  | <path fill="#FFFFFF" d="M102.535,167.5c-16.518,0-31.621-6.036-43.298-16.021L30.5,155.405l11.102-27.401 | ||||||
|  | 	c-3.868-8.535-6.038-18.01-6.038-28.004c0-37.277,29.984-67.5,66.971-67.5c36.984,0,66.965,30.223,66.965,67.5 | ||||||
|  | 	C169.5,137.284,139.52,167.5,102.535,167.5z M139.102,99.807v-0.188c0-19.479-13.736-33.367-37.42-33.367h-25.58v67.5h25.201 | ||||||
|  | 	C125.171,133.753,139.102,119.284,139.102,99.807L139.102,99.807z M101.964,117.168h-7.482V82.841h7.482 | ||||||
|  | 	c10.989,0,18.283,6.265,18.283,17.07v0.188C120.247,110.995,112.953,117.168,101.964,117.168z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										21
									
								
								packages/backend/src/apps/disqus/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								packages/backend/src/apps/disqus/auth/generate-auth-url.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | 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.apiKey, | ||||||
|  |     scope: authScope.join(','), | ||||||
|  |     response_type: 'code', | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const url = `https://disqus.com/api/oauth/2.0/authorize/?${searchParams.toString()}`; | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     url, | ||||||
|  |   }); | ||||||
|  | } | ||||||
							
								
								
									
										48
									
								
								packages/backend/src/apps/disqus/auth/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								packages/backend/src/apps/disqus/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/disqus/connections/add', | ||||||
|  |       placeholder: null, | ||||||
|  |       description: | ||||||
|  |         'When asked to input a redirect URL in Disqus, enter the URL above.', | ||||||
|  |       clickToCopy: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'apiKey', | ||||||
|  |       label: 'API Key', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       readOnly: false, | ||||||
|  |       value: null, | ||||||
|  |       placeholder: null, | ||||||
|  |       description: null, | ||||||
|  |       clickToCopy: false, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       key: 'apiSecret', | ||||||
|  |       label: 'API 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.response.username; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default isStillVerified; | ||||||
							
								
								
									
										26
									
								
								packages/backend/src/apps/disqus/auth/refresh-token.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								packages/backend/src/apps/disqus/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({ | ||||||
|  |     grant_type: 'refresh_token', | ||||||
|  |     client_id: $.auth.data.apiKey, | ||||||
|  |     client_secret: $.auth.data.apiSecret, | ||||||
|  |     refresh_token: $.auth.data.refreshToken, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     `https://disqus.com/api/oauth/2.0/access_token/`, | ||||||
|  |     params.toString() | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     refreshToken: data.refresh_token, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     scope: authScope.join(','), | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default refreshToken; | ||||||
							
								
								
									
										34
									
								
								packages/backend/src/apps/disqus/auth/verify-credentials.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								packages/backend/src/apps/disqus/auth/verify-credentials.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | import { URLSearchParams } from 'url'; | ||||||
|  |  | ||||||
|  | const verifyCredentials = async ($) => { | ||||||
|  |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|  |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|  |   ); | ||||||
|  |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|  |   const params = new URLSearchParams({ | ||||||
|  |     grant_type: 'authorization_code', | ||||||
|  |     client_id: $.auth.data.apiKey, | ||||||
|  |     client_secret: $.auth.data.apiSecret, | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|  |     code: $.auth.data.code, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     `https://disqus.com/api/oauth/2.0/access_token/`, | ||||||
|  |     params.toString() | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   await $.auth.set({ | ||||||
|  |     accessToken: data.access_token, | ||||||
|  |     tokenType: data.token_type, | ||||||
|  |     apiKey: $.auth.data.apiKey, | ||||||
|  |     apiSecret: $.auth.data.apiSecret, | ||||||
|  |     scope: $.auth.data.scope, | ||||||
|  |     userId: data.user_id, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|  |     refreshToken: data.refresh_token, | ||||||
|  |     screenName: data.username, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default verifyCredentials; | ||||||
							
								
								
									
										15
									
								
								packages/backend/src/apps/disqus/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								packages/backend/src/apps/disqus/common/add-auth-header.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | import { URLSearchParams } from 'url'; | ||||||
|  |  | ||||||
|  | const addAuthHeader = ($, requestConfig) => { | ||||||
|  |   const params = new URLSearchParams({ | ||||||
|  |     access_token: $.auth.data.accessToken, | ||||||
|  |     api_key: $.auth.data.apiKey, | ||||||
|  |     api_secret: $.auth.data.apiSecret, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   requestConfig.params = params; | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default addAuthHeader; | ||||||
							
								
								
									
										3
									
								
								packages/backend/src/apps/disqus/common/auth-scope.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								packages/backend/src/apps/disqus/common/auth-scope.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | const authScope = ['read', 'write', 'admin', 'email']; | ||||||
|  |  | ||||||
|  | export default authScope; | ||||||
							
								
								
									
										10
									
								
								packages/backend/src/apps/disqus/common/get-current-user.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								packages/backend/src/apps/disqus/common/get-current-user.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | const getCurrentUser = async ($) => { | ||||||
|  |   try { | ||||||
|  |     const { data: currentUser } = await $.http.get('/3.0/users/details.json'); | ||||||
|  |     return currentUser; | ||||||
|  |   } catch (error) { | ||||||
|  |     throw new Error('You are not authenticated.'); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default getCurrentUser; | ||||||
							
								
								
									
										3
									
								
								packages/backend/src/apps/disqus/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								packages/backend/src/apps/disqus/dynamic-data/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | import listForums from './list-forums/index.js'; | ||||||
|  |  | ||||||
|  | export default [listForums]; | ||||||
| @@ -0,0 +1,36 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List forums', | ||||||
|  |   key: 'listForums', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const forums = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       limit: 100, | ||||||
|  |       order: 'desc', | ||||||
|  |       cursor: undefined, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     let more; | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get('/3.0/users/listForums.json', { | ||||||
|  |         params, | ||||||
|  |       }); | ||||||
|  |       params.cursor = data.cursor.next; | ||||||
|  |       more = data.cursor.hasNext; | ||||||
|  |  | ||||||
|  |       if (data.response?.length) { | ||||||
|  |         for (const forum of data.response) { | ||||||
|  |           forums.data.push({ | ||||||
|  |             value: forum.id, | ||||||
|  |             name: forum.id, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (more); | ||||||
|  |  | ||||||
|  |     return forums; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										20
									
								
								packages/backend/src/apps/disqus/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								packages/backend/src/apps/disqus/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 dynamicData from './dynamic-data/index.js'; | ||||||
|  | import triggers from './triggers/index.js'; | ||||||
|  |  | ||||||
|  | export default defineApp({ | ||||||
|  |   name: 'Disqus', | ||||||
|  |   key: 'disqus', | ||||||
|  |   baseUrl: 'https://disqus.com', | ||||||
|  |   apiBaseUrl: 'https://disqus.com/api', | ||||||
|  |   iconUrl: '{BASE_URL}/apps/disqus/assets/favicon.svg', | ||||||
|  |   authDocUrl: '{DOCS_URL}/apps/disqus/connection', | ||||||
|  |   primaryColor: '2E9FFF', | ||||||
|  |   supportsConnections: true, | ||||||
|  |   beforeRequest: [addAuthHeader], | ||||||
|  |   auth, | ||||||
|  |   dynamicData, | ||||||
|  |   triggers, | ||||||
|  | }); | ||||||
							
								
								
									
										4
									
								
								packages/backend/src/apps/disqus/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								packages/backend/src/apps/disqus/triggers/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | import newComments from './new-comments/index.js'; | ||||||
|  | import newFlaggedComments from './new-flagged-comments/index.js'; | ||||||
|  |  | ||||||
|  | export default [newComments, newFlaggedComments]; | ||||||
| @@ -0,0 +1,92 @@ | |||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  | import { URLSearchParams } from 'url'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New comments', | ||||||
|  |   key: 'newComments', | ||||||
|  |   pollInterval: 15, | ||||||
|  |   description: 'Triggers when a new comment is posted in a forum using Disqus.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Post Types', | ||||||
|  |       key: 'postTypes', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Which posts should be considered for inclusion in the trigger?', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Type', | ||||||
|  |           key: 'type', | ||||||
|  |           type: 'dropdown', | ||||||
|  |           required: false, | ||||||
|  |           description: '', | ||||||
|  |           variables: true, | ||||||
|  |           options: [ | ||||||
|  |             { label: 'Unapproved Posts', value: 'unapproved' }, | ||||||
|  |             { label: 'Approved Posts', value: 'approved' }, | ||||||
|  |             { label: 'Spam Posts', value: 'spam' }, | ||||||
|  |             { label: 'Deleted Posts', value: 'deleted' }, | ||||||
|  |             { label: 'Flagged Posts', value: 'flagged' }, | ||||||
|  |             { label: 'Highlighted Posts', value: 'highlighted' }, | ||||||
|  |           ], | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Forum', | ||||||
|  |       key: 'forumId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Select the forum where you want comments to be triggered.', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listForums', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const forumId = $.step.parameters.forumId; | ||||||
|  |     const postTypes = $.step.parameters.postTypes; | ||||||
|  |     const formattedCommentTypes = postTypes | ||||||
|  |       .filter((type) => type.type !== '') | ||||||
|  |       .map((type) => type.type); | ||||||
|  |  | ||||||
|  |     const params = new URLSearchParams({ | ||||||
|  |       limit: '100', | ||||||
|  |       forum: forumId, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     if (formattedCommentTypes.length) { | ||||||
|  |       formattedCommentTypes.forEach((type) => params.append('include', type)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let more; | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get( | ||||||
|  |         `/3.0/posts/list.json?${params.toString()}` | ||||||
|  |       ); | ||||||
|  |       params.set('cursor', data.cursor.next); | ||||||
|  |       more = data.cursor.hasNext; | ||||||
|  |  | ||||||
|  |       if (data.response?.length) { | ||||||
|  |         for (const comment of data.response) { | ||||||
|  |           $.pushTriggerItem({ | ||||||
|  |             raw: comment, | ||||||
|  |             meta: { | ||||||
|  |               internalId: comment.id, | ||||||
|  |             }, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (more); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,60 @@ | |||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  | import { URLSearchParams } from 'url'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New flagged comments', | ||||||
|  |   key: 'newFlaggedComments', | ||||||
|  |   pollInterval: 15, | ||||||
|  |   description: 'Triggers when a Disqus comment is marked with a flag', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Forum', | ||||||
|  |       key: 'forumId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: 'Select the forum where you want comments to be triggered.', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listForums', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const forumId = $.step.parameters.forumId; | ||||||
|  |     const isFlaggedFilter = 5; | ||||||
|  |  | ||||||
|  |     const params = new URLSearchParams({ | ||||||
|  |       limit: 100, | ||||||
|  |       forum: forumId, | ||||||
|  |       filters: [isFlaggedFilter], | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     let more; | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get( | ||||||
|  |         `/3.0/posts/list.json?${params.toString()}` | ||||||
|  |       ); | ||||||
|  |       params.set('cursor', data.cursor.next); | ||||||
|  |       more = data.cursor.hasNext; | ||||||
|  |  | ||||||
|  |       if (data.response?.length) { | ||||||
|  |         for (const comment of data.response) { | ||||||
|  |           $.pushTriggerItem({ | ||||||
|  |             raw: comment, | ||||||
|  |             meta: { | ||||||
|  |               internalId: comment.id, | ||||||
|  |             }, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (more); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -7,7 +7,7 @@ export default defineApp({ | |||||||
|   name: 'Dropbox', |   name: 'Dropbox', | ||||||
|   key: 'dropbox', |   key: 'dropbox', | ||||||
|   iconUrl: '{BASE_URL}/apps/dropbox/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/dropbox/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/dropbox/connection', |   authDocUrl: '{DOCS_URL}/apps/dropbox/connection', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   baseUrl: 'https://dropbox.com', |   baseUrl: 'https://dropbox.com', | ||||||
|   apiBaseUrl: 'https://api.dropboxapi.com', |   apiBaseUrl: 'https://api.dropboxapi.com', | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ export default defineApp({ | |||||||
|   name: 'Filter', |   name: 'Filter', | ||||||
|   key: 'filter', |   key: 'filter', | ||||||
|   iconUrl: '{BASE_URL}/apps/filter/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/filter/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/filter/connection', |   authDocUrl: '{DOCS_URL}/apps/filter/connection', | ||||||
|   supportsConnections: false, |   supportsConnections: false, | ||||||
|   baseUrl: '', |   baseUrl: '', | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ export default defineApp({ | |||||||
|   name: 'Flickr', |   name: 'Flickr', | ||||||
|   key: 'flickr', |   key: 'flickr', | ||||||
|   iconUrl: '{BASE_URL}/apps/flickr/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/flickr/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/flickr/connection', |   authDocUrl: '{DOCS_URL}/apps/flickr/connection', | ||||||
|   docUrl: 'https://automatisch.io/docs/flickr', |   docUrl: 'https://automatisch.io/docs/flickr', | ||||||
|   primaryColor: '000000', |   primaryColor: '000000', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ export default defineApp({ | |||||||
|   name: 'Flowers Software', |   name: 'Flowers Software', | ||||||
|   key: 'flowers-software', |   key: 'flowers-software', | ||||||
|   iconUrl: '{BASE_URL}/apps/flowers-software/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/flowers-software/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/flowers-software/connection', |   authDocUrl: '{DOCS_URL}/apps/flowers-software/connection', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   baseUrl: 'https://flowers-software.com', |   baseUrl: 'https://flowers-software.com', | ||||||
|   apiBaseUrl: 'https://webapp.flowers-software.com/api', |   apiBaseUrl: 'https://webapp.flowers-software.com/api', | ||||||
|   | |||||||
| @@ -5,11 +5,24 @@ const formatDateTime = ($) => { | |||||||
|  |  | ||||||
|   const fromFormat = $.step.parameters.fromFormat; |   const fromFormat = $.step.parameters.fromFormat; | ||||||
|   const fromTimezone = $.step.parameters.fromTimezone; |   const fromTimezone = $.step.parameters.fromTimezone; | ||||||
|  |   let inputDateTime; | ||||||
|  |  | ||||||
|   const inputDateTime = DateTime.fromFormat(input, fromFormat, { |   if (fromFormat === 'X') { | ||||||
|  |     inputDateTime = DateTime.fromSeconds(Number(input), fromFormat, { | ||||||
|       zone: fromTimezone, |       zone: fromTimezone, | ||||||
|       setZone: true, |       setZone: true, | ||||||
|     }); |     }); | ||||||
|  |   } else if (fromFormat === 'x') { | ||||||
|  |     inputDateTime = DateTime.fromMillis(Number(input), fromFormat, { | ||||||
|  |       zone: fromTimezone, | ||||||
|  |       setZone: true, | ||||||
|  |     }); | ||||||
|  |   } else { | ||||||
|  |     inputDateTime = DateTime.fromFormat(input, fromFormat, { | ||||||
|  |       zone: fromTimezone, | ||||||
|  |       setZone: true, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const toFormat = $.step.parameters.toFormat; |   const toFormat = $.step.parameters.toFormat; | ||||||
|   const toTimezone = $.step.parameters.toTimezone; |   const toTimezone = $.step.parameters.toTimezone; | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import defineAction from '../../../../helpers/define-action.js'; | |||||||
|  |  | ||||||
| import base64ToString from './transformers/base64-to-string.js'; | import base64ToString from './transformers/base64-to-string.js'; | ||||||
| import capitalize from './transformers/capitalize.js'; | import capitalize from './transformers/capitalize.js'; | ||||||
|  | import encodeUriComponent from './transformers/encode-uri-component.js'; | ||||||
| import extractEmailAddress from './transformers/extract-email-address.js'; | import extractEmailAddress from './transformers/extract-email-address.js'; | ||||||
| import extractNumber from './transformers/extract-number.js'; | import extractNumber from './transformers/extract-number.js'; | ||||||
| import htmlToMarkdown from './transformers/html-to-markdown.js'; | import htmlToMarkdown from './transformers/html-to-markdown.js'; | ||||||
| @@ -10,12 +11,14 @@ import markdownToHtml from './transformers/markdown-to-html.js'; | |||||||
| import pluralize from './transformers/pluralize.js'; | import pluralize from './transformers/pluralize.js'; | ||||||
| import replace from './transformers/replace.js'; | import replace from './transformers/replace.js'; | ||||||
| import stringToBase64 from './transformers/string-to-base64.js'; | import stringToBase64 from './transformers/string-to-base64.js'; | ||||||
|  | import encodeUri from './transformers/encode-uri.js'; | ||||||
| import trimWhitespace from './transformers/trim-whitespace.js'; | import trimWhitespace from './transformers/trim-whitespace.js'; | ||||||
| import useDefaultValue from './transformers/use-default-value.js'; | import useDefaultValue from './transformers/use-default-value.js'; | ||||||
|  |  | ||||||
| const transformers = { | const transformers = { | ||||||
|   base64ToString, |   base64ToString, | ||||||
|   capitalize, |   capitalize, | ||||||
|  |   encodeUriComponent, | ||||||
|   extractEmailAddress, |   extractEmailAddress, | ||||||
|   extractNumber, |   extractNumber, | ||||||
|   htmlToMarkdown, |   htmlToMarkdown, | ||||||
| @@ -24,6 +27,7 @@ const transformers = { | |||||||
|   pluralize, |   pluralize, | ||||||
|   replace, |   replace, | ||||||
|   stringToBase64, |   stringToBase64, | ||||||
|  |   encodeUri, | ||||||
|   trimWhitespace, |   trimWhitespace, | ||||||
|   useDefaultValue, |   useDefaultValue, | ||||||
| }; | }; | ||||||
| @@ -43,6 +47,10 @@ export default defineAction({ | |||||||
|       options: [ |       options: [ | ||||||
|         { label: 'Base64 to String', value: 'base64ToString' }, |         { label: 'Base64 to String', value: 'base64ToString' }, | ||||||
|         { label: 'Capitalize', value: 'capitalize' }, |         { label: 'Capitalize', value: 'capitalize' }, | ||||||
|  |         { | ||||||
|  |           label: 'Encode URI Component', | ||||||
|  |           value: 'encodeUriComponent', | ||||||
|  |         }, | ||||||
|         { label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' }, |         { label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' }, | ||||||
|         { label: 'Convert Markdown to HTML', value: 'markdownToHtml' }, |         { label: 'Convert Markdown to HTML', value: 'markdownToHtml' }, | ||||||
|         { label: 'Extract Email Address', value: 'extractEmailAddress' }, |         { label: 'Extract Email Address', value: 'extractEmailAddress' }, | ||||||
| @@ -51,6 +59,7 @@ export default defineAction({ | |||||||
|         { label: 'Pluralize', value: 'pluralize' }, |         { label: 'Pluralize', value: 'pluralize' }, | ||||||
|         { label: 'Replace', value: 'replace' }, |         { label: 'Replace', value: 'replace' }, | ||||||
|         { label: 'String to Base64', value: 'stringToBase64' }, |         { label: 'String to Base64', value: 'stringToBase64' }, | ||||||
|  |         { label: 'Encode URI', value: 'encodeUri' }, | ||||||
|         { label: 'Trim Whitespace', value: 'trimWhitespace' }, |         { label: 'Trim Whitespace', value: 'trimWhitespace' }, | ||||||
|         { label: 'Use Default Value', value: 'useDefaultValue' }, |         { label: 'Use Default Value', value: 'useDefaultValue' }, | ||||||
|       ], |       ], | ||||||
|   | |||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | const encodeUriComponent = ($) => { | ||||||
|  |   const input = $.step.parameters.input; | ||||||
|  |   const encodedString = encodeURIComponent(input); | ||||||
|  |  | ||||||
|  |   return encodedString; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default encodeUriComponent; | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | const encodeUri = ($) => { | ||||||
|  |   const input = $.step.parameters.input; | ||||||
|  |   const encodedString = encodeURI(input); | ||||||
|  |  | ||||||
|  |   return encodedString; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default encodeUri; | ||||||
| @@ -1,5 +1,6 @@ | |||||||
| import base64ToString from './text/base64-to-string.js'; | import base64ToString from './text/base64-to-string.js'; | ||||||
| import capitalize from './text/capitalize.js'; | import capitalize from './text/capitalize.js'; | ||||||
|  | import encodeUriComponent from './text/encode-uri-component.js'; | ||||||
| import extractEmailAddress from './text/extract-email-address.js'; | import extractEmailAddress from './text/extract-email-address.js'; | ||||||
| import extractNumber from './text/extract-number.js'; | import extractNumber from './text/extract-number.js'; | ||||||
| import htmlToMarkdown from './text/html-to-markdown.js'; | import htmlToMarkdown from './text/html-to-markdown.js'; | ||||||
| @@ -8,6 +9,7 @@ import markdownToHtml from './text/markdown-to-html.js'; | |||||||
| import pluralize from './text/pluralize.js'; | import pluralize from './text/pluralize.js'; | ||||||
| import replace from './text/replace.js'; | import replace from './text/replace.js'; | ||||||
| import stringToBase64 from './text/string-to-base64.js'; | import stringToBase64 from './text/string-to-base64.js'; | ||||||
|  | import encodeUri from './text/encode-uri.js'; | ||||||
| import trimWhitespace from './text/trim-whitespace.js'; | import trimWhitespace from './text/trim-whitespace.js'; | ||||||
| import useDefaultValue from './text/use-default-value.js'; | import useDefaultValue from './text/use-default-value.js'; | ||||||
| import performMathOperation from './numbers/perform-math-operation.js'; | import performMathOperation from './numbers/perform-math-operation.js'; | ||||||
| @@ -19,6 +21,7 @@ import formatDateTime from './date-time/format-date-time.js'; | |||||||
| const options = { | const options = { | ||||||
|   base64ToString, |   base64ToString, | ||||||
|   capitalize, |   capitalize, | ||||||
|  |   encodeUriComponent, | ||||||
|   extractEmailAddress, |   extractEmailAddress, | ||||||
|   extractNumber, |   extractNumber, | ||||||
|   htmlToMarkdown, |   htmlToMarkdown, | ||||||
| @@ -27,6 +30,7 @@ const options = { | |||||||
|   pluralize, |   pluralize, | ||||||
|   replace, |   replace, | ||||||
|   stringToBase64, |   stringToBase64, | ||||||
|  |   encodeUri, | ||||||
|   trimWhitespace, |   trimWhitespace, | ||||||
|   useDefaultValue, |   useDefaultValue, | ||||||
|   performMathOperation, |   performMathOperation, | ||||||
|   | |||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | const encodeUriComponent = [ | ||||||
|  |   { | ||||||
|  |     label: 'Input', | ||||||
|  |     key: 'input', | ||||||
|  |     type: 'string', | ||||||
|  |     required: true, | ||||||
|  |     description: 'URI Component to encode', | ||||||
|  |     variables: true, | ||||||
|  |   }, | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | export default encodeUriComponent; | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | const encodeUri = [ | ||||||
|  |   { | ||||||
|  |     label: 'Input', | ||||||
|  |     key: 'input', | ||||||
|  |     type: 'string', | ||||||
|  |     required: true, | ||||||
|  |     description: 'URI to encode', | ||||||
|  |     variables: true, | ||||||
|  |   }, | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | export default encodeUri; | ||||||
| @@ -6,7 +6,7 @@ export default defineApp({ | |||||||
|   name: 'Formatter', |   name: 'Formatter', | ||||||
|   key: 'formatter', |   key: 'formatter', | ||||||
|   iconUrl: '{BASE_URL}/apps/formatter/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/formatter/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/formatter/connection', |   authDocUrl: '{DOCS_URL}/apps/formatter/connection', | ||||||
|   supportsConnections: false, |   supportsConnections: false, | ||||||
|   baseUrl: '', |   baseUrl: '', | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ export default defineApp({ | |||||||
|   baseUrl: 'https://ghost.org', |   baseUrl: 'https://ghost.org', | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   iconUrl: '{BASE_URL}/apps/ghost/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/ghost/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/ghost/connection', |   authDocUrl: '{DOCS_URL}/apps/ghost/connection', | ||||||
|   primaryColor: '15171A', |   primaryColor: '15171A', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   beforeRequest: [setBaseUrl, addAuthHeader], |   beforeRequest: [setBaseUrl, addAuthHeader], | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ export default defineApp({ | |||||||
|   baseUrl: 'https://github.com', |   baseUrl: 'https://github.com', | ||||||
|   apiBaseUrl: 'https://api.github.com', |   apiBaseUrl: 'https://api.github.com', | ||||||
|   iconUrl: '{BASE_URL}/apps/github/assets/favicon.svg', |   iconUrl: '{BASE_URL}/apps/github/assets/favicon.svg', | ||||||
|   authDocUrl: 'https://automatisch.io/docs/apps/github/connection', |   authDocUrl: '{DOCS_URL}/apps/github/connection', | ||||||
|   primaryColor: '000000', |   primaryColor: '000000', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   beforeRequest: [addAuthHeader], |   beforeRequest: [addAuthHeader], | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user