Compare commits
	
		
			9 Commits
		
	
	
		
			AUT-157-AU
			...
			AUT-740
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | daa38ab846 | ||
|   | 00e18cf42d | ||
|   | b3d2a1167c | ||
|   | 43c34fcb7b | ||
|   | c98f24338f | ||
|   | d0748bb7e2 | ||
|   | 81e8e4963c | ||
|   | 282b2374a2 | ||
|   | b7e38f9d1c | 
| @@ -28,7 +28,7 @@ cd packages/web | |||||||
| rm -rf .env | rm -rf .env | ||||||
| echo " | echo " | ||||||
| PORT=$WEB_PORT | PORT=$WEB_PORT | ||||||
| REACT_APP_BACKEND_URL=http://localhost:$BACKEND_PORT | REACT_APP_GRAPHQL_URL=http://localhost:$BACKEND_PORT/graphql | ||||||
| " >> .env | " >> .env | ||||||
| cd $CURRENT_DIR | cd $CURRENT_DIR | ||||||
|  |  | ||||||
|   | |||||||
| @@ -36,6 +36,7 @@ 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,9 +4,5 @@ | |||||||
| **/.devcontainer | **/.devcontainer | ||||||
| **/.github | **/.github | ||||||
| **/.vscode | **/.vscode | ||||||
| **/.env |  | ||||||
| **/.env.test |  | ||||||
| **/.env.production |  | ||||||
| **/yarn-error.log |  | ||||||
| packages/docs | packages/docs | ||||||
| packages/e2e-test | packages/e2e-test | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | module.exports = { | ||||||
|  |   root: true, | ||||||
|  |   parser: '@typescript-eslint/parser', | ||||||
|  |   plugins: ['@typescript-eslint'], | ||||||
|  |   extends: [ | ||||||
|  |     'eslint:recommended', | ||||||
|  |     'plugin:@typescript-eslint/recommended', | ||||||
|  |     'prettier', | ||||||
|  |   ], | ||||||
|  |   overrides: [ | ||||||
|  |     { | ||||||
|  |       files: ['**/*.test.ts', '**/test/**/*.ts'], | ||||||
|  |       rules: { | ||||||
|  |         '@typescript-eslint/ban-ts-comment': ['off'], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  | }; | ||||||
							
								
								
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -22,7 +22,7 @@ jobs: | |||||||
|       - run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner." |       - run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner." | ||||||
|       - run: echo "🖥️ The workflow is now ready to test your code on the runner." |       - run: echo "🖥️ The workflow is now ready to test your code on the runner." | ||||||
|       - run: yarn --frozen-lockfile |       - run: yarn --frozen-lockfile | ||||||
|       - run: cd packages/backend && yarn lint |       - run: yarn lint | ||||||
|       - run: echo "🍏 This job's status is ${{ job.status }}." |       - run: echo "🍏 This job's status is ${{ job.status }}." | ||||||
|   start-backend-server: |   start-backend-server: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|   | |||||||
							
								
								
									
										32
									
								
								.github/workflows/docs-change.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										32
									
								
								.github/workflows/docs-change.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,32 +0,0 @@ | |||||||
| 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}`); |  | ||||||
|             } |  | ||||||
							
								
								
									
										5
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
								
							| @@ -62,9 +62,8 @@ jobs: | |||||||
|         run: yarn && yarn lerna bootstrap |         run: yarn && yarn lerna bootstrap | ||||||
|       - name: Install Playwright Browsers |       - name: Install Playwright Browsers | ||||||
|         run: yarn playwright install --with-deps |         run: yarn playwright install --with-deps | ||||||
|       - name: Build Automatisch web |       - name: Build Automatisch | ||||||
|         working-directory: ./packages/web |         run: yarn lerna run --scope=@*/{web,cli} build | ||||||
|         run: yarn build |  | ||||||
|         env: |         env: | ||||||
|           # Keep this until we clean up warnings in build processes |           # Keep this until we clean up warnings in build processes | ||||||
|           CI: false |           CI: false | ||||||
|   | |||||||
| @@ -1,25 +1,14 @@ | |||||||
| # syntax=docker/dockerfile:1 | # syntax=docker/dockerfile:1 | ||||||
| FROM node:18-alpine | FROM node:18-alpine | ||||||
|  |  | ||||||
| ENV PORT 3000 |  | ||||||
|  |  | ||||||
| RUN \ |  | ||||||
|   apk --no-cache add --virtual build-dependencies python3 build-base git |  | ||||||
|  |  | ||||||
| WORKDIR /automatisch | 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 ./docker/entrypoint.sh /entrypoint.sh | COPY ./entrypoint.sh /entrypoint.sh | ||||||
|  |  | ||||||
| EXPOSE 3000 | EXPOSE 3000 | ||||||
| ENTRYPOINT ["sh", "/entrypoint.sh"] | ENTRYPOINT ["sh", "/entrypoint.sh"] | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								docker/Dockerfile.cloud
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								docker/Dockerfile.cloud
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | # 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:latest | FROM automatischio/automatisch:0.10.0 | ||||||
| WORKDIR /automatisch | WORKDIR /automatisch | ||||||
|  |  | ||||||
| RUN apk add --no-cache openssl dos2unix | RUN apk add --no-cache openssl dos2unix | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								docker/entrypoint-cloud.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										13
									
								
								docker/entrypoint-cloud.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | #!/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,12 +2,8 @@ | |||||||
|  |  | ||||||
| set -e | set -e | ||||||
|  |  | ||||||
| cd packages/backend |  | ||||||
|  |  | ||||||
| if [ -n "$WORKER" ]; then | if [ -n "$WORKER" ]; then | ||||||
|   yarn start:worker |   automatisch start-worker | ||||||
| else | else | ||||||
|   yarn db:migrate |   automatisch start | ||||||
|   yarn db:seed:user |  | ||||||
|   yarn start |  | ||||||
| fi | fi | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|     "start": "lerna run --stream --parallel --scope=@*/{web,backend} dev", |     "start": "lerna run --stream --parallel --scope=@*/{web,backend} dev", | ||||||
|     "start:web": "lerna run --stream --scope=@*/web dev", |     "start:web": "lerna run --stream --scope=@*/web dev", | ||||||
|     "start:backend": "lerna run --stream --scope=@*/backend dev", |     "start:backend": "lerna run --stream --scope=@*/backend dev", | ||||||
|  |     "lint": "lerna run --no-bail --stream --parallel --scope=@*/{web,backend} lint", | ||||||
|     "build:docs": "cd ./packages/docs && yarn install && yarn build" |     "build:docs": "cd ./packages/docs && yarn install && yarn build" | ||||||
|   }, |   }, | ||||||
|   "workspaces": { |   "workspaces": { | ||||||
| @@ -20,6 +21,8 @@ | |||||||
|     ] |     ] | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|  |     "@typescript-eslint/eslint-plugin": "^5.9.1", | ||||||
|  |     "@typescript-eslint/parser": "^5.9.1", | ||||||
|     "eslint": "^8.13.0", |     "eslint": "^8.13.0", | ||||||
|     "eslint-config-prettier": "^8.3.0", |     "eslint-config-prettier": "^8.3.0", | ||||||
|     "eslint-plugin-prettier": "^4.0.0", |     "eslint-plugin-prettier": "^4.0.0", | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ 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'; | ||||||
| @@ -22,14 +21,6 @@ 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(); | ||||||
| @@ -46,8 +37,6 @@ 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.'); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ | |||||||
|     "start:worker": "node src/worker.js", |     "start:worker": "node src/worker.js", | ||||||
|     "pretest": "APP_ENV=test node ./test/setup/prepare-test-env.js", |     "pretest": "APP_ENV=test node ./test/setup/prepare-test-env.js", | ||||||
|     "test": "APP_ENV=test vitest run", |     "test": "APP_ENV=test vitest run", | ||||||
|     "lint": "eslint .", |     "lint": "eslint . --ignore-path ../../.eslintignore", | ||||||
|     "db:create": "node ./bin/database/create.js", |     "db:create": "node ./bin/database/create.js", | ||||||
|     "db:seed:user": "node ./bin/database/seed-user.js", |     "db:seed:user": "node ./bin/database/seed-user.js", | ||||||
|     "db:drop": "node ./bin/database/drop.js", |     "db:drop": "node ./bin/database/drop.js", | ||||||
| @@ -31,14 +31,13 @@ | |||||||
|     "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.1.0", |     "bcrypt": "^5.0.1", | ||||||
|     "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", | ||||||
|     "debug": "~2.6.9", |     "debug": "~2.6.9", | ||||||
|     "dotenv": "^10.0.0", |     "dotenv": "^10.0.0", | ||||||
|     "express": "~4.18.2", |     "express": "~4.18.2", | ||||||
|     "express-async-handler": "^1.2.0", |  | ||||||
|     "express-basic-auth": "^1.2.1", |     "express-basic-auth": "^1.2.1", | ||||||
|     "express-graphql": "^0.12.0", |     "express-graphql": "^0.12.0", | ||||||
|     "fast-xml-parser": "^4.0.11", |     "fast-xml-parser": "^4.0.11", | ||||||
| @@ -67,7 +66,6 @@ | |||||||
|     "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" | ||||||
|   }, |   }, | ||||||
| @@ -96,7 +94,6 @@ | |||||||
|     "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" | ||||||
|   | |||||||
| @@ -1,92 +0,0 @@ | |||||||
| 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, |  | ||||||
|     }); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -1,174 +0,0 @@ | |||||||
| 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, |  | ||||||
|     }); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| import createRecord from './create-record/index.js'; |  | ||||||
| import findRecord from './find-record/index.js'; |  | ||||||
|  |  | ||||||
| export default [createRecord, findRecord]; |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| <?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> |  | ||||||
| Before Width: | Height: | Size: 1.9 KiB | 
| @@ -1,38 +0,0 @@ | |||||||
| 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, |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
| @@ -1,48 +0,0 @@ | |||||||
| 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, |  | ||||||
| }; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import getCurrentUser from '../common/get-current-user.js'; |  | ||||||
|  |  | ||||||
| const isStillVerified = async ($) => { |  | ||||||
|   const currentUser = await getCurrentUser($); |  | ||||||
|   return !!currentUser.id; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default isStillVerified; |  | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,56 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| const addAuthHeader = ($, requestConfig) => { |  | ||||||
|   if ( |  | ||||||
|     !requestConfig.additionalProperties?.skipAddingAuthHeader && |  | ||||||
|     $.auth.data?.accessToken |  | ||||||
|   ) { |  | ||||||
|     requestConfig.headers.Authorization = `${$.auth.data.tokenType} ${$.auth.data.accessToken}`; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return requestConfig; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default addAuthHeader; |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| const getCurrentUser = async ($) => { |  | ||||||
|   const { data: currentUser } = await $.http.get('/v0/meta/whoami'); |  | ||||||
|   return currentUser; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default getCurrentUser; |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| 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]; |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,35 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| import listFields from './list-fields/index.js'; |  | ||||||
|  |  | ||||||
| export default [listFields]; |  | ||||||
| @@ -1,86 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| 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 +0,0 @@ | |||||||
| <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> |  | ||||||
| Before Width: | Height: | Size: 2.6 KiB | 
| @@ -1,65 +0,0 @@ | |||||||
| 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, |  | ||||||
| }; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import verifyCredentials from './verify-credentials.js'; |  | ||||||
|  |  | ||||||
| const isStillVerified = async ($) => { |  | ||||||
|   await verifyCredentials($); |  | ||||||
|   return true; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default isStillVerified; |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| const verifyCredentials = async ($) => { |  | ||||||
|   await $.http.get('/v1/users'); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default verifyCredentials; |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| import listCollections from './list-collections/index.js'; |  | ||||||
| import listDatabases from './list-databases/index.js'; |  | ||||||
|  |  | ||||||
| export default [listCollections, listDatabases]; |  | ||||||
| @@ -1,44 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,36 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| 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, |  | ||||||
| }); |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| import newDocuments from './new-documents/index.js'; |  | ||||||
|  |  | ||||||
| export default [newDocuments]; |  | ||||||
| @@ -1,104 +0,0 @@ | |||||||
| 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); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -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: '{DOCS_URL}/apps/azure-openai/connection', |   authDocUrl: 'https://automatisch.io/docs/apps/azure-openai/connection', | ||||||
|   primaryColor: '000000', |   primaryColor: '000000', | ||||||
|   supportsConnections: true, |   supportsConnections: true, | ||||||
|   beforeRequest: [setBaseUrl, addAuthHeader], |   beforeRequest: [setBaseUrl, addAuthHeader], | ||||||
|   | |||||||
| @@ -0,0 +1,161 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Create company', | ||||||
|  |   key: 'createCompany', | ||||||
|  |   description: 'Creates a new company.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Company Owner', | ||||||
|  |       key: 'companyOwnerId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listContactOwners', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Company Name', | ||||||
|  |       key: 'companyName', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Phone', | ||||||
|  |       key: 'phone', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Website', | ||||||
|  |       key: 'website', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Tags', | ||||||
|  |       key: 'tags', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Tag', | ||||||
|  |           key: 'tag', | ||||||
|  |           type: 'string', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Description', | ||||||
|  |       key: 'description', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Billing Street', | ||||||
|  |       key: 'billingStreet', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Billing City', | ||||||
|  |       key: 'billingCity', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Billing State', | ||||||
|  |       key: 'billingState', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Billing Country', | ||||||
|  |       key: 'billingCountry', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Billing Code', | ||||||
|  |       key: 'billingCode', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { | ||||||
|  |       contactOwnerId, | ||||||
|  |       companyName, | ||||||
|  |       phone, | ||||||
|  |       website, | ||||||
|  |       tags, | ||||||
|  |       description, | ||||||
|  |       billingStreet, | ||||||
|  |       billingCity, | ||||||
|  |       billingState, | ||||||
|  |       billingCountry, | ||||||
|  |       billingCode, | ||||||
|  |     } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const allTags = tags.map((tag) => ({ | ||||||
|  |       name: tag.tag, | ||||||
|  |     })); | ||||||
|  |  | ||||||
|  |     const body = { | ||||||
|  |       data: [ | ||||||
|  |         { | ||||||
|  |           Owner: { | ||||||
|  |             id: contactOwnerId, | ||||||
|  |           }, | ||||||
|  |           Account_Name: companyName, | ||||||
|  |           Phone: phone, | ||||||
|  |           Website: website, | ||||||
|  |           Tag: allTags, | ||||||
|  |           Description: description, | ||||||
|  |           Billing_Street: billingStreet, | ||||||
|  |           Billing_City: billingCity, | ||||||
|  |           Billing_State: billingState, | ||||||
|  |           Billing_Country: billingCountry, | ||||||
|  |           Billing_Code: billingCode, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.post(`/bigin/v2/Accounts`, body); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: data[0], | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,217 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Create contact', | ||||||
|  |   key: 'createContact', | ||||||
|  |   description: 'Creates a new contact.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Contact Owner', | ||||||
|  |       key: 'contactOwnerId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listContactOwners', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'First Name', | ||||||
|  |       key: 'firstName', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Last Name', | ||||||
|  |       key: 'lastName', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Title', | ||||||
|  |       key: 'title', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Email', | ||||||
|  |       key: 'email', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Company ID', | ||||||
|  |       key: 'companyId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listCompanies', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mobile', | ||||||
|  |       key: 'mobile', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Email Opt Out', | ||||||
|  |       key: 'emailOptOut', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'True', value: true }, | ||||||
|  |         { label: 'False', value: false }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Tags', | ||||||
|  |       key: 'tags', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Tag', | ||||||
|  |           key: 'tag', | ||||||
|  |           type: 'string', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Description', | ||||||
|  |       key: 'description', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mailing Street', | ||||||
|  |       key: 'mailingStreet', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mailing City', | ||||||
|  |       key: 'mailingCity', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mailing State', | ||||||
|  |       key: 'mailingState', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mailing Country', | ||||||
|  |       key: 'mailingCountry', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Mailing Zip', | ||||||
|  |       key: 'mailingZip', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { | ||||||
|  |       contactOwnerId, | ||||||
|  |       firstName, | ||||||
|  |       lastName, | ||||||
|  |       title, | ||||||
|  |       email, | ||||||
|  |       companyId, | ||||||
|  |       mobile, | ||||||
|  |       emailOptOut, | ||||||
|  |       tags, | ||||||
|  |       description, | ||||||
|  |       mailingStreet, | ||||||
|  |       mailingCity, | ||||||
|  |       mailingState, | ||||||
|  |       mailingCountry, | ||||||
|  |       mailingZip, | ||||||
|  |     } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const allTags = tags.map((tag) => ({ | ||||||
|  |       name: tag.tag, | ||||||
|  |     })); | ||||||
|  |  | ||||||
|  |     const body = { | ||||||
|  |       data: [ | ||||||
|  |         { | ||||||
|  |           Owner: { | ||||||
|  |             id: contactOwnerId, | ||||||
|  |           }, | ||||||
|  |           Account_Name: { | ||||||
|  |             id: companyId, | ||||||
|  |           }, | ||||||
|  |           First_Name: firstName, | ||||||
|  |           Last_Name: lastName, | ||||||
|  |           Title: title, | ||||||
|  |           Email: email, | ||||||
|  |           Mobile: mobile, | ||||||
|  |           Email_Opt_Out: emailOptOut, | ||||||
|  |           Tag: allTags, | ||||||
|  |           Description: description, | ||||||
|  |           Mailing_Street: mailingStreet, | ||||||
|  |           Mailing_City: mailingCity, | ||||||
|  |           Mailing_State: mailingState, | ||||||
|  |           Mailing_Country: mailingCountry, | ||||||
|  |           Mailing_Zip: mailingZip, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.post(`/bigin/v2/Contacts`, body); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: data[0], | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,293 @@ | |||||||
|  | import defineAction from '../../../../helpers/define-action.js'; | ||||||
|  |  | ||||||
|  | export default defineAction({ | ||||||
|  |   name: 'Create event', | ||||||
|  |   key: 'createEvent', | ||||||
|  |   description: 'Creates a new event.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Host', | ||||||
|  |       key: 'hostId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listContactOwners', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Title', | ||||||
|  |       key: 'title', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'From', | ||||||
|  |       key: 'from', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: 'The date format is ISO8601 (yyyy-mm-ddTHH:mm:ssZ).', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'To', | ||||||
|  |       key: 'to', | ||||||
|  |       type: 'string', | ||||||
|  |       required: true, | ||||||
|  |       description: 'The date format is ISO8601 (yyyy-mm-ddTHH:mm:ssZ).', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'All Day', | ||||||
|  |       key: 'allDay', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'True', value: true }, | ||||||
|  |         { label: 'False', value: false }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Frequency (Recurring Activity)', | ||||||
|  |       key: 'frequency', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Specifies the frequency of event recurrence. The options include DAILY, WEEKLY, MONTHLY, or YEARLY.', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Daily Events', value: 'DAILY' }, | ||||||
|  |         { label: 'Weekly Events', value: 'WEEKLY' }, | ||||||
|  |         { label: 'Monthly Events', value: 'MONTHLY' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Interval (Recurring Activity)', | ||||||
|  |       key: 'interval', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Specifies the time difference between individual events. The INTERVAL can be anywhere from 1 to 99. For instance, with a WEEKLY event set at an INTERVAL of 2, there will be a two-week gap between each occurrence.', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'By Month Day (Recurring Activity)', | ||||||
|  |       key: 'byMonthDay', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'This specifies the date within the month when the event recurs. The BYMONTHDAY value can be any number from 1 to 31. This rule applies exclusively to events that repeat monthly or yearly.', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'By Week (Recurring Activity)', | ||||||
|  |       key: 'byWeek', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Only relevant for events that occur on a monthly or yearly basis.', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'First week of the month', value: '1' }, | ||||||
|  |         { label: 'Second week of the month', value: '2' }, | ||||||
|  |         { label: 'Third week of the month', value: '3' }, | ||||||
|  |         { label: 'Fourth week of the month', value: '4' }, | ||||||
|  |         { label: 'Last week of the month', value: '-1' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'By Day (Recurring Activity)', | ||||||
|  |       key: 'byDay', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'This signifies the weekday when the event recurs. The options include SU, MO, TU, WE, TH, FR, or SA. This rule applies to events that repeat daily, weekly, monthly, and yearly (should not be combined with INTERVAL).', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Count (Recurring Activity)', | ||||||
|  |       key: 'count', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Specifies the number of events you wish to generate. The count value range from 1 to 99.', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Until (Recurring Activity)', | ||||||
|  |       key: 'until', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Specifies the concluding date for the event recurrence. Please input the date in the YYYY-MM-DD format.', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Reminder', | ||||||
|  |       key: 'reminder', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: | ||||||
|  |         'Provide the reminder list to notify or prompt participants prior to the event.', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: '5 min', value: '5 minutes' }, | ||||||
|  |         { label: '10 min', value: '10 minutes' }, | ||||||
|  |         { label: '15 min', value: '15 minutes' }, | ||||||
|  |         { label: '1 hrs', value: '1 hours' }, | ||||||
|  |         { label: '2 hrs', value: '2 hours' }, | ||||||
|  |         { label: '1 days', value: '1 days' }, | ||||||
|  |         { label: '2 days', value: '2 days' }, | ||||||
|  |         { label: '1 weeks', value: '1 weeks' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Location', | ||||||
|  |       key: 'location', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Related Module', | ||||||
|  |       key: 'relatedModule', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |       options: [ | ||||||
|  |         { label: 'Companies', value: 'Accounts' }, | ||||||
|  |         { label: 'Contacts', value: 'Contacts' }, | ||||||
|  |         { label: 'Deals', value: 'Deals' }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Participants', | ||||||
|  |       key: 'participants', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: 'Email Address of participants.', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Participant', | ||||||
|  |           key: 'participant', | ||||||
|  |           type: 'string', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Description', | ||||||
|  |       key: 'description', | ||||||
|  |       type: 'string', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       variables: true, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: 'Tags', | ||||||
|  |       key: 'tags', | ||||||
|  |       type: 'dynamic', | ||||||
|  |       required: false, | ||||||
|  |       description: '', | ||||||
|  |       fields: [ | ||||||
|  |         { | ||||||
|  |           label: 'Tag', | ||||||
|  |           key: 'tag', | ||||||
|  |           type: 'string', | ||||||
|  |           required: false, | ||||||
|  |           variables: true, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const { | ||||||
|  |       hostId, | ||||||
|  |       title, | ||||||
|  |       from, | ||||||
|  |       to, | ||||||
|  |       allDay, | ||||||
|  |       frequency, | ||||||
|  |       interval, | ||||||
|  |       byMonthDay, | ||||||
|  |       byDay, | ||||||
|  |       byWeek, | ||||||
|  |       count, | ||||||
|  |       until, | ||||||
|  |       reminder, | ||||||
|  |       location, | ||||||
|  |       relatedModule, | ||||||
|  |       participants, | ||||||
|  |       description, | ||||||
|  |       tags, | ||||||
|  |     } = $.step.parameters; | ||||||
|  |  | ||||||
|  |     const allTags = tags.map((tag) => ({ | ||||||
|  |       name: tag.tag, | ||||||
|  |     })); | ||||||
|  |  | ||||||
|  |     const allParticipants = participants.map((participant) => ({ | ||||||
|  |       type: 'email', | ||||||
|  |       participant: participant.participant, | ||||||
|  |     })); | ||||||
|  |  | ||||||
|  |     const [unit, period] = reminder.split(' '); | ||||||
|  |  | ||||||
|  |     let rrule = `FREQ=${frequency};INTERVAL=${interval};`; | ||||||
|  |     if (count) rrule += `COUNT=${count};`; | ||||||
|  |     if (byMonthDay) rrule += `BYMONTHDAY=${byMonthDay};`; | ||||||
|  |     if (byDay) rrule += `BYDAY=${byDay};`; | ||||||
|  |     if (byWeek) rrule += `BYSETPOS=${byWeek};`; | ||||||
|  |     if (until) rrule += `UNTIL=${until};`; | ||||||
|  |  | ||||||
|  |     const body = { | ||||||
|  |       data: [ | ||||||
|  |         { | ||||||
|  |           Owner: { | ||||||
|  |             id: hostId, | ||||||
|  |           }, | ||||||
|  |           Event_Title: title, | ||||||
|  |           Start_DateTime: from, | ||||||
|  |           End_DateTime: to, | ||||||
|  |           All_day: allDay, | ||||||
|  |           $se_module: relatedModule, | ||||||
|  |           Recurring_Activity: { | ||||||
|  |             RRULE: rrule, | ||||||
|  |           }, | ||||||
|  |           Remind_At: [ | ||||||
|  |             { | ||||||
|  |               unit: Number(unit), | ||||||
|  |               period, | ||||||
|  |             }, | ||||||
|  |           ], | ||||||
|  |           Venue: location, | ||||||
|  |           Participants: allParticipants, | ||||||
|  |           Description: description, | ||||||
|  |           Tag: allTags, | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.post(`/bigin/v2/Events`, body); | ||||||
|  |  | ||||||
|  |     $.setActionItem({ | ||||||
|  |       raw: data, | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,5 @@ | |||||||
|  | import createCompany from './create-company/index.js'; | ||||||
|  | import createContact from './create-contact/index.js'; | ||||||
|  | import createEvent from './create-event/index.js'; | ||||||
|  |  | ||||||
|  | export default [createCompany, createContact, createEvent]; | ||||||
| @@ -0,0 +1,32 @@ | |||||||
|  | <svg xmlns:xlink="http://www.w3.org/1999/xlink" xmlns="http://www.w3.org/2000/svg" id="b" data-name="Layer 2" width="327.714" height="120" viewBox="0 0 327.714 120"> | ||||||
|  |   <defs> | ||||||
|  |     <style> | ||||||
|  |       .d { | ||||||
|  |         fill: #039649; | ||||||
|  |       } | ||||||
|  |     </style> | ||||||
|  |   </defs> | ||||||
|  |   <g id="c" data-name="Layer 1"> | ||||||
|  |     <g> | ||||||
|  |       <path class="d" d="m52.39,120c-1.801,0-3.6-.6-5.101-1.5-2.1-1.5-3.6-4.2-3.6-6.9v-24.6L1.09,12.901C-.41,9.9-.41,6.601,1.391,4.2,2.891,1.5,5.89,0,8.89,0h78c2.999,0,5.698,1.5,7.199,4.2,1.801,2.702,1.801,6.3.301,9.002l-5.101,9h10.201c2.999,0,5.698,1.5,7.199,4.2,1.801,2.7,1.801,6.3.301,9l-29.701,51.3v18.599c0,3.899-2.399,6.9-5.999,8.099l-16.2,6.3c-.6.301-1.799.301-2.7.301M8.89,8.4q-.301.301-.301.602l42.301,73.798c1.199,2.1,1.199,3.299,1.199,4.501v23.998s.301.301.6,0l16.2-6.3h.6v-17.999c0-1.199,0-3.299,1.201-4.8l29.4-50.999c0-.301,0-.6-.301-.6l-53.699-.301,14.399,24.901,10.201-17.7c1.199-2.1,3.6-2.7,5.7-1.5,2.1,1.199,2.7,3.6,1.498,5.7l-11.998,20.699c-2.702,4.2-8.701,3.901-11.102.301l-17.4-30.9c-1.199-1.799-1.199-4.2,0-6.3,1.201-2.1,3.301-3.6,5.7-3.6h36.6l7.499-12.899s0-.301-.299-.602H8.89Z"/> | ||||||
|  |       <g> | ||||||
|  |         <path d="m181.725,48.737c0,3.089-.54,5.684-1.618,7.788-1.081,2.104-2.548,3.79-4.407,5.062-1.858,1.27-4.016,2.179-6.476,2.726-2.459.547-5.069.819-7.828.819h-24.591V6.521h23.034c2.676,0,5.198.24,7.561.718,2.364.479,4.427,1.311,6.189,2.5,1.762,1.189,3.149,2.787,4.16,4.795,1.011,2.008,1.517,4.53,1.517,7.562,0,3.17-.786,5.813-2.357,7.93-1.57,2.119-3.913,3.628-7.028,4.53,4.017.819,6.995,2.371,8.935,4.652s2.91,5.458,2.91,9.529Zm-33.813-17.624h10.533c1.612,0,3.033-.136,4.263-.41,1.23-.272,2.268-.738,3.115-1.393.846-.656,1.489-1.55,1.927-2.684.436-1.134.655-2.562.655-4.284,0-1.612-.267-2.923-.799-3.934-.532-1.01-1.25-1.809-2.152-2.398-.902-.587-1.967-.99-3.197-1.209s-2.542-.328-3.934-.328h-10.411v16.64Zm0,26.067h12.09c1.64,0,3.115-.169,4.427-.512,1.311-.342,2.424-.894,3.341-1.66.915-.764,1.612-1.748,2.091-2.951.478-1.202.716-2.663.716-4.385,0-1.802-.294-3.285-.881-4.447-.588-1.161-1.394-2.083-2.419-2.766s-2.227-1.154-3.606-1.414c-1.381-.26-2.863-.39-4.447-.39h-11.312v18.525Z"/> | ||||||
|  |         <path d="m202.257,9.555c0,.847-.151,1.633-.451,2.356-.3.724-.716,1.34-1.25,1.844-.532.507-1.167.902-1.905,1.189s-1.53.431-2.377.431c-.819,0-1.598-.144-2.336-.431s-1.388-.69-1.947-1.209c-.56-.519-.998-1.133-1.311-1.844-.315-.711-.471-1.489-.471-2.336s.156-1.626.471-2.336c.314-.711.744-1.325,1.291-1.845.546-.519,1.187-.922,1.925-1.209s1.53-.431,2.377-.431c.821,0,1.598.144,2.336.431s1.373.69,1.907,1.209c.532.52.955,1.134,1.27,1.845.314.71.471,1.489.471,2.336Zm-.778,12.705v42.871h-10.246V22.261h10.246Z"/> | ||||||
|  |         <path d="m254.186,61.935c0,3.361-.587,6.236-1.762,8.627-1.174,2.391-2.78,4.352-4.815,5.882-2.036,1.529-4.407,2.65-7.111,3.361-2.706.71-5.589,1.065-8.648,1.065-2.023,0-4.037-.157-6.045-.471-2.009-.314-3.908-.861-5.697-1.64-1.79-.778-3.395-1.837-4.816-3.175-1.421-1.34-2.555-3.021-3.402-5.042l8.074-3.525c.519,1.202,1.202,2.193,2.049,2.971.847.779,1.797,1.408,2.848,1.885,1.051.479,2.172.814,3.361,1.005s2.398.287,3.628.287c1.885,0,3.572-.232,5.062-.696,1.489-.466,2.752-1.169,3.79-2.111,1.039-.943,1.838-2.132,2.399-3.566.559-1.434.839-3.107.839-5.02v-6.393c-.71,1.229-1.592,2.336-2.643,3.319-1.053.983-2.207,1.824-3.464,2.52s-2.582,1.237-3.976,1.62c-1.393.383-2.814.574-4.263.574-3.033,0-5.737-.56-8.114-1.681-2.377-1.119-4.379-2.636-6.005-4.55-1.625-1.912-2.862-4.139-3.709-6.68-.847-2.542-1.27-5.233-1.27-8.074,0-2.814.451-5.491,1.353-8.033.901-2.542,2.192-4.775,3.873-6.702,1.68-1.927,3.709-3.464,6.086-4.611,2.376-1.147,5.054-1.721,8.033-1.721,2.923,0,5.621.594,8.094,1.782,2.472,1.189,4.473,3.082,6.004,5.677v-6.557h10.246v39.674Zm-33.198-19.017c0,1.666.252,3.265.758,4.795s1.237,2.876,2.193,4.037c.957,1.162,2.137,2.084,3.545,2.767s3.013,1.025,4.816,1.025c2.021,0,3.784-.355,5.287-1.066,1.502-.711,2.746-1.673,3.729-2.89.985-1.215,1.722-2.643,2.213-4.283.492-1.64.738-3.374.738-5.206,0-1.802-.239-3.49-.716-5.062-.479-1.57-1.203-2.93-2.173-4.077s-2.179-2.056-3.626-2.726c-1.449-.67-3.157-1.005-5.123-1.005-2.049,0-3.812.363-5.287,1.086-1.476.724-2.684,1.709-3.628,2.951-.943,1.243-1.633,2.699-2.069,4.365-.438,1.666-.656,3.429-.656,5.287Z"/> | ||||||
|  |         <path d="m277.096,9.555c0,.847-.151,1.633-.451,2.356-.3.724-.716,1.34-1.25,1.844-.532.507-1.167.902-1.905,1.189s-1.53.431-2.377.431c-.819,0-1.598-.144-2.336-.431s-1.388-.69-1.947-1.209c-.56-.519-.998-1.133-1.311-1.844-.315-.711-.471-1.489-.471-2.336s.156-1.626.471-2.336c.314-.711.744-1.325,1.291-1.845.546-.519,1.187-.922,1.925-1.209s1.53-.431,2.377-.431c.821,0,1.598.144,2.336.431s1.373.69,1.907,1.209c.532.52.955,1.134,1.27,1.845.314.71.471,1.489.471,2.336Zm-.778,12.705v42.871h-10.246V22.261h10.246Z"/> | ||||||
|  |         <path d="m308.451,29.801c-1.585,0-2.999.24-4.243.718-1.243.479-2.288,1.162-3.135,2.049-.847.889-1.496,1.961-1.947,3.217-.451,1.258-.676,2.651-.676,4.181v25.165h-10.246V22.261h10.246v6.803c.683-1.338,1.537-2.492,2.562-3.462,1.025-.97,2.165-1.769,3.422-2.399,1.257-.627,2.59-1.091,3.997-1.393,1.406-.3,2.82-.451,4.241-.451,2.623,0,4.884.431,6.783,1.291s3.464,2.049,4.694,3.565c1.229,1.517,2.131,3.307,2.704,5.37s.861,4.311.861,6.742v26.805h-10.328v-25.822c0-3.005-.711-5.341-2.132-7.008-1.421-1.666-3.688-2.5-6.803-2.5Z"/> | ||||||
|  |       </g> | ||||||
|  |       <g> | ||||||
|  |         <path d="m152.871,102.12c0,1.138-.168,2.221-.507,3.25-.337,1.028-.828,1.935-1.471,2.72s-1.436,1.41-2.379,1.874-2.021.696-3.234.696c-.579,0-1.155-.061-1.723-.183-.569-.121-1.107-.306-1.613-.553-.505-.247-.974-.561-1.407-.941s-.801-.822-1.107-1.328v2.72h-2.625v-24.446h2.625v10.689c.273-.464.632-.882,1.076-1.257.442-.374.93-.696,1.463-.964.531-.27,1.082-.477,1.652-.626.569-.146,1.122-.22,1.66-.22,1.244,0,2.34.227,3.289.679.95.454,1.743,1.068,2.38,1.843s1.117,1.685,1.44,2.728c.321,1.044.482,2.151.482,3.321Zm-13.536.126c0,.917.123,1.756.371,2.515.249.759.614,1.415,1.1,1.968.485.553,1.079.984,1.787,1.289.706.306,1.517.459,2.435.459.991,0,1.813-.178,2.467-.53.653-.354,1.178-.828,1.573-1.424.395-.595.674-1.283.838-2.063.163-.78.245-1.603.245-2.467,0-.801-.104-1.576-.308-2.325-.206-.748-.52-1.415-.941-1.999-.422-.586-.958-1.055-1.605-1.407-.648-.354-1.415-.53-2.3-.53-.981,0-1.827.174-2.538.521-.711.349-1.3.818-1.764,1.409-.464.59-.806,1.28-1.028,2.071s-.332,1.629-.332,2.514Z"/> | ||||||
|  |         <path d="m169.949,93.834l-9.141,22.297h-2.656l3.131-7.464-6.388-14.833h2.814l4.965,11.86,4.586-11.86h2.689Z"/> | ||||||
|  |         <path d="m194.11,108.034v2.34h-16.414v-1.139l13.189-19.228h-12.05v-2.246h15.465v1.139l-12.84,19.134h12.65Z"/> | ||||||
|  |         <path d="m213.229,102.215c0,1.202-.206,2.319-.617,3.352-.411,1.034-.986,1.927-1.723,2.681-.739.753-1.611,1.344-2.618,1.77-1.007.428-2.106.641-3.296.641-1.256,0-2.388-.222-3.4-.665s-1.874-1.053-2.585-1.834-1.257-1.697-1.637-2.752c-.38-1.053-.57-2.192-.57-3.416,0-1.191.201-2.3.601-3.329.4-1.028.964-1.92,1.692-2.68.727-.759,1.594-1.354,2.601-1.787s2.116-.648,3.329-.648c1.254,0,2.391.22,3.408.663s1.881,1.055,2.593,1.835,1.26,1.697,1.644,2.751c.385,1.055.578,2.192.578,3.416Zm-13.726.031c0,.886.117,1.708.349,2.467s.579,1.418,1.043,1.977c.464.558,1.038.995,1.723,1.311.685.317,1.481.476,2.388.476.938,0,1.753-.175,2.443-.522.691-.349,1.263-.816,1.717-1.407.452-.591.79-1.275,1.012-2.056.22-.78.332-1.602.332-2.466,0-.844-.117-1.646-.349-2.404-.232-.759-.579-1.429-1.043-2.008s-1.038-1.038-1.723-1.376c-.685-.337-1.481-.505-2.388-.505-.971,0-1.802.174-2.498.521-.696.349-1.265.82-1.708,1.416-.443.595-.77,1.288-.981,2.078-.21.792-.316,1.624-.316,2.498Z"/> | ||||||
|  |         <path d="m224.423,95.827c-.696,0-1.323.105-1.881.316s-1.034.512-1.423.902c-.391.39-.691.864-.902,1.423s-.316,1.186-.316,1.881v10.026h-2.625v-24.446h2.625v10.531c.253-.516.564-.956.933-1.32s.777-.663,1.226-.902c.447-.237.933-.411,1.454-.521.522-.111,1.057-.166,1.605-.166,1.033,0,1.937.171,2.712.513.775.343,1.423.818,1.945,1.424.522.605.915,1.326,1.178,2.157.263.833.395,1.745.395,2.735v9.994h-2.625v-10.184c0-1.423-.36-2.506-1.083-3.25-.722-.742-1.795-1.114-3.217-1.114Z"/> | ||||||
|  |         <path d="m251.685,102.215c0,1.202-.206,2.319-.617,3.352-.411,1.034-.986,1.927-1.723,2.681-.739.753-1.611,1.344-2.618,1.77-1.007.428-2.106.641-3.296.641-1.256,0-2.388-.222-3.4-.665s-1.874-1.053-2.585-1.834-1.257-1.697-1.637-2.752c-.38-1.053-.57-2.192-.57-3.416,0-1.191.201-2.3.601-3.329.4-1.028.964-1.92,1.692-2.68.727-.759,1.594-1.354,2.601-1.787s2.116-.648,3.329-.648c1.254,0,2.391.22,3.408.663s1.881,1.055,2.593,1.835,1.26,1.697,1.644,2.751c.385,1.055.578,2.192.578,3.416Zm-13.726.031c0,.886.117,1.708.349,2.467s.579,1.418,1.043,1.977c.464.558,1.038.995,1.723,1.311.685.317,1.481.476,2.388.476.938,0,1.753-.175,2.443-.522.691-.349,1.263-.816,1.717-1.407.452-.591.79-1.275,1.012-2.056.22-.78.332-1.602.332-2.466,0-.844-.117-1.646-.349-2.404-.232-.759-.579-1.429-1.043-2.008s-1.038-1.038-1.723-1.376c-.685-.337-1.481-.505-2.388-.505-.971,0-1.802.174-2.498.521-.696.349-1.265.82-1.708,1.416-.443.595-.77,1.288-.981,2.078-.21.792-.316,1.624-.316,2.498Z"/> | ||||||
|  |         <path d="m281.097,103.923c-.38.959-.884,1.85-1.511,2.672-.627.823-1.346,1.534-2.157,2.135-.812.601-1.703,1.073-2.673,1.415-.969.342-1.976.514-3.02.514-1.76,0-3.312-.295-4.656-.886-1.345-.59-2.472-1.407-3.385-2.45-.912-1.044-1.603-2.279-2.072-3.709-.469-1.428-.704-2.98-.704-4.657,0-1.033.114-2.037.341-3.013.227-.974.553-1.889.98-2.743.428-.854.95-1.637,1.565-2.348.617-.711,1.318-1.32,2.104-1.827.785-.505,1.652-.901,2.601-1.186.949-.284,1.961-.426,3.036-.426,1.012,0,2.003.148,2.973.442.971.295,1.866.718,2.689,1.266.822.548,1.55,1.209,2.182,1.984s1.127,1.642,1.486,2.602l-2.246.98c-.295-.768-.673-1.468-1.13-2.095-.459-.627-.989-1.162-1.59-1.604-.601-.443-1.265-.785-1.992-1.029-.728-.242-1.508-.363-2.341-.363-1.359,0-2.532.268-3.518.806s-1.797,1.254-2.435,2.151c-.638.895-1.107,1.919-1.407,3.067-.301,1.149-.451,2.335-.451,3.558,0,1.244.166,2.424.498,3.543.333,1.117.833,2.098,1.503,2.94.669.844,1.507,1.513,2.514,2.008,1.007.496,2.18.744,3.518.744.801,0,1.576-.143,2.325-.428.749-.284,1.44-.671,2.072-1.162.632-.49,1.191-1.064,1.675-1.723.485-.658.864-1.357,1.139-2.095l2.088.917Z"/> | ||||||
|  |         <path d="m287.991,100.539v9.835h-2.751v-22.613h7.431c1.044,0,2.042.111,2.997.332.954.222,1.795.586,2.522,1.092.728.505,1.307,1.168,1.74,1.984.431.818.648,1.822.648,3.013,0,.885-.137,1.673-.411,2.364-.275.691-.66,1.289-1.155,1.795-.496.507-1.086.925-1.771,1.257-.685.333-1.44.583-2.261.752l6.926,10.026h-3.068l-6.736-9.835h-4.112Zm0-2.183h4.871c.727,0,1.393-.075,1.999-.228s1.131-.402,1.574-.744c.442-.342.785-.785,1.028-1.328s.363-1.21.363-2.001c0-.801-.126-1.466-.378-1.992-.254-.527-.601-.943-1.044-1.25-.443-.305-.967-.521-1.573-.648s-1.262-.189-1.968-.189h-4.871v8.38Z"/> | ||||||
|  |         <path d="m308.231,91.335v19.039h-2.529v-22.613h3.826l7.242,17.932,7.148-17.932h3.795v22.613h-2.752v-19.039l-7.653,19.039h-1.17l-7.907-19.039Z"/> | ||||||
|  |       </g> | ||||||
|  |     </g> | ||||||
|  |   </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 10 KiB | 
| @@ -7,15 +7,17 @@ export default async function generateAuthUrl($) { | |||||||
|   ); |   ); | ||||||
|   const redirectUri = oauthRedirectUrlField.value; |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|   const searchParams = new URLSearchParams({ |   const searchParams = new URLSearchParams({ | ||||||
|  |     scope: authScope.join(','), | ||||||
|     client_id: $.auth.data.clientId, |     client_id: $.auth.data.clientId, | ||||||
|     redirect_uri: redirectUri, |  | ||||||
|     prompt: 'select_account', |  | ||||||
|     scope: authScope.join(' '), |  | ||||||
|     response_type: 'code', |     response_type: 'code', | ||||||
|     access_type: 'offline', |     access_type: 'offline', | ||||||
|  |     redirect_uri: redirectUri, | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   const url = `https://accounts.google.com/o/oauth2/v2/auth?${searchParams.toString()}`; |   const domain = | ||||||
|  |     $.auth.data.region !== 'cn' ? 'account.zoho.com' : 'accounts.zoho.com.cn'; | ||||||
|  | 
 | ||||||
|  |   const url = `https://${domain}/oauth/v2/auth?${searchParams.toString()}`; | ||||||
| 
 | 
 | ||||||
|   await $.auth.set({ |   await $.auth.set({ | ||||||
|     url, |     url, | ||||||
| @@ -11,22 +11,28 @@ export default { | |||||||
|       type: 'string', |       type: 'string', | ||||||
|       required: true, |       required: true, | ||||||
|       readOnly: true, |       readOnly: true, | ||||||
|       value: '{WEB_APP_URL}/app/you-need-a-budget/connections/add', |       value: '{WEB_APP_URL}/app/bigin-by-zoho-crm/connections/add', | ||||||
|       placeholder: null, |       placeholder: null, | ||||||
|       description: |       description: | ||||||
|         'When asked to input a redirect URL in You Need A Budget, enter the URL above.', |         'When asked to input a redirect URL in Bigin By Zoho CRM, enter the URL above.', | ||||||
|       clickToCopy: true, |       clickToCopy: true, | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       key: 'screenName', |       key: 'region', | ||||||
|       label: 'Screen Name', |       label: 'Region', | ||||||
|       type: 'string', |       type: 'dropdown', | ||||||
|       required: true, |       required: true, | ||||||
|       readOnly: false, |       readOnly: false, | ||||||
|       value: null, |       value: null, | ||||||
|       placeholder: null, |       placeholder: null, | ||||||
|       description: |       description: '', | ||||||
|         'Screen name of your connection to be used on Automatisch UI.', |       options: [ | ||||||
|  |         { label: 'United States', value: 'us' }, | ||||||
|  |         { label: 'European Union', value: 'eu' }, | ||||||
|  |         { label: 'Australia', value: 'au' }, | ||||||
|  |         { label: 'India', value: 'in' }, | ||||||
|  |         { label: 'China', value: 'cn' }, | ||||||
|  |       ], | ||||||
|       clickToCopy: false, |       clickToCopy: false, | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | import getCurrentOrganization from '../common/get-current-organization.js'; | ||||||
|  |  | ||||||
|  | const isStillVerified = async ($) => { | ||||||
|  |   const org = await getCurrentOrganization($); | ||||||
|  |   return !!org.id; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default isStillVerified; | ||||||
| @@ -1,24 +1,33 @@ | |||||||
| import { URLSearchParams } from 'node:url'; | import { URLSearchParams } from 'node:url'; | ||||||
|  | 
 | ||||||
| import authScope from '../common/auth-scope.js'; | import authScope from '../common/auth-scope.js'; | ||||||
|  | import { regionUrlMap } from '../common/region-url-map.js'; | ||||||
| 
 | 
 | ||||||
| const refreshToken = async ($) => { | const refreshToken = async ($) => { | ||||||
|  |   const location = $.auth.data.location; | ||||||
|   const params = new URLSearchParams({ |   const params = new URLSearchParams({ | ||||||
|     client_id: $.auth.data.clientId, |     client_id: $.auth.data.clientId, | ||||||
|     client_secret: $.auth.data.clientSecret, |     client_secret: $.auth.data.clientSecret, | ||||||
|     grant_type: 'refresh_token', |  | ||||||
|     refresh_token: $.auth.data.refreshToken, |     refresh_token: $.auth.data.refreshToken, | ||||||
|  |     grant_type: 'refresh_token', | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   const { data } = await $.http.post( |   const { data } = await $.http.post( | ||||||
|     'https://oauth2.googleapis.com/token', |     `${regionUrlMap[location]}/oauth/v2/token`, | ||||||
|     params.toString() |     params.toString(), | ||||||
|  |     { | ||||||
|  |       additionalProperties: { | ||||||
|  |         skipAddingBaseUrl: true, | ||||||
|  |       }, | ||||||
|  |     } | ||||||
|   ); |   ); | ||||||
| 
 | 
 | ||||||
|   await $.auth.set({ |   await $.auth.set({ | ||||||
|     accessToken: data.access_token, |     accessToken: data.access_token, | ||||||
|     expiresIn: data.expires_in, |     apiDomain: data.api_domain, | ||||||
|     scope: authScope.join(' '), |     scope: authScope.join(','), | ||||||
|     tokenType: data.token_type, |     tokenType: data.token_type, | ||||||
|  |     expiresIn: data.expires_in, | ||||||
|   }); |   }); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| @@ -1,41 +1,45 @@ | |||||||
| import getCurrentUser from '../common/get-current-user.js'; | import { URLSearchParams } from 'node:url'; | ||||||
|  | import { regionUrlMap } from '../common/region-url-map.js'; | ||||||
|  | import getCurrentOrganization from '../common/get-current-organization.js'; | ||||||
| 
 | 
 | ||||||
| const verifyCredentials = async ($) => { | const verifyCredentials = async ($) => { | ||||||
|   const oauthRedirectUrlField = $.app.auth.fields.find( |   const oauthRedirectUrlField = $.app.auth.fields.find( | ||||||
|     (field) => field.key == 'oAuthRedirectUrl' |     (field) => field.key == 'oAuthRedirectUrl' | ||||||
|   ); |   ); | ||||||
|   const redirectUri = oauthRedirectUrlField.value; |   const redirectUri = oauthRedirectUrlField.value; | ||||||
|   const { data } = await $.http.post(`https://oauth2.googleapis.com/token`, { |   const location = $.auth.data.location; | ||||||
|  |   const params = new URLSearchParams({ | ||||||
|     client_id: $.auth.data.clientId, |     client_id: $.auth.data.clientId, | ||||||
|     client_secret: $.auth.data.clientSecret, |     client_secret: $.auth.data.clientSecret, | ||||||
|     code: $.auth.data.code, |     code: $.auth.data.code, | ||||||
|     grant_type: 'authorization_code', |  | ||||||
|     redirect_uri: redirectUri, |     redirect_uri: redirectUri, | ||||||
|  |     grant_type: 'authorization_code', | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|  |   const { data } = await $.http.post( | ||||||
|  |     `${regionUrlMap[location]}/oauth/v2/token`, | ||||||
|  |     params.toString() | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|   await $.auth.set({ |   await $.auth.set({ | ||||||
|     accessToken: data.access_token, |     accessToken: data.access_token, | ||||||
|     tokenType: data.token_type, |     tokenType: data.token_type, | ||||||
|  |     apiDomain: data.api_domain, | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   const currentUser = await getCurrentUser($); |   const organization = await getCurrentOrganization($); | ||||||
| 
 | 
 | ||||||
|   const { displayName } = currentUser.names.find( |   const screenName = [organization.company_name, organization.primary_email] | ||||||
|     (name) => name.metadata.primary |     .filter(Boolean) | ||||||
|   ); |     .join(' @ '); | ||||||
|   const { value: email } = currentUser.emailAddresses.find( |  | ||||||
|     (emailAddress) => emailAddress.metadata.primary |  | ||||||
|   ); |  | ||||||
| 
 | 
 | ||||||
|   await $.auth.set({ |   await $.auth.set({ | ||||||
|     clientId: $.auth.data.clientId, |     clientId: $.auth.data.clientId, | ||||||
|     clientSecret: $.auth.data.clientSecret, |     clientSecret: $.auth.data.clientSecret, | ||||||
|     scope: $.auth.data.scope, |     scope: $.auth.data.scope, | ||||||
|     idToken: data.id_token, |  | ||||||
|     expiresIn: data.expires_in, |     expiresIn: data.expires_in, | ||||||
|     refreshToken: data.refresh_token, |     refreshToken: data.refresh_token, | ||||||
|     resourceName: currentUser.resourceName, |     screenName, | ||||||
|     screenName: `${displayName} - ${email}`, |  | ||||||
|   }); |   }); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| const addAuthHeader = ($, requestConfig) => { | const addAuthHeader = ($, requestConfig) => { | ||||||
|   if ($.auth.data?.accessToken) { |   if ($.auth.data?.accessToken) { | ||||||
|     requestConfig.headers.Authorization = `${$.auth.data.tokenType} ${$.auth.data.accessToken}`; |     requestConfig.headers.Authorization = `Zoho-oauthtoken ${$.auth.data.accessToken}`; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return requestConfig; |   return requestConfig; | ||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | const authScope = [ | ||||||
|  |   'ZohoBigin.notifications.ALL', | ||||||
|  |   'ZohoBigin.users.ALL', | ||||||
|  |   'ZohoBigin.modules.ALL', | ||||||
|  |   'ZohoBigin.org.READ', | ||||||
|  |   'ZohoBigin.settings.ALL', | ||||||
|  |   'ZohoBigin.modules.ALL', | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | export default authScope; | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | const getCurrentOrganization = async ($) => { | ||||||
|  |   const response = await $.http.get('/bigin/v2/org'); | ||||||
|  |   return response.data.org[0]; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default getCurrentOrganization; | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | export const regionUrlMap = { | ||||||
|  |   us: 'https://accounts.zoho.com', | ||||||
|  |   au: 'https://accounts.zoho.com.au', | ||||||
|  |   eu: 'https://accounts.zoho.eu', | ||||||
|  |   in: 'https://accounts.zoho.in', | ||||||
|  |   cn: 'https://accounts.zoho.com.cn', | ||||||
|  |   jp: 'https://accounts.zoho.jp', | ||||||
|  | }; | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | const setBaseUrl = ($, requestConfig) => { | ||||||
|  |   if (requestConfig.additionalProperties?.skipAddingBaseUrl) | ||||||
|  |     return requestConfig; | ||||||
|  |  | ||||||
|  |   const apiDomain = $.auth.data.apiDomain; | ||||||
|  |  | ||||||
|  |   if (apiDomain) { | ||||||
|  |     requestConfig.baseURL = apiDomain; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return requestConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default setBaseUrl; | ||||||
| @@ -0,0 +1,5 @@ | |||||||
|  | import listCompanies from './list-companies/index.js'; | ||||||
|  | import listOrganizations from './list-organizations/index.js'; | ||||||
|  | import listContactOwners from './list-contact-owners/index.js'; | ||||||
|  |  | ||||||
|  | export default [listCompanies, listOrganizations, listContactOwners]; | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List companies', | ||||||
|  |   key: 'listCompanies', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const companies = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const params = new URLSearchParams({ | ||||||
|  |       page: 1, | ||||||
|  |       pageSize: 200, | ||||||
|  |       fields: 'Account_Name', | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     let next = false; | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get('/bigin/v2/Accounts', { params }); | ||||||
|  |  | ||||||
|  |       if (data.info.more_records) { | ||||||
|  |         params.page = params.page + 1; | ||||||
|  |         next = true; | ||||||
|  |       } else { | ||||||
|  |         next = false; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (data.data) { | ||||||
|  |         for (const account of data.data) { | ||||||
|  |           companies.data.push({ | ||||||
|  |             value: account.id, | ||||||
|  |             name: account.Account_Name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (next); | ||||||
|  |  | ||||||
|  |     return companies; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List contact owners', | ||||||
|  |   key: 'listContactOwners', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const contactOwners = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const params = { | ||||||
|  |       type: 'AllUsers', | ||||||
|  |       page: 1, | ||||||
|  |       pageSize: 200, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     let next = false; | ||||||
|  |     do { | ||||||
|  |       const { data } = await $.http.get('/bigin/v2/users', params); | ||||||
|  |  | ||||||
|  |       if (data.users.length === params.pageSize) { | ||||||
|  |         next = true; | ||||||
|  |         params.page = params.page + 1; | ||||||
|  |       } else { | ||||||
|  |         next = false; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (data.users) { | ||||||
|  |         for (const user of data.users) { | ||||||
|  |           contactOwners.data.push({ | ||||||
|  |             value: user.id, | ||||||
|  |             name: user.full_name, | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (next); | ||||||
|  |  | ||||||
|  |     return contactOwners; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -0,0 +1,23 @@ | |||||||
|  | export default { | ||||||
|  |   name: 'List organizations', | ||||||
|  |   key: 'listOrganizations', | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const organizations = { | ||||||
|  |       data: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const { data } = await $.http.get('/bigin/v2/org'); | ||||||
|  |  | ||||||
|  |     if (data.org) { | ||||||
|  |       for (const org of data.org) { | ||||||
|  |         organizations.data.push({ | ||||||
|  |           value: org.id, | ||||||
|  |           name: org.company_name, | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return organizations; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -1,23 +1,23 @@ | |||||||
| import defineApp from '../../helpers/define-app.js'; | import defineApp from '../../helpers/define-app.js'; | ||||||
| import addAuthHeader from './common/add-auth-header.js'; | import addAuthHeader from './common/add-auth-header.js'; | ||||||
| import setBaseUrl from './common/set-base-url.js'; |  | ||||||
| import auth from './auth/index.js'; | import auth from './auth/index.js'; | ||||||
|  | import setBaseUrl from './common/set-base-url.js'; | ||||||
| import triggers from './triggers/index.js'; | import triggers from './triggers/index.js'; | ||||||
| import actions from './actions/index.js'; |  | ||||||
| import dynamicData from './dynamic-data/index.js'; | import dynamicData from './dynamic-data/index.js'; | ||||||
|  | import actions from './actions/index.js'; | ||||||
| 
 | 
 | ||||||
| export default defineApp({ | export default defineApp({ | ||||||
|   name: 'Vtiger CRM', |   name: 'Bigin By Zoho CRM', | ||||||
|   key: 'vtiger-crm', |   key: 'bigin-by-zoho-crm', | ||||||
|   iconUrl: '{BASE_URL}/apps/vtiger-crm/assets/favicon.svg', |   baseUrl: 'https://www.bigin.com', | ||||||
|   authDocUrl: '{DOCS_URL}/apps/vtiger-crm/connection', |  | ||||||
|   supportsConnections: true, |  | ||||||
|   baseUrl: '', |  | ||||||
|   apiBaseUrl: '', |   apiBaseUrl: '', | ||||||
|   primaryColor: '39a86d', |   iconUrl: '{BASE_URL}/apps/bigin-by-zoho-crm/assets/favicon.svg', | ||||||
|  |   authDocUrl: 'https://automatisch.io/docs/apps/bigin-by-zoho-crm/connection', | ||||||
|  |   primaryColor: '039649', | ||||||
|  |   supportsConnections: true, | ||||||
|   beforeRequest: [setBaseUrl, addAuthHeader], |   beforeRequest: [setBaseUrl, addAuthHeader], | ||||||
|   auth, |   auth, | ||||||
|   triggers, |   triggers, | ||||||
|   actions, |  | ||||||
|   dynamicData, |   dynamicData, | ||||||
|  |   actions, | ||||||
| }); | }); | ||||||
| @@ -0,0 +1,7 @@ | |||||||
|  | import newCalls from './new-calls/index.js'; | ||||||
|  | import newCompanies from './new-companies/index.js'; | ||||||
|  | import newContacts from './new-contacts/index.js'; | ||||||
|  | import newProducts from './new-products/index.js'; | ||||||
|  | import newTasks from './new-tasks/index.js'; | ||||||
|  |  | ||||||
|  | export default [newCalls, newCompanies, newContacts, newProducts, newTasks]; | ||||||
| @@ -0,0 +1,89 @@ | |||||||
|  | import Crypto from 'crypto'; | ||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New calls', | ||||||
|  |   key: 'newCalls', | ||||||
|  |   type: 'webhook', | ||||||
|  |   description: 'Triggers when a new call is added.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Organization', | ||||||
|  |       key: 'organizationId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: false, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listOrganizations', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: $.request.body, | ||||||
|  |       meta: { | ||||||
|  |         internalId: Crypto.randomUUID(), | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async testRun($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const sampleEventData = { | ||||||
|  |       ids: ['111111111111111111'], | ||||||
|  |       token: null, | ||||||
|  |       module: 'Calls', | ||||||
|  |       operation: 'insert', | ||||||
|  |       channel_id: organizationId, | ||||||
|  |       server_time: 1708426963120, | ||||||
|  |       query_params: {}, | ||||||
|  |       resource_uri: `${$.auth.data.apiDomain}/bigin/v1/Calls`, | ||||||
|  |       affected_fields: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: sampleEventData, | ||||||
|  |       meta: { | ||||||
|  |         internalId: sampleEventData.channel_id, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async registerHook($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const payload = { | ||||||
|  |       watch: [ | ||||||
|  |         { | ||||||
|  |           channel_id: organizationId, | ||||||
|  |           notify_url: $.webhookUrl, | ||||||
|  |           events: ['Calls.create'], | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     await $.http.post('/bigin/v2/actions/watch', payload); | ||||||
|  |  | ||||||
|  |     await $.flow.setRemoteWebhookId(organizationId); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async unregisterHook($) { | ||||||
|  |     await $.http.delete( | ||||||
|  |       `/bigin/v2/actions/watch?channel_ids=${$.flow.remoteWebhookId}` | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,89 @@ | |||||||
|  | import Crypto from 'crypto'; | ||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New companies', | ||||||
|  |   key: 'newCompanies', | ||||||
|  |   type: 'webhook', | ||||||
|  |   description: 'Triggers when a new company is created.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Organization', | ||||||
|  |       key: 'organizationId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: false, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listOrganizations', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: $.request.body, | ||||||
|  |       meta: { | ||||||
|  |         internalId: Crypto.randomUUID(), | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async testRun($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const sampleEventData = { | ||||||
|  |       ids: ['111111111111111111'], | ||||||
|  |       token: null, | ||||||
|  |       module: 'Accounts', | ||||||
|  |       operation: 'insert', | ||||||
|  |       channel_id: organizationId, | ||||||
|  |       server_time: 1708426963120, | ||||||
|  |       query_params: {}, | ||||||
|  |       resource_uri: `${$.auth.data.apiDomain}/bigin/v1/Accounts`, | ||||||
|  |       affected_fields: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: sampleEventData, | ||||||
|  |       meta: { | ||||||
|  |         internalId: sampleEventData.channel_id, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async registerHook($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const payload = { | ||||||
|  |       watch: [ | ||||||
|  |         { | ||||||
|  |           channel_id: organizationId, | ||||||
|  |           notify_url: $.webhookUrl, | ||||||
|  |           events: ['Accounts.create'], | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     await $.http.post('/bigin/v2/actions/watch', payload); | ||||||
|  |  | ||||||
|  |     await $.flow.setRemoteWebhookId(organizationId); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async unregisterHook($) { | ||||||
|  |     await $.http.delete( | ||||||
|  |       `/bigin/v2/actions/watch?channel_ids=${$.flow.remoteWebhookId}` | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,89 @@ | |||||||
|  | import Crypto from 'crypto'; | ||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New contacts', | ||||||
|  |   key: 'newContacts', | ||||||
|  |   type: 'webhook', | ||||||
|  |   description: 'Triggers when a new contact is created.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Organization', | ||||||
|  |       key: 'organizationId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: false, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listOrganizations', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: $.request.body, | ||||||
|  |       meta: { | ||||||
|  |         internalId: Crypto.randomUUID(), | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async testRun($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const sampleEventData = { | ||||||
|  |       ids: ['111111111111111111'], | ||||||
|  |       token: null, | ||||||
|  |       module: 'Contacts', | ||||||
|  |       operation: 'insert', | ||||||
|  |       channel_id: organizationId, | ||||||
|  |       server_time: 1708426963120, | ||||||
|  |       query_params: {}, | ||||||
|  |       resource_uri: `${$.auth.data.apiDomain}/bigin/v1/Contacts`, | ||||||
|  |       affected_fields: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: sampleEventData, | ||||||
|  |       meta: { | ||||||
|  |         internalId: sampleEventData.channel_id, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async registerHook($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const payload = { | ||||||
|  |       watch: [ | ||||||
|  |         { | ||||||
|  |           channel_id: organizationId, | ||||||
|  |           notify_url: $.webhookUrl, | ||||||
|  |           events: ['Contacts.create'], | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     await $.http.post('/bigin/v2/actions/watch', payload); | ||||||
|  |  | ||||||
|  |     await $.flow.setRemoteWebhookId(organizationId); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async unregisterHook($) { | ||||||
|  |     await $.http.delete( | ||||||
|  |       `/bigin/v2/actions/watch?channel_ids=${$.flow.remoteWebhookId}` | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,89 @@ | |||||||
|  | import Crypto from 'crypto'; | ||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New products', | ||||||
|  |   key: 'newProducts', | ||||||
|  |   type: 'webhook', | ||||||
|  |   description: 'Triggers when a new product is created.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Organization', | ||||||
|  |       key: 'organizationId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: false, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listOrganizations', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: $.request.body, | ||||||
|  |       meta: { | ||||||
|  |         internalId: Crypto.randomUUID(), | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async testRun($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const sampleEventData = { | ||||||
|  |       ids: ['111111111111111111'], | ||||||
|  |       token: null, | ||||||
|  |       module: 'Products', | ||||||
|  |       operation: 'insert', | ||||||
|  |       channel_id: organizationId, | ||||||
|  |       server_time: 1708426963120, | ||||||
|  |       query_params: {}, | ||||||
|  |       resource_uri: `${$.auth.data.apiDomain}/bigin/v1/Products`, | ||||||
|  |       affected_fields: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: sampleEventData, | ||||||
|  |       meta: { | ||||||
|  |         internalId: sampleEventData.channel_id, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async registerHook($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const payload = { | ||||||
|  |       watch: [ | ||||||
|  |         { | ||||||
|  |           channel_id: organizationId, | ||||||
|  |           notify_url: $.webhookUrl, | ||||||
|  |           events: ['Products.create'], | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     await $.http.post('/bigin/v2/actions/watch', payload); | ||||||
|  |  | ||||||
|  |     await $.flow.setRemoteWebhookId(organizationId); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async unregisterHook($) { | ||||||
|  |     await $.http.delete( | ||||||
|  |       `/bigin/v2/actions/watch?channel_ids=${$.flow.remoteWebhookId}` | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -0,0 +1,89 @@ | |||||||
|  | import Crypto from 'crypto'; | ||||||
|  | import defineTrigger from '../../../../helpers/define-trigger.js'; | ||||||
|  |  | ||||||
|  | export default defineTrigger({ | ||||||
|  |   name: 'New tasks', | ||||||
|  |   key: 'newTasks', | ||||||
|  |   type: 'webhook', | ||||||
|  |   description: 'Triggers when a new task is created.', | ||||||
|  |   arguments: [ | ||||||
|  |     { | ||||||
|  |       label: 'Organization', | ||||||
|  |       key: 'organizationId', | ||||||
|  |       type: 'dropdown', | ||||||
|  |       required: true, | ||||||
|  |       description: '', | ||||||
|  |       variables: false, | ||||||
|  |       source: { | ||||||
|  |         type: 'query', | ||||||
|  |         name: 'getDynamicData', | ||||||
|  |         arguments: [ | ||||||
|  |           { | ||||||
|  |             name: 'key', | ||||||
|  |             value: 'listOrganizations', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   async run($) { | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: $.request.body, | ||||||
|  |       meta: { | ||||||
|  |         internalId: Crypto.randomUUID(), | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async testRun($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const sampleEventData = { | ||||||
|  |       ids: ['111111111111111111'], | ||||||
|  |       token: null, | ||||||
|  |       module: 'Tasks', | ||||||
|  |       operation: 'insert', | ||||||
|  |       channel_id: organizationId, | ||||||
|  |       server_time: 1708426963120, | ||||||
|  |       query_params: {}, | ||||||
|  |       resource_uri: `${$.auth.data.apiDomain}/bigin/v1/Tasks`, | ||||||
|  |       affected_fields: [], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const dataItem = { | ||||||
|  |       raw: sampleEventData, | ||||||
|  |       meta: { | ||||||
|  |         internalId: sampleEventData.channel_id, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.pushTriggerItem(dataItem); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async registerHook($) { | ||||||
|  |     const organizationId = $.step.parameters.organizationId; | ||||||
|  |  | ||||||
|  |     const payload = { | ||||||
|  |       watch: [ | ||||||
|  |         { | ||||||
|  |           channel_id: organizationId, | ||||||
|  |           notify_url: $.webhookUrl, | ||||||
|  |           events: ['Tasks.create'], | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     await $.http.post('/bigin/v2/actions/watch', payload); | ||||||
|  |  | ||||||
|  |     await $.flow.setRemoteWebhookId(organizationId); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   async unregisterHook($) { | ||||||
|  |     await $.http.delete( | ||||||
|  |       `/bigin/v2/actions/watch?channel_ids=${$.flow.remoteWebhookId}` | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | }); | ||||||
| @@ -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: '{DOCS_URL}/apps/carbone/connection', |   authDocUrl: 'https://automatisch.io/docs/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', | ||||||
|   | |||||||
| @@ -1,27 +0,0 @@ | |||||||
| import defineAction from '../../../../helpers/define-action.js'; |  | ||||||
|  |  | ||||||
| export default defineAction({ |  | ||||||
|   name: 'Get value', |  | ||||||
|   key: 'getValue', |  | ||||||
|   description: 'Get value from the persistent datastore.', |  | ||||||
|   arguments: [ |  | ||||||
|     { |  | ||||||
|       label: 'Key', |  | ||||||
|       key: 'key', |  | ||||||
|       type: 'string', |  | ||||||
|       required: true, |  | ||||||
|       description: 'The key of your value to get.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|   ], |  | ||||||
|  |  | ||||||
|   async run($) { |  | ||||||
|     const keyValuePair = await $.datastore.get({ |  | ||||||
|       key: $.step.parameters.key, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     $.setActionItem({ |  | ||||||
|       raw: keyValuePair, |  | ||||||
|     }); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| import getValue from './get-value/index.js'; |  | ||||||
| import setValue from './set-value/index.js'; |  | ||||||
|  |  | ||||||
| export default [getValue, setValue]; |  | ||||||
| @@ -1,36 +0,0 @@ | |||||||
| import defineAction from '../../../../helpers/define-action.js'; |  | ||||||
|  |  | ||||||
| export default defineAction({ |  | ||||||
|   name: 'Set value', |  | ||||||
|   key: 'setValue', |  | ||||||
|   description: 'Set value to the persistent datastore.', |  | ||||||
|   arguments: [ |  | ||||||
|     { |  | ||||||
|       label: 'Key', |  | ||||||
|       key: 'key', |  | ||||||
|       type: 'string', |  | ||||||
|       required: true, |  | ||||||
|       description: 'The key of your value to set.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       label: 'Value', |  | ||||||
|       key: 'value', |  | ||||||
|       type: 'string', |  | ||||||
|       required: true, |  | ||||||
|       description: 'The value to set.', |  | ||||||
|       variables: true, |  | ||||||
|     }, |  | ||||||
|   ], |  | ||||||
|  |  | ||||||
|   async run($) { |  | ||||||
|     const keyValuePair = await $.datastore.set({ |  | ||||||
|       key: $.step.parameters.key, |  | ||||||
|       value: $.step.parameters.value, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     $.setActionItem({ |  | ||||||
|       raw: keyValuePair, |  | ||||||
|     }); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| <?xml version="1.0"?> |  | ||||||
| <svg xmlns="http://www.w3.org/2000/svg" fill="#000000" width="800px" height="800px" viewBox="0 0 32 32" id="icon"> |  | ||||||
|   <defs> |  | ||||||
|     <style>.cls-1{fill:none;}</style> |  | ||||||
|   </defs> |  | ||||||
|   <title>datastore</title> |  | ||||||
|   <circle cx="23" cy="23" r="1"/> |  | ||||||
|   <rect x="8" y="22" width="12" height="2"/> |  | ||||||
|   <circle cx="23" cy="9" r="1"/> |  | ||||||
|   <rect x="8" y="8" width="12" height="2"/> |  | ||||||
|   <path d="M26,14a2,2,0,0,0,2-2V6a2,2,0,0,0-2-2H6A2,2,0,0,0,4,6v6a2,2,0,0,0,2,2H8v4H6a2,2,0,0,0-2,2v6a2,2,0,0,0,2,2H26a2,2,0,0,0,2-2V20a2,2,0,0,0-2-2H24V14ZM6,6H26v6H6ZM26,26H6V20H26Zm-4-8H10V14H22Z"/> |  | ||||||
|   <rect id="_Transparent_Rectangle_" data-name="<Transparent Rectangle>" class="cls-1" width="32" height="32"/> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 704 B | 
| @@ -1,14 +0,0 @@ | |||||||
| import defineApp from '../../helpers/define-app.js'; |  | ||||||
| import actions from './actions/index.js'; |  | ||||||
|  |  | ||||||
| export default defineApp({ |  | ||||||
|   name: 'Datastore', |  | ||||||
|   key: 'datastore', |  | ||||||
|   iconUrl: '{BASE_URL}/apps/datastore/assets/favicon.svg', |  | ||||||
|   authDocUrl: '{DOCS_URL}/apps/datastore/connection', |  | ||||||
|   supportsConnections: false, |  | ||||||
|   baseUrl: '', |  | ||||||
|   apiBaseUrl: '', |  | ||||||
|   primaryColor: '001F52', |  | ||||||
|   actions, |  | ||||||
| }); |  | ||||||
| @@ -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: '{DOCS_URL}/apps/deepl/connection', |   authDocUrl: 'https://automatisch.io/docs/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: '{DOCS_URL}/apps/delay/connection', |   authDocUrl: 'https://automatisch.io/docs/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: '{DOCS_URL}/apps/discord/connection', |   authDocUrl: 'https://automatisch.io/docs/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', | ||||||
|   | |||||||
| @@ -1,16 +0,0 @@ | |||||||
| <?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> |  | ||||||
| Before Width: | Height: | Size: 1.1 KiB | 
| @@ -1,21 +0,0 @@ | |||||||
| 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, |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
| @@ -1,48 +0,0 @@ | |||||||
| 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, |  | ||||||
| }; |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| import getCurrentUser from '../common/get-current-user.js'; |  | ||||||
|  |  | ||||||
| const isStillVerified = async ($) => { |  | ||||||
|   const currentUser = await getCurrentUser($); |  | ||||||
|   return !!currentUser.response.username; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default isStillVerified; |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,34 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| const authScope = ['read', 'write', 'admin', 'email']; |  | ||||||
|  |  | ||||||
| export default authScope; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| 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; |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| import listForums from './list-forums/index.js'; |  | ||||||
|  |  | ||||||
| export default [listForums]; |  | ||||||
| @@ -1,36 +0,0 @@ | |||||||
| 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; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| 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, |  | ||||||
| }); |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| import newComments from './new-comments/index.js'; |  | ||||||
| import newFlaggedComments from './new-flagged-comments/index.js'; |  | ||||||
|  |  | ||||||
| export default [newComments, newFlaggedComments]; |  | ||||||
| @@ -1,92 +0,0 @@ | |||||||
| 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); |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
| @@ -1,60 +0,0 @@ | |||||||
| 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: '{DOCS_URL}/apps/dropbox/connection', |   authDocUrl: 'https://automatisch.io/docs/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', | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user