Compare commits

...

431 Commits

Author SHA1 Message Date
dependabot[bot]
be8b6e6f7a chore(deps): bump rollup from 2.66.0 to 2.79.2
Bumps [rollup](https://github.com/rollup/rollup) from 2.66.0 to 2.79.2.
- [Release notes](https://github.com/rollup/rollup/releases)
- [Changelog](https://github.com/rollup/rollup/blob/master/CHANGELOG.md)
- [Commits](https://github.com/rollup/rollup/compare/v2.66.0...v2.79.2)

---
updated-dependencies:
- dependency-name: rollup
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-09-27 08:04:09 +00:00
Ali BARIN
5769c82fca Merge pull request #2098 from automatisch/aut-1258
feat(useAuthenticateApp): use REST API endpoint to create connection
2024-09-25 10:03:57 +02:00
Ali BARIN
3754783ce2 Merge pull request #2103 from automatisch/AUT-1279
test: add change own user data test
2024-09-24 21:25:41 +02:00
Jakub P.
55ebe2cc0b test: add change own user data test 2024-09-24 18:42:30 +02:00
Ali BARIN
23c1a42163 chore: remove redundant create connection mutation 2024-09-24 08:31:26 +00:00
Ali BARIN
c879ffa768 feat(useAuthenticateApp): use REST API endpoint to create connection 2024-09-24 08:30:53 +00:00
Ali BARIN
32f7bbfbab chore: remove redundant execute flow mutation 2024-09-24 10:23:33 +02:00
Ali BARIN
84d5b3b158 feat: use REST API endpoint to test step 2024-09-24 10:23:33 +02:00
Ömer Faruk Aydın
c8dae9ea9a Merge pull request #2097 from automatisch/to-require-property
feat: Introduce toRequireProperty custom assertion
2024-09-23 16:24:01 +03:00
Faruk AYDIN
cbfb5dd8a6 feat: Introduce toRequireProperty custom assertion 2024-09-23 16:16:13 +03:00
Jakub P.
09b2b7350c test: add applications settings tests 2024-09-20 13:03:22 +02:00
Ömer Faruk Aydın
e146793d32 Merge pull request #2091 from automatisch/aut-1216
feat: use REST API endpoint to update user
2024-09-20 13:32:38 +03:00
Faruk AYDIN
0bad2ead10 chore: Fix typo for useAdminUpdateUser hook 2024-09-20 13:11:46 +03:00
Jakub P
25176884e7 fix: add missing snackbar on user update error 2024-09-20 13:11:43 +03:00
Ali BARIN
266d4cddb0 chore: remove redundant update user mutation 2024-09-20 13:10:42 +03:00
Ali BARIN
6e529a4205 feat: use REST API endpoint to update user 2024-09-20 13:09:46 +03:00
Ömer Faruk Aydın
d2cce278bd Merge pull request #2089 from automatisch/aut-1266
feat: use REST API endpoint to update flow status
2024-09-19 14:10:44 +03:00
Jakub P.
b774a62f8c fix: add missing snackbars on flow status update 2024-09-19 14:03:45 +03:00
Ali BARIN
66c12e1a92 chore: remove redundant update flow status mutation 2024-09-19 14:03:43 +03:00
Ali BARIN
184d748890 feat: use REST API endpoint to update flow status 2024-09-19 14:02:51 +03:00
Ömer Faruk Aydın
8cc732c8d1 Merge pull request #2085 from automatisch/aut-1259
feat: use REST API endpoint to create user
2024-09-19 13:58:13 +03:00
Faruk AYDIN
c688d67e4b feat: Implement admin user serializer test 2024-09-19 13:51:04 +03:00
Jakub P.
3408be2840 test: search for proper snackbar on user create error 2024-09-19 13:50:11 +03:00
Ali BARIN
e126066132 chore: remove redundant create user mutation 2024-09-19 13:50:09 +03:00
Ali BARIN
cfec447d8a feat(web): use REST API endpoint to create user 2024-09-19 13:49:12 +03:00
Ali BARIN
805d1fdd52 fix(useAdminCreateRole): correct query in invalidation 2024-09-19 13:49:12 +03:00
Ali BARIN
9299948072 feat: add admin user serializer with accept invitation url 2024-09-19 13:49:12 +03:00
Ömer Faruk Aydın
b02960a5ec Merge pull request #2083 from automatisch/aut-1257
feat: write and implement REST API endpoint to update step
2024-09-19 13:40:57 +03:00
Ali BARIN
8134b6db6a refactor(update-step): move connection authorization to model 2024-09-19 13:31:33 +03:00
Ali BARIN
cd16a3cc15 chore: remove redundant update step mutation 2024-09-19 13:31:31 +03:00
Ali BARIN
074e7828f3 feat(web): use REST API endpoint to update step 2024-09-19 13:30:45 +03:00
Ali BARIN
5f7d1f9219 feat: write REST API endpoint to update step 2024-09-19 13:30:45 +03:00
Ömer Faruk Aydın
b29c6105a1 Merge pull request #2088 from automatisch/aut-1213
feat: use REST API endpoint to delete step
2024-09-19 13:15:26 +03:00
Ali BARIN
1297f5d43c chore: remove redundant delete step mutation 2024-09-19 13:08:19 +03:00
Ali BARIN
f94a5385d7 feat: use REST API endpoint to delete step 2024-09-19 13:07:37 +03:00
Ömer Faruk Aydın
8ef2000e45 Merge pull request #2087 from automatisch/aut-1261
feat: use REST API endpoint to delete flow
2024-09-19 13:06:48 +03:00
Ali BARIN
30d496076b chore: remove redundant delete flow mutation 2024-09-19 12:37:47 +03:00
Ali BARIN
5de06d4482 feat: use REST API endpoint to delete flow 2024-09-19 12:36:59 +03:00
Ömer Faruk Aydın
97fa983305 Merge pull request #2086 from automatisch/aut-1260
feat: use REST API endpoint to delete current user
2024-09-19 12:34:35 +03:00
Ali BARIN
dccc3c5bc1 chore: remove redundant delete current user mutation 2024-09-19 12:27:28 +03:00
Ali BARIN
ed12465975 feat: use REST API endpoint to delete current user 2024-09-19 12:25:58 +03:00
Ömer Faruk Aydın
b78be222d7 Merge pull request #2084 from automatisch/aut-1228
feat: use REST API endpoint to create flow
2024-09-19 12:22:33 +03:00
Ali BARIN
342990e1bf chore: remove redundant create flow mutation 2024-09-18 14:53:06 +00:00
Ali BARIN
2e5dfdbb0d feat: use REST API endpoint to create flow 2024-09-18 14:51:23 +00:00
Faruk AYDIN
1790ef0ee6 feat: Implement update flow status rest API endpoint 2024-09-17 16:07:12 +03:00
Ömer Faruk Aydın
712a5756e2 Merge pull request #2070 from automatisch/aut-1248
feat: write and implement REST API endpoint to duplicate flow
2024-09-17 14:11:27 +03:00
Faruk AYDIN
bf6ff6b0f7 refactor: Revise duplicate flow controller tests 2024-09-17 13:18:48 +03:00
Faruk AYDIN
c6003b6695 fix: Correct duplicate flow mock name 2024-09-17 13:05:23 +03:00
Ali BARIN
8352540fcb chore: remove redundant duplicate flow mutation 2024-09-17 12:50:54 +03:00
Ali BARIN
22299868fa feat: use REST API endpoint to duplicate flow 2024-09-17 12:49:16 +03:00
Ali BARIN
0d126a8e2b feat: write REST API endpoint to duplicate flow 2024-09-17 12:49:13 +03:00
Ömer Faruk Aydın
776d027dfa Merge pull request #2081 from automatisch/rest-create-user
feat: Implement rest API endpoint for admins to create user
2024-09-17 12:45:08 +03:00
Faruk AYDIN
fa9b6d1006 chore: Mark create user mutation as converted mutation 2024-09-17 11:42:29 +03:00
Faruk AYDIN
89aa7ffc73 feat: Implement rest API endpoint for admins to create user 2024-09-17 11:41:18 +03:00
Faruk AYDIN
fd971449ca chore: Mark update connection as converted mutation 2024-09-16 13:39:44 +03:00
Ömer Faruk Aydın
f7cd57e549 Merge pull request #2080 from automatisch/rest-update-connection
feat: Implement rest API endpoint to update connection
2024-09-16 13:38:43 +03:00
Faruk AYDIN
34aadbfb09 feat: Implement rest API endpoint to update connection 2024-09-16 12:55:39 +03:00
Ömer Faruk Aydın
167195a01c Merge pull request #2077 from automatisch/rest-reset-connection
refactor: Move reset logic to connection model
2024-09-15 14:01:56 +03:00
Faruk AYDIN
b0abf94191 refactor: Move reset logic to connection model 2024-09-15 01:03:17 +03:00
Faruk AYDIN
1009c71e72 chore: Move reset connection to converted mutations 2024-09-15 01:02:58 +03:00
Ömer Faruk Aydın
29b1695159 Merge pull request #2076 from automatisch/rest-delete-flow
feat: Implement rest API endpoint to delete flow
2024-09-15 00:19:45 +03:00
Faruk AYDIN
44f782221f chore: Mark delete flow as converted mutation 2024-09-15 00:11:15 +03:00
Faruk AYDIN
bab25c51d9 feat: Implement rest API endpoint to delete flow 2024-09-14 21:54:10 +03:00
Ömer Faruk Aydın
89277e1665 Merge pull request #2075 from automatisch/converted-mutations
chore: Mark create connection as converted mutation
2024-09-14 00:46:07 +03:00
Faruk AYDIN
1e9c5a1682 chore: Mark create connection as converted mutation 2024-09-14 00:27:09 +03:00
Ömer Faruk Aydın
0c75486c7a Merge pull request #2073 from automatisch/aut-1251
feat: write REST API endpoint to generate auth url
2024-09-12 14:05:39 +03:00
Faruk AYDIN
8c39739880 refactor: Use getApp method for generateAuthUrl method of connection 2024-09-12 13:58:23 +03:00
Ali BARIN
ace1f84094 feat: write REST API endpoint to generate auth url 2024-09-12 13:57:25 +03:00
Ömer Faruk Aydın
2cfd6739ca Merge pull request #2068 from automatisch/aut-1242
feat: write REST API endpoint to create connection
2024-09-12 13:51:37 +03:00
Faruk AYDIN
e0d6f0d653 refactor: Fetch role and permissions for current user in create connection tests 2024-09-12 13:41:16 +03:00
Faruk AYDIN
a4d3b387d0 refactor: Add extra empty line for create connection test block 2024-09-12 13:35:00 +03:00
Ali BARIN
7394aca02d feat: write REST API endpoint to create connection 2024-09-12 13:33:38 +03:00
Ömer Faruk Aydın
e812725182 Merge pull request #2072 from automatisch/aut-1249
feat: write and implement REST API endpoint to register user
2024-09-12 13:26:12 +03:00
Faruk AYDIN
bb76bfcd8b fix: Typo for register user tests 2024-09-12 13:19:01 +03:00
Ali BARIN
6ff6d0a7dc fix(register-user): correct trial expiry date 2024-09-12 13:19:01 +03:00
Ali BARIN
ddc9867058 chore: remove redundant register user mutation 2024-09-12 13:19:01 +03:00
Ali BARIN
ba0d46c6cd feat(web): use REST API endpoint to register user 2024-09-12 13:19:01 +03:00
Ali BARIN
369f04fdbc feat: write REST API endpoint to register user 2024-09-12 13:19:01 +03:00
Ömer Faruk Aydın
09dd8abe23 Merge pull request #2069 from automatisch/rest-update-password
feat: Implement rest API endpoint to update current user password
2024-09-12 13:10:22 +03:00
Faruk AYDIN
8c692758ae chore: Remove redundant only flag for tests 2024-09-12 13:01:49 +03:00
Faruk AYDIN
32d39b88bd feat: Ask for current password while updating user password 2024-09-12 12:54:46 +03:00
Faruk AYDIN
8bd66da511 chore: Move update current user mutation to converted ones 2024-09-11 12:51:32 +03:00
Faruk AYDIN
f86799e494 feat: Implement rest API endpoint to update current user password 2024-09-11 12:51:32 +03:00
Jakub P.
8d10f26f56 test: add pop-up notification test 2024-09-10 17:40:34 +02:00
Ali BARIN
297543f9dd feat(useAuthenticateApp): early exit connection creation at blocked pop-up 2024-09-10 17:40:34 +02:00
Ali BARIN
0c53ee8460 feat(AddAppConnection): show pop-up reminder hint 2024-09-10 17:40:34 +02:00
Ömer Faruk Aydın
862842e3e1 Merge pull request #2064 from automatisch/aut-1237
feat: write and implement REST API endpoint to delete role
2024-09-10 17:19:55 +03:00
Ali BARIN
a4fad360df Merge pull request #2066 from automatisch/aut-1240
feat: write and implement REST API endpoint to create step
2024-09-10 14:31:12 +02:00
Ali BARIN
3ba4c8b3bf chore(EditorNew): remove console log 2024-09-10 10:50:11 +00:00
Ali BARIN
2992236be4 test(create-step): make current and another user explicit 2024-09-10 10:35:19 +00:00
Ali BARIN
82161f028e test(create-step): state permission conditions explicitly 2024-09-10 10:21:42 +00:00
Ali BARIN
1bcaec144b test(create-step): use non-admin user 2024-09-10 10:15:59 +00:00
Ali BARIN
852d4bba0a chore: remove redundant create step mutation 2024-09-10 10:12:59 +00:00
Ali BARIN
af56fa2830 feat(Editor): use REST API endpoint to create step 2024-09-10 10:12:59 +00:00
Ali BARIN
813646e392 feat: write REST API endpoint to create step 2024-09-10 10:12:59 +00:00
Ali BARIN
1ce31eefc6 Merge pull request #2057 from automatisch/aut-1226
feat: write endpoint to update SamlAuthProvidersRoleMappings
2024-09-10 12:09:00 +02:00
Ali BARIN
fdf53844e1 test(update-role-mappings): use name over key 2024-09-10 10:00:40 +00:00
Ali BARIN
be57a82302 test(factories/role): re-create different role if it exists 2024-09-10 09:56:15 +00:00
Ali BARIN
fb82e863e0 test(update-role-mappings): correct the test case name 2024-09-10 09:50:54 +00:00
Ali BARIN
101483409f style(update-role-mappings): add a breakline 2024-09-10 09:50:54 +00:00
Ali BARIN
6fe863eec1 test(update-role-mappings): use explicit remote role name 2024-09-10 09:50:54 +00:00
Ali BARIN
fbb6526aac refactor(update-role-mappings): move logic to model 2024-09-10 09:50:54 +00:00
Ali BARIN
5556aea913 chore: remove upsert-saml-auth-providers-role-mappings mutation 2024-09-10 09:50:54 +00:00
Ali BARIN
95dc5fb849 refactor(RoleMappings): rewrite mutations with REST API endpoints 2024-09-10 09:50:53 +00:00
Ali BARIN
062199d0e3 feat: write endpoints to update SamlAuthProvidersRoleMappings 2024-09-10 09:50:53 +00:00
Ali BARIN
53ce327516 Merge pull request #2060 from automatisch/aut-1233
feat: write and implement REST API endpoint to update role
2024-09-10 11:35:52 +02:00
Ali BARIN
61a1ce57c2 refactor(useAdminDeleteRole): remove redundant invalidateQueries 2024-09-06 09:52:03 +00:00
Jakub P.
687295f772 test: update roles related snackbar locator 2024-09-06 09:51:22 +00:00
Ali BARIN
e5366534ed chore: remove redundant delete role mutation 2024-09-06 09:51:22 +00:00
Ali BARIN
66fe84e126 feat(DeleteRoleButton): use REST API endpoint to delete role 2024-09-06 09:51:22 +00:00
Ali BARIN
0b6c28422c feat: write REST API endpoint to delete role 2024-09-06 09:51:22 +00:00
Ali BARIN
ea667bb6a9 refactor(useAdminUpdateRole): remove redundant invalidateQueries 2024-09-06 09:26:34 +00:00
Ali BARIN
c9ba219de1 chore: remove redundant update role mutation 2024-09-05 15:35:44 +00:00
Ali BARIN
9df1b29d70 feat(EditRole): use REST API endpoint to update role 2024-09-05 15:35:44 +00:00
Ali BARIN
3e34359fa9 feat: write REST API endpoint to update role 2024-09-05 15:35:44 +00:00
Ali BARIN
1818930d2f Merge pull request #2059 from automatisch/aut-1229
feat: write and implement REST API endpoint to create role
2024-09-05 17:35:19 +02:00
Ali BARIN
c03e674001 chore: add isAdmin virtual attribute in role model 2024-09-05 12:15:37 +00:00
Ali BARIN
10a25b82e0 chore(migrations/remove_key_column_in_roles): use arrow functions 2024-09-05 12:13:33 +00:00
Ömer Faruk Aydın
3b2489d738 Merge pull request #2062 from automatisch/fix-trigger-type-problem
fix: Throw error if trigger doesn't have a type
2024-09-05 13:51:37 +03:00
Faruk AYDIN
5c4ca3c84f fix: Add missing pollInverval keys for triggers 2024-09-05 13:40:44 +03:00
Faruk AYDIN
06c4b7ed2e fix: Throw error if trigger doesn't have a type 2024-09-05 13:32:19 +03:00
Ali BARIN
5591f6ccc9 fix(migrations): backfill role.key column in down migration 2024-09-04 11:30:39 +00:00
Ali BARIN
63dfb6947e feat: make role name unique and remove key usage 2024-09-04 11:13:39 +00:00
Ali BARIN
b089069b8e test: use faker in role factory 2024-09-03 14:18:14 +00:00
Ali BARIN
e76a99fd68 chore: remove redundant create role mutation 2024-09-03 13:56:23 +00:00
Ali BARIN
a4ea6c1fad refactor(CreateRole): implement REST API endpoint to create role 2024-09-03 13:56:06 +00:00
Ali BARIN
64ebdce1b2 feat: write migration to make roles unique by key 2024-09-03 13:55:28 +00:00
Ali BARIN
2709491d59 feat: write REST API endpoint to create role 2024-09-03 13:55:18 +00:00
Ömer Faruk Aydın
81beedede6 Merge pull request #2058 from automatisch/rest-delete-current-user
feat: Implement rest API endpoint to remove current user
2024-09-03 16:13:37 +03:00
Faruk AYDIN
33a2386d74 feat: Remove associated access tokens while removing user 2024-09-03 16:03:48 +03:00
Faruk AYDIN
0df5e5283e refactor: Remove user associations before removing user 2024-09-03 16:00:24 +03:00
Faruk AYDIN
560407b972 chore: Comment delete current user mutations as converted 2024-09-03 15:51:55 +03:00
Faruk AYDIN
f8c25ae508 feat: Implement rest API endpoint to remove current user 2024-09-03 15:50:44 +03:00
Ali BARIN
c524277665 Merge pull request #2054 from automatisch/aut-1224
Split UpsertSamlAuthProvider mutation into two endpoints
2024-09-03 11:42:05 +02:00
Ömer Faruk Aydın
a70fb009c7 Merge pull request #2055 from automatisch/rest-test-step
feat: Implement rest api endpoint to test step
2024-09-02 15:22:10 +03:00
Ömer Faruk Aydın
11e67f2ea3 Merge pull request #2056 from automatisch/rest-create-flow
feat: Implement create flow rest API endpoint
2024-09-02 15:21:46 +03:00
Faruk AYDIN
e7118ffe15 chore: Comment create flow mutation as converted 2024-09-02 15:03:59 +03:00
Faruk AYDIN
79e9455244 feat: Implement create flow rest API endpoint 2024-09-02 15:01:20 +03:00
Faruk AYDIN
6ca8e8958a feat: Implement rest api endpoint to test step 2024-09-02 14:00:31 +03:00
Faruk AYDIN
d3dc207166 feat: Extend step serializer with last execution step 2024-09-02 14:00:31 +03:00
Faruk AYDIN
51e200533b feat: Introduce lastExecutionStep relation to Step model 2024-09-02 14:00:31 +03:00
Ali BARIN
b8a25b87d8 Merge pull request #2053 from automatisch/webhook-tests-improvements
test: add wait for response on update flow
2024-09-02 10:04:47 +02:00
Ali BARIN
f4fe0a0d4f refactor(SamlConfiguration): rewrite mutations with REST API endpoints 2024-08-30 11:46:35 +00:00
Ali BARIN
1d4f829d29 chore: remove upsert-saml-auth-provider mutation 2024-08-30 11:46:06 +00:00
Ali BARIN
4afa79fca4 feat: write endpoitns to create and update SamlAuthProvider 2024-08-30 10:56:56 +00:00
Jakub P.
ec22184087 test: add wait for substeps response 2024-08-30 12:53:35 +02:00
Ali BARIN
413f3db5b4 test(update-flow): align user role permissions 2024-08-30 11:07:52 +02:00
Ali BARIN
8ddfcce787 chore: remove update-flow mutation 2024-08-30 11:07:52 +02:00
Ali BARIN
562341adfe refactor(web): rewrite mutation with PATCH /v1/flows/:flowId 2024-08-30 11:07:52 +02:00
Ali BARIN
9519ec53ef feat: write PATCH /v1/flows/:flowId 2024-08-30 11:07:52 +02:00
Ali BARIN
35bada360d refactor(update-config): move logic to config model 2024-08-30 11:07:44 +02:00
Ali BARIN
1f39765efe chore: remove update-config mutation 2024-08-30 11:07:44 +02:00
Ali BARIN
cce5b3b533 refactor(web): rewrite mutation with PATCH /v1/admin/config 2024-08-30 11:07:44 +02:00
Ali BARIN
e77a03b855 feat: write PATCH /v1/admin/config 2024-08-30 11:07:44 +02:00
Ali BARIN
7a54ff212e feat: incorporate NotAuthorized error in error handler 2024-08-30 10:23:28 +02:00
Ali BARIN
01340f4597 chore: remove redundant update-app-config mutation 2024-08-29 15:31:50 +02:00
Ali BARIN
af6fa80d20 test(reset-connection): cover formattedData with DB changes 2024-08-29 11:20:09 +02:00
Ali BARIN
e7474dcb9e refactor(reset-connection): streamline patches 2024-08-29 11:20:09 +02:00
Ali BARIN
0eb906d5df test(reset-connection): remove unnecessary read connection permission 2024-08-29 11:20:09 +02:00
Ali BARIN
c7babf227c feat: write POST /v1/connections/:connectionId/reset 2024-08-29 11:20:09 +02:00
Ali BARIN
d5c81c14f5 fix(FlowAppIcons): fix crash when no steps exist 2024-08-29 10:44:50 +02:00
Ali BARIN
59278378e5 chore: remove delete-connection mutation 2024-08-29 09:26:24 +02:00
Ali BARIN
990e69143d refactor(web): rewrite mutation with DELETE /v1/connections/:connectionId 2024-08-29 09:26:24 +02:00
Ali BARIN
c413ae06dc feat: write DELETE /v1/connections/:connectionId 2024-08-29 09:26:24 +02:00
Ali BARIN
456f8a30cc test(update-app-config): correct test case name 2024-08-29 09:26:16 +02:00
Ali BARIN
19c4561feb chore: remove update-app-config mutation 2024-08-29 09:26:16 +02:00
Ali BARIN
1392fed023 refactor(web): rewrite mutation with PATCH /v1/admin/apps/:appKey/config 2024-08-29 09:26:16 +02:00
Ali BARIN
ca81e14d63 feat: write PATCH /v1/admin/apps/:appKey/config 2024-08-29 09:26:16 +02:00
Ömer Faruk Aydın
8e7a8a02ec Merge pull request #2045 from automatisch/rest-admin-update-user
feat: Implement rest API endpoint to update users for admin
2024-08-28 18:07:24 +03:00
Faruk AYDIN
e45dfa94ed fix: Use updateUserMock instead of updateCurrentUserMock for admin API endpoint 2024-08-28 17:55:06 +03:00
Faruk AYDIN
a153787ae6 chore: Use patch instead of get for update user tests 2024-08-28 17:54:15 +03:00
Faruk AYDIN
cf37c43bc7 chore: Add update user mutation to converted mutations 2024-08-28 17:50:06 +03:00
Faruk AYDIN
d0aa2bca69 feat: Implement rest API endpoint to update users for admin 2024-08-28 17:49:38 +03:00
Ömer Faruk Aydın
7b3811e6ee Merge pull request #2044 from automatisch/rest-update-user
feat: Implement update user rest API endpoint
2024-08-28 17:21:54 +03:00
Faruk AYDIN
4054f551d4 feat: Implement update user rest API endpoint 2024-08-28 17:07:13 +03:00
Ömer Faruk Aydın
4eeda10f3f Merge pull request #2041 from automatisch/remove-redundant-async-handler-wrapper
fix: remove redundant asyncHandler wrapper
2024-08-28 12:38:36 +03:00
Ali BARIN
4231784ed2 fix: remove redundant asyncHandler wrapper 2024-08-28 09:20:30 +00:00
Ali BARIN
61ff6986d3 test(update-auth-client): cover HTTP 422 response 2024-08-28 11:16:46 +02:00
Ali BARIN
09bc0bba1e refactor(web): rewrite mutation with PATCH /v1/admin/apps/:appKey/auth-clients/:appAuthClientId 2024-08-28 11:16:46 +02:00
Ali BARIN
b2bda8479e feat: write PATCH /v1/admin/apps/:appKey/auth-clients/:appAuthClientId 2024-08-28 11:16:46 +02:00
Ömer Faruk Aydın
09b255f99e Merge pull request #2037 from automatisch/refactor-async-errors
refactor: Use express-async-errors instead of express-async-handler
2024-08-28 12:13:22 +03:00
Ali BARIN
0800642a2a refactor(create-config): move unique violation error handling to error-handler 2024-08-28 10:32:06 +02:00
Ali BARIN
48b2b006c0 refactor(web): rewrite mutation with POST /v1/admin/apps/:appKey/config 2024-08-28 10:32:06 +02:00
Ali BARIN
af4c1f08ec feat: write POST /v1/admin/apps/:key/config 2024-08-28 10:32:06 +02:00
Ali BARIN
87b26b6342 fix(app-config): add missing createdAt updatedAt fields in schema 2024-08-28 10:32:06 +02:00
Ali BARIN
706fb8d82f chore(backend): add test:watch script 2024-08-28 10:32:06 +02:00
Faruk AYDIN
f7ca59bd5f refactor: Use express-async-errors instead of express-async-handler 2024-08-28 11:10:31 +03:00
Ali BARIN
6540d0ea53 fix(AdminApplicationCreateAuthClient): correct typo in auth word 2024-08-27 16:34:09 +02:00
Ali BARIN
5995038e21 fix(error-handler): use HTTP 422 for validation errors 2024-08-27 16:34:09 +02:00
Ali BARIN
337ba6ea87 test(create-auth-client): cover error handler 2024-08-27 16:34:09 +02:00
Ali BARIN
04c6183752 refactor(create-auth-client): accept only auth client schema fields 2024-08-27 16:34:09 +02:00
Ali BARIN
0b63922f46 feat: map and expose objection.js errors 2024-08-27 16:34:09 +02:00
Ali BARIN
81c39d7d93 refactor: rewrite create-app-auth-client mutation as REST endpoint 2024-08-27 16:34:09 +02:00
Ömer Faruk Aydın
164d31dfbc Merge pull request #2034 from automatisch/rest-verify-connection-api
feat: Implement rest API endpoint to verify connection
2024-08-27 15:11:53 +03:00
Faruk AYDIN
1a833aad52 feat: Implement rest API endpoint to verify connection 2024-08-27 14:51:28 +03:00
Ömer Faruk Aydın
77246c1fde Merge pull request #2033 from automatisch/refactor-test-connection
Refactor test connection action
2024-08-26 16:35:16 +03:00
Faruk AYDIN
56c08a3587 fix: Solve truncate deadlock problem for test suite 2024-08-26 16:26:29 +03:00
Faruk AYDIN
ce6214dc0f refactor: Rename create test action as test connection 2024-08-26 16:26:03 +03:00
Faruk AYDIN
22002d50ac chore: Comment converted graphQL mutations 2024-08-26 16:01:15 +03:00
Ali BARIN
ab4e94695d Merge pull request #2030 from automatisch/AUT-1132
fix: prevent resetting test data until data is loaded
2024-08-26 11:37:37 +02:00
Ömer Faruk Aydın
47a01cec7e Merge pull request #2032 from automatisch/rest-delete-step
feat: Implement delete step rest API endpoint
2024-08-23 17:44:23 +03:00
Faruk AYDIN
0cf9bc1a32 feat: Implement delete step rest API endpoint 2024-08-23 16:55:44 +03:00
kasia.oczkowska
6552ebcd3c fix: prevent resetting test data until data is loaded 2024-08-23 13:07:05 +01:00
Ali BARIN
5af1d94fc0 Merge pull request #2028 from automatisch/AUT-1183
feat: update step only when data changes
2024-08-22 16:22:16 +02:00
Ali BARIN
a4ec7b3047 feat(compute-parameters): add valueType with parse option and string by default (#2025)
* feat(compute-parameters): add valueType with parse option and string by default

* test(compute-parameters): write tests for valueType with parse and undefined

* fix(compute-parameters): cover valueType = parse in nested objects

* test(compute-parameters): cover valueType = 'parse' in nested non-primitives

* fix(compute-parameters): mark fields optional
2024-08-22 16:20:18 +02:00
Ali BARIN
66f5003d91 Merge pull request #2027 from automatisch/harden-webhook-tests
test: harden webhook tests
2024-08-22 16:19:51 +02:00
kasia.oczkowska
0c754e4b4b feat: update step only when data changes 2024-08-22 11:57:25 +01:00
Jakub P.
e008087c4a test: harden webhook tests 2024-08-22 12:05:52 +02:00
Ali BARIN
53f63996bd Merge pull request #2024 from kuba618/AUT-1186
test: add simple webhook flow test
2024-08-22 11:16:22 +02:00
Ali BARIN
4fedf77991 Merge pull request #2026 from automatisch/aut-1187
refactor(compute-parameters): rewrite logic
2024-08-22 10:18:49 +02:00
Ali BARIN
34331d8763 refactor(compute-parameters) rename functions and variables 2024-08-21 11:35:09 +00:00
Jakub P.
2c21b7762c test: add simple webhook flow test 2024-08-20 19:17:08 +02:00
Ömer Faruk Aydın
7f9c2b687f Merge pull request #2022 from automatisch/use-objects-as-variables
feat(PowerInput): support whole objects as variables
2024-08-19 14:21:52 +03:00
Ali BARIN
b452ed648c feat(PowerInput): support whole objects as variables 2024-08-16 13:22:31 +00:00
Rıdvan Akca
98e4b843ea feat(clickup/actions): add create task and find task by id (#1615)
* feat(clickup): add create task action

* feat(clickup): add find task by id action

* fix(clickup): send optional parameters only when provided

---------

Co-authored-by: Ali BARIN <ali.barin53@gmail.com>
2024-08-15 15:59:58 +02:00
Ali BARIN
da2884d53c Merge pull request #1594 from automatisch/AUT-666
feat(clickup): add clickup integration
2024-08-15 14:44:11 +02:00
Ali BARIN
7f2937400a fix(clickup/create-list): send priority and due date only when provided 2024-08-15 12:07:40 +00:00
Ali BARIN
a3a4a8e431 fix(clickup/new-folders): correct internal ID 2024-08-15 12:07:40 +00:00
Ali BARIN
3249c954d3 fix(clickup/new-lists): correct internal ID 2024-08-15 12:07:40 +00:00
Ali BARIN
7395d2a74e fix(clickup/new-tasks): correct internal ID 2024-08-15 12:07:40 +00:00
Ali BARIN
2060b7b49d feat(clickup/updated-task): mark whatChanged optional 2024-08-15 12:07:40 +00:00
Rıdvan Akca
d263726c19 feat(clickup): add create list action 2024-08-15 12:07:40 +00:00
Rıdvan Akca
7e3325e959 feat(clickup): add create folder action 2024-08-15 12:07:40 +00:00
Rıdvan Akca
ec075f05c5 feat(clickup): add task changes trigger 2024-08-15 12:07:40 +00:00
Rıdvan Akca
200e483574 feat(clickup): add new tasks trigger 2024-08-15 12:07:40 +00:00
Rıdvan Akca
6c11bfe93d feat(clickup): add new lists trigger 2024-08-15 12:07:40 +00:00
Rıdvan Akca
b4cc7f4d81 feat(clickup): add new folders trigger 2024-08-15 12:07:38 +00:00
Rıdvan Akca
26fc63c52c feat(clickup): add clickup integration 2024-08-15 12:07:24 +00:00
Ali BARIN
116bf59b68 Merge pull request #1577 from automatisch/AUT-641
feat(jotform): add jotform integration
2024-08-14 16:23:42 +02:00
Ali BARIN
566f9dd5cc fix(jotform): use right custom API URL 2024-08-14 14:14:55 +00:00
Ali BARIN
ec740c07fa feat(jotform/auth): show default API URL 2024-08-14 14:14:20 +00:00
Ali BARIN
7506bf186b feat(jotform): prettify screen name 2024-08-14 14:14:20 +00:00
Ali BARIN
27c36b644d fix(jotform/list-forms): fix pagination 2024-08-14 14:14:20 +00:00
Ali BARIN
ed2b1029f6 docs(jotform/connection): incorporate API URL field 2024-08-14 14:14:20 +00:00
Rıdvan Akca
530a920517 feat(jotform): add new submissions trigger 2024-08-14 14:14:19 +00:00
Rıdvan Akca
42f8e635ed feat(jotform): add jotform integration 2024-08-14 14:13:57 +00:00
Ali BARIN
7ace67f906 Merge pull request #1571 from automatisch/AUT-632
feat(mailchimp): add mailchimp integration
2024-08-13 15:29:39 +02:00
Ali BARIN
1b437778dc docs: sort apps alphabetically 2024-08-13 13:16:54 +00:00
Ali BARIN
43b0c65aab fix(mailchimp): update mailchimp triggers and dynamic data 2024-08-13 13:16:39 +00:00
Rıdvan Akca
3f6a319ebe feat(mailchimp): add send campaign action 2024-08-13 13:16:39 +00:00
Rıdvan Akca
4cbd342e17 feat(mailchimp): add create campaign action 2024-08-13 13:16:39 +00:00
Rıdvan Akca
273f04128c feat(mailchimp): add email opened trigger 2024-08-13 13:16:39 +00:00
Rıdvan Akca
9a5cef08d6 feat(mailchimp): add new unsubscribers trigger 2024-08-13 13:16:39 +00:00
Rıdvan Akca
bbecfdb718 feat(mailchimp): add new subscribers trigger 2024-08-13 13:16:36 +00:00
Rıdvan Akca
cbed79fbf1 feat(mailchimp): add mailchimp integration 2024-08-13 13:16:09 +00:00
Ali BARIN
c4cbc024e6 Merge pull request #2019 from automatisch/aut-1093
fix(duplicate-flow): let users duplicate flows they have access to
2024-08-09 17:23:10 +02:00
Ali BARIN
2db8dbd5a3 Merge pull request #2018 from automatisch/AUT-1090
fix(carbone/add-template): correct typo in template
2024-08-09 11:37:49 +02:00
Ömer Faruk Aydın
13b995c9f2 Merge pull request #2017 from automatisch/AUT-1094
fix(update-flow-status): throw at incomplete trigger step
2024-08-09 11:06:14 +02:00
Ali BARIN
a0944193b6 Merge pull request #1627 from automatisch/dependabot/npm_and_yarn/ip-1.1.9
chore(deps): Bump ip from 1.1.5 to 1.1.9
2024-08-08 13:41:38 +02:00
Ali BARIN
c7f343020a Merge pull request #2016 from automatisch/dependabot/npm_and_yarn/fast-xml-parser-4.4.1
chore(deps): bump fast-xml-parser from 4.2.5 to 4.4.1
2024-08-08 13:38:32 +02:00
Ali BARIN
7a48ccc4f4 Merge pull request #1938 from automatisch/dependabot/npm_and_yarn/ws-7.5.10
chore(deps): bump ws from 7.5.6 to 7.5.10
2024-08-08 13:38:20 +02:00
Ali BARIN
cc1218b7a3 Merge pull request #1937 from automatisch/dependabot/npm_and_yarn/braces-3.0.3
chore(deps): bump braces from 3.0.2 to 3.0.3
2024-08-08 13:38:01 +02:00
Ali BARIN
385e640a92 Merge pull request #2014 from automatisch/AUT-1141
feat: make role input required in CreateUser
2024-08-08 12:08:23 +02:00
Ali BARIN
b40a59cbef Merge pull request #1990 from kuba618/AUT-1137
fix: improve paths for windows os
2024-08-08 12:08:05 +02:00
Ali BARIN
1a45ce5ea4 fix(duplicate-flow): let users duplicate flows they have access to 2024-08-08 09:23:54 +00:00
Ali BARIN
ab05048409 fix(carbone/add-template): correct typo in template 2024-08-08 08:56:58 +00:00
Ali BARIN
47aabbe9c5 fix(update-flow-status): throw at incomplete trigger step 2024-08-08 08:48:27 +00:00
Ali BARIN
58cd2c522e Merge pull request #1461 from automatisch/AUT-537
feat(google-sheets): add find worksheet action
2024-08-07 16:22:06 +02:00
Rıdvan Akca
1fe0cd9f84 feat(google-sheets): add find worksheet action 2024-08-07 14:21:51 +00:00
Ali BARIN
eeee1ba1a3 Merge pull request #1565 from automatisch/AUT-618
feat(mailerlite): add campaign sent trigger
2024-08-07 14:31:27 +02:00
Ali BARIN
3cb8880c5c Merge pull request #1564 from automatisch/AUT-617
feat(mailerlite): add subscriber unsubscribed trigger
2024-08-07 14:31:19 +02:00
Ali BARIN
80cec86225 Merge pull request #1563 from automatisch/AUT-616
feat(mailerlite): add subscriber created trigger
2024-08-07 14:31:12 +02:00
Ali BARIN
cc65ed8fb0 Merge pull request #1562 from automatisch/AUT-615
feat(mailerlite): add spam complaint trigger
2024-08-07 14:31:05 +02:00
Ali BARIN
0c3f1f4a5d Merge pull request #1561 from automatisch/AUT-614
feat(mailerlite): add mailerlite integration
2024-08-07 14:30:53 +02:00
dependabot[bot]
76375941ca chore(deps): bump fast-xml-parser from 4.2.5 to 4.4.1
Bumps [fast-xml-parser](https://github.com/NaturalIntelligence/fast-xml-parser) from 4.2.5 to 4.4.1.
- [Release notes](https://github.com/NaturalIntelligence/fast-xml-parser/releases)
- [Changelog](https://github.com/NaturalIntelligence/fast-xml-parser/blob/master/CHANGELOG.md)
- [Commits](https://github.com/NaturalIntelligence/fast-xml-parser/compare/v4.2.5...v4.4.1)

---
updated-dependencies:
- dependency-name: fast-xml-parser
  dependency-type: direct:production
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-08-07 12:22:09 +00:00
Ali BARIN
d2a0415def Merge pull request #2015 from automatisch/dependabot/npm_and_yarn/msgpackr-1.11.0
chore(deps): bump msgpackr from 1.5.6 to 1.11.0
2024-08-07 14:21:07 +02:00
dependabot[bot]
815c0834b2 chore(deps): bump msgpackr from 1.5.6 to 1.11.0
Bumps [msgpackr](https://github.com/kriszyp/msgpackr) from 1.5.6 to 1.11.0.
- [Release notes](https://github.com/kriszyp/msgpackr/releases)
- [Commits](https://github.com/kriszyp/msgpackr/commits)

---
updated-dependencies:
- dependency-name: msgpackr
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-08-07 12:15:36 +00:00
kasia.oczkowska
7e1e1e2524 feat: make role input required in CreateUser 2024-08-07 12:41:45 +01:00
Ali BARIN
957d2793cb Merge pull request #2009 from automatisch/test-compute-parameters
test(compute-parameters): write tests for different type of values
2024-08-07 10:57:50 +02:00
Ömer Faruk Aydın
6ab86b7574 Merge pull request #2010 from automatisch/aut-1098
fix: mark steps incomplete when their parameters change
2024-08-07 10:07:00 +02:00
Ömer Faruk Aydın
1c75b7226d Merge pull request #1999 from automatisch/redis-sentinel
feat: support redis sentinel
2024-08-07 10:00:24 +02:00
Ömer Faruk Aydın
db22d8e2c9 Merge pull request #2012 from automatisch/aut-1154
feat(http-request/custom-request): expose response headers and status
2024-08-07 10:00:07 +02:00
Ömer Faruk Aydın
7838b9609c Merge pull request #2013 from automatisch/aut-1089
fix(github/create-issue): mark repo field as required
2024-08-07 09:59:51 +02:00
Ali BARIN
2a1a0421b6 fix(github/create-issue): mark repo field as required 2024-08-06 14:42:21 +00:00
Ali BARIN
125ea0457e feat(http-request/custom-request): expose response headers and status 2024-08-06 13:35:42 +00:00
Ali BARIN
c1f5f0632b fix: mark steps incomplete when their parameters change 2024-08-05 16:34:09 +00:00
Ömer Faruk Aydın
8c84ab29c6 Merge pull request #1998 from automatisch/code-execution-app
feat(code/run-javascript): run JS code and introduce monaco editor
2024-08-05 17:32:18 +02:00
Ali BARIN
2767af11b2 fix: move nodemon config to separate package.json property 2024-08-05 14:00:02 +00:00
Ali BARIN
59bbc4c182 fix: remove single quotes in package scripts 2024-08-05 14:00:02 +00:00
Jakub P.
a755ee8dc1 fix: improve paths for windows os 2024-08-05 14:00:02 +00:00
Ali BARIN
4b1e66add3 feat(code/run-javascript): omit empty key entries 2024-08-05 13:46:41 +00:00
Faruk AYDIN
4f6727810b feat(code): Improve the default code by adding usage comments 2024-08-05 13:46:41 +00:00
Faruk AYDIN
60fdfc2b48 chore(code): Make font size of the code editor a bit bigger 2024-08-05 13:46:41 +00:00
Faruk AYDIN
268d8c8b7d feat(code): Add linebreaks to the default code value 2024-08-05 13:46:41 +00:00
Ali BARIN
49d4071928 feat(code/run-javascript): run async code and introduce monaco editor 2024-08-05 13:46:41 +00:00
Faruk AYDIN
c99b9dbe0a feat: Implement draft version of the code execution app 2024-08-05 13:46:41 +00:00
Ali BARIN
09d3a06b27 Merge pull request #2005 from automatisch/AUT-1065
fix: in FlowStep wrap in Form component only controlled inputs
2024-08-05 15:45:06 +02:00
Ali BARIN
0d49bc003f Merge pull request #2004 from automatisch/fix-dockerfile
fix: Adjust dockerfile env syntax to new dsl
2024-08-05 15:43:41 +02:00
Ali BARIN
86a5569bf7 test(compute-parameters): write tests for different type of values 2024-08-05 13:35:16 +00:00
Ali BARIN
ad9fe7dec6 Merge pull request #2007 from automatisch/release/v0.13.1
Update version to 0.13.1
2024-08-05 12:28:16 +02:00
Ali BARIN
24bf07e068 Update version to 0.13.1 2024-08-02 15:54:26 +02:00
Ali BARIN
bae234827f Merge pull request #2006 from automatisch/use-unparsed-numbers-in-compute-parameters
fix(compute-parameters): use unparsed numbers
2024-08-02 15:42:21 +02:00
Ali BARIN
81c698f45b fix(compute-parameters): use unparsed numbers 2024-08-02 12:22:03 +00:00
Ali BARIN
c9fecec575 Merge pull request #2002 from automatisch/aut-1255
feat: expose installationCompleted in GET /v1/automatisch/info
2024-08-02 11:46:48 +02:00
Ali BARIN
2f42dfdc51 Merge pull request #2001 from automatisch/remove-pwa
feat: remove PWA
2024-08-02 11:16:44 +02:00
kasia.oczkowska
5bac68b0de fix: in FlowStep wrap in Form component only controlled inputs 2024-08-02 09:31:36 +01:00
Ali BARIN
5afd500c26 Merge pull request #2003 from automatisch/release/v0.13.0
Update version to 0.13.0
2024-08-01 18:30:55 +02:00
Faruk AYDIN
03f3e5f6ab fix: Adjust dockerfile env syntax to new dsl 2024-08-01 18:15:00 +02:00
Faruk AYDIN
50d91405a9 Update version to 0.13.0 2024-08-01 17:32:19 +02:00
Ali BARIN
69eed65c9b Merge pull request #1991 from automatisch/AUT-1064
fix: introduce fix for overflowing inputs with long parameters names
2024-08-01 16:22:17 +02:00
kasia.oczkowska
f1355cd0ab fix: introduce fix for overflowing inputs with long parameters names 2024-08-01 15:10:41 +01:00
Ali BARIN
cc1a924c8b feat: use installationCompleted from useAutomatischInfo hook 2024-08-01 13:38:06 +00:00
Ali BARIN
02005a3f09 feat: expose installationCompleted in GET /v1/automatisch/info 2024-08-01 13:02:21 +00:00
Ali BARIN
d9219a5a48 feat: remove PWA 2024-08-01 12:28:23 +00:00
Ali BARIN
dffbdf544c Merge pull request #2000 from automatisch/AUT-1131
fix: prevent removing last filter criteria
2024-08-01 10:50:42 +02:00
kasia.oczkowska
e2dbf1a215 fix: prevent removing last filter criteria 2024-08-01 08:40:54 +01:00
Ali BARIN
66f9cb8d25 feat: support redis sentinel 2024-07-30 18:31:57 +00:00
Ömer Faruk Aydın
c1396b97f0 Merge pull request #1996 from automatisch/support-arrays-in-flows
feat: support arrays in flows
2024-07-30 13:26:56 +02:00
Ali BARIN
920a711c00 feat: support arrays in flows 2024-07-29 11:33:36 +00:00
Ömer Faruk Aydın
02a872a376 Merge pull request #1995 from automatisch/support-async-before-request
feat(http-client): support async beforeRequest interceptors
2024-07-27 18:41:40 +02:00
Ali BARIN
be4493710f feat(http-client): support async beforeRequest interceptors 2024-07-27 13:05:04 +00:00
Ömer Faruk Aydın
52c0c5e0c5 Merge pull request #1992 from automatisch/fix-proxy-configuration
fix(axios): update order of interceptors
2024-07-26 20:16:38 +02:00
Ali BARIN
4c639f170e fix(axios): update order of interceptors 2024-07-26 13:52:41 +00:00
Ali BARIN
509a414151 Merge pull request #1986 from automatisch/aut-1126
refactor(http-client): inherit interceptors from parent instance
2024-07-26 11:45:02 +02:00
Ali BARIN
8f3c793a69 Merge pull request #1988 from automatisch/aut-1130
feat(formatter/text): add regex support in replace transfomer
2024-07-26 11:21:12 +02:00
Ömer Faruk Aydın
a2dd9cf1b8 Merge pull request #1989 from automatisch/fix-typos-in-appwrite-docs
docs(appwrite): fix typos
2024-07-26 10:14:15 +02:00
Ömer Faruk Aydın
42285a5879 Merge pull request #1987 from automatisch/aut-1129
fix(webhook): add missing filter coverage
2024-07-26 10:13:40 +02:00
Ali BARIN
5d63fce6f0 docs(appwrite): fix typos 2024-07-25 09:18:25 +00:00
Ali BARIN
46a4c8faec feat(formatter/text): add regex support in replace transfomer 2024-07-24 17:34:14 +00:00
Ali BARIN
ba14481151 fix(webhook): add missing filter coverage 2024-07-24 16:01:28 +00:00
Ali BARIN
bdd8da98c4 Merge pull request #1985 from automatisch/aut-1061
feat: use dynamic DOCS URL
2024-07-24 14:20:03 +02:00
Ali BARIN
5a4207414d refactor(http-client): inherit interceptors from parent instance 2024-07-23 15:01:12 +00:00
Ali BARIN
730fdd32b1 feat: use dynamic DOCS URL 2024-07-23 09:48:20 +00:00
Ali BARIN
aa7f6694fc Merge pull request #1974 from kuba618/AUT-1115
test: add admin account setup tests
2024-07-22 17:03:55 +02:00
Jakub P.
8fba6704df Merge remote-tracking branch 'upstream/main' into AUT-1115 2024-07-22 16:32:09 +02:00
Jakub P.
37d02eba02 test: add teardown dependency to clean db after tests 2024-07-22 15:42:21 +02:00
Ali BARIN
75a87cf070 Merge pull request #1965 from automatisch/add-cryptography-app
feat: add cryptography app with createHmac and createSignature actions
2024-07-22 14:22:14 +02:00
Ömer Faruk Aydın
8acdc5853d Merge pull request #1978 from automatisch/AUT-1105
feat: refactor delete user mutation with the REST API endpoint
2024-07-22 14:11:35 +02:00
kasia.oczkowska
1aa1f441b3 feat: refactor delete user mutation with the REST API endpoint 2024-07-22 13:27:57 +02:00
Ömer Faruk Aydın
0e26032ac3 Merge pull request #1977 from automatisch/add-create-uuid-text-formatter
feat(formatter/text): add create uuid action
2024-07-22 13:24:14 +02:00
Ömer Faruk Aydın
871f25c6d9 Merge pull request #1975 from automatisch/add-get-current-timestamp
feat(formatter/date-time): add get current timestamp action
2024-07-22 13:23:06 +02:00
Ömer Faruk Aydın
d8199e7ba7 Merge pull request #1983 from automatisch/strict-installation-redirection
fix: safeguard redirect to /installation
2024-07-22 13:20:25 +02:00
Ali BARIN
52b938eabe Merge pull request #1980 from kuba618/AUT-1119
test: add removed user invitation test
2024-07-19 11:42:29 +02:00
Ali BARIN
94fddf3d9b Merge pull request #1979 from kuba618/AUT-1118
chore: add linter to tests
2024-07-19 11:41:34 +02:00
Ali BARIN
02c98c1ece Merge pull request #1982 from automatisch/remove-deleted-connection
fix(AppConnectionRow): invalidate app connections upon deletion
2024-07-19 11:38:31 +02:00
Ali BARIN
6ba77667e9 Merge pull request #1981 from automatisch/AUT-1103
feat: use create access token api endpoint instead of login mutation
2024-07-19 11:38:17 +02:00
Ali BARIN
e201a5b806 fix: safeguard redirect to /installation 2024-07-19 09:31:58 +00:00
Ali BARIN
adac68c407 fix(AppConnectionRow): invalidate app connections upon deletion 2024-07-19 08:43:49 +00:00
kasia.oczkowska
d051275e54 feat: use create access token api endpoint instead of login mutation 2024-07-18 13:05:59 +01:00
Jakub P.
2afc00364a test: add removed user invitation test 2024-07-18 12:40:48 +02:00
Jakub P.
0a1461231b chore: add linter to tests 2024-07-18 12:39:04 +02:00
Jakub P.
129327f40d test: add admin account setup tests 2024-07-18 12:29:35 +02:00
Ali BARIN
58819aad94 feat(cryptography/actions): add output encoding field 2024-07-17 18:44:00 +02:00
Ali BARIN
949a2543f5 feat(cryptography): add create signature action 2024-07-17 18:44:00 +02:00
Ali BARIN
0f9d732667 feat: add cryptography app with createHmac action 2024-07-17 18:44:00 +02:00
Ali BARIN
b7df175950 feat(formatter/text): add create uuid action 2024-07-17 16:07:23 +00:00
Ali BARIN
4f7ce9874f feat(formatter/date-time): add get current timestamp action 2024-07-17 15:56:26 +00:00
Ömer Faruk Aydın
f63cc80383 Merge pull request #1973 from automatisch/AUT-1109
feat: refactor reset password mutation with the REST API endpoint
2024-07-17 17:24:55 +02:00
kasia.oczkowska
dd0a1328e8 feat: refactor reset password mutation with the REST API endpoint 2024-07-17 15:05:46 +01:00
Ali BARIN
27610c002c Merge pull request #1972 from automatisch/AUT-1107
feat: refactor forgot password mutation with the REST API endpoint
2024-07-17 15:32:37 +02:00
kasia.oczkowska
46ec9b5229 feat: refactor forgot password mutation with the REST API endpoint 2024-07-17 13:23:45 +01:00
Ali BARIN
778559d537 Merge pull request #1971 from automatisch/fix-email-worker
fix(email): correct sandbox safeguard check
2024-07-17 13:53:12 +02:00
Ali BARIN
1e64b8a903 fix(email): correct sandbox safeguard check 2024-07-17 10:25:01 +02:00
Ali BARIN
0e000693d6 Merge pull request #1970 from automatisch/rest-reset-password
feat: Implement reset password rest API endpoint
2024-07-16 16:45:20 +02:00
Faruk AYDIN
cf9e09ea7a feat: Implement reset password rest API endpoint 2024-07-16 16:22:10 +02:00
Ali BARIN
eac2f729a5 Merge pull request #1969 from automatisch/fix-migrations-rollback
fix: Remove step constraint down migration
2024-07-16 16:04:07 +02:00
Faruk AYDIN
9578cd27dd fix: Remove step contraint down migration 2024-07-16 15:52:24 +02:00
Ali BARIN
9304ffffc9 Merge pull request #1968 from automatisch/rest-forgot-password
feat: Implement forgot password rest API endpoint
2024-07-16 15:52:07 +02:00
Faruk AYDIN
3fd628cb05 feat: Implement forgot password rest API endpoint 2024-07-16 15:43:15 +02:00
Ömer Faruk Aydın
fe68b70bd9 Merge pull request #1967 from automatisch/rest-delete-user
feat: Implement rest API endpoint to delete user
2024-07-16 15:42:01 +02:00
Faruk AYDIN
ec2863d218 feat: Implement rest API endpoint to delete user 2024-07-16 15:18:01 +02:00
Ali BARIN
dc56e7f883 Merge pull request #1966 from kuba618/AUT-1099
test: add accept invitation invalid token tests
2024-07-16 13:53:32 +02:00
Jakub P.
2f1f537e00 test: add accept invitation invalid token tests 2024-07-16 12:08:50 +02:00
Ali BARIN
b7b3a3025b Merge pull request #1964 from automatisch/AUT-1095
feat: create onboarding UX flow
2024-07-15 16:11:25 +02:00
Ali BARIN
edb9526538 feat(InstallationForm): rephrase submit button 2024-07-15 12:24:35 +00:00
Ali BARIN
62117ece06 Merge pull request #1939 from automatisch/aut-1035
feat(salesforce): add find partially matching record action
2024-07-15 14:01:21 +02:00
Ali BARIN
913a2a0ac1 Merge pull request #1953 from automatisch/add-security-icon-in-additional-drawer
feat(Layout): support security icon in additional drawer link
2024-07-15 14:01:10 +02:00
Ali BARIN
92ec3d07a3 feat(salesforce/find-partially-matching-record): sanitize user inputs 2024-07-15 09:04:16 +00:00
Ali BARIN
67ee7899fd feat(salesforce): add find partially matching record action 2024-07-15 09:04:16 +00:00
Ali BARIN
b0ceb3fe7e feat(InstallationForm): use thinner font weight in success alert 2024-07-15 09:00:13 +00:00
kasia.oczkowska
ba6b4c6854 feat: create onboarding UX flow 2024-07-12 14:55:30 +01:00
Ali BARIN
61e90aed60 chore: upgrade @playwright/test with 1.45.1 2024-07-12 13:40:51 +00:00
Ali BARIN
45b7a399f2 feat(Layout): support security icon in additional drawer link 2024-07-12 13:40:22 +00:00
Ali BARIN
673ed25598 Merge pull request #1963 from automatisch/add-parse-stringified-json-action
feat(formatter): add parse stringified json action
2024-07-12 12:44:49 +02:00
Ali BARIN
14fc460174 Merge pull request #1962 from automatisch/missing-power-input-label
fix: bring back the label for PowerInput
2024-07-12 12:38:54 +02:00
Ali BARIN
211b4537f9 Merge pull request #1961 from automatisch/fix-undefined-datastore-entry-throwing
fix(datastore): stop undefined datastore entry throwing
2024-07-12 12:37:53 +02:00
Ali BARIN
a683e059aa feat(formatter): add parse stringified json action 2024-07-12 10:37:16 +00:00
kasia.oczkowska
d2d2cea567 fix: return Item in renderItemFactory 2024-07-12 11:31:43 +01:00
kasia.oczkowska
e5bda65a35 fix: bring back the label for PowerInput 2024-07-12 11:27:52 +01:00
Ali BARIN
6aaef9df4b fix(datastore): stop undefined datastore entry throwing 2024-07-12 08:58:39 +00:00
Ali BARIN
7fba4d72b0 Merge pull request #1958 from automatisch/AUT-1091
fix: introduce fix for logo being stretched
2024-07-11 15:35:48 +02:00
Ali BARIN
a6dc2ed4ba Merge pull request #1956 from automatisch/AUT-1007
feat: limit vertical scroll to a reasonable boundary
2024-07-11 15:34:04 +02:00
Ali BARIN
20a40eb2f6 Merge pull request #1959 from automatisch/AUT-1088
fix: introduce fix for undefined id on dialog close when adding a connection
2024-07-11 13:23:59 +02:00
kasia.oczkowska
ec4ac9d075 fix: introduce fix for undefined id on dialog close when adding a connection 2024-07-11 12:15:51 +01:00
kasia.oczkowska
b2c14f4226 fix: introduce fix for logo being stretched 2024-07-11 11:28:45 +01:00
Ömer Faruk Aydın
099dfbd0b0 Merge pull request #1954 from automatisch/user-invitation
feat: Implement user invitation functionality
2024-07-11 12:18:20 +02:00
Ali BARIN
a9f5736c12 test: cover accept invitation flow 2024-07-11 11:38:23 +02:00
Ali BARIN
2bd4dd3ab0 test(e2e-tests): cover password field removal 2024-07-11 11:38:23 +02:00
Ali BARIN
186850c256 test: cover user status in user serializer 2024-07-11 11:38:23 +02:00
Ali BARIN
9922033d33 feat: add accept-invitation page 2024-07-11 11:38:23 +02:00
Faruk AYDIN
3c3e6e4144 feat: Implement user invitation backend functionality 2024-07-11 11:38:23 +02:00
Faruk AYDIN
0e4ac3b7f3 feat: Add status column to user model 2024-07-11 11:38:23 +02:00
Ali BARIN
c9813e0316 Merge pull request #1950 from automatisch/AUT-1074
fix: fix error for only Create role when publishing flow
2024-07-11 11:02:08 +02:00
Ali BARIN
577c0cfab9 Merge pull request #1957 from automatisch/AUT-677
feat: add missing propTypes
2024-07-11 10:54:17 +02:00
kasia.oczkowska
0966f9d715 feat: disable publish button when no flow or permissions 2024-07-11 09:39:31 +01:00
kasia.oczkowska
3f5df118a0 feat: add missing propTypes 2024-07-11 08:54:32 +01:00
kasia.oczkowska
e05c1b26f1 feat: limit vertical scroll to a reasonable boundary 2024-07-10 14:52:07 +01:00
Ömer Faruk Aydın
725b38c697 Merge pull request #1955 from automatisch/repository-structure
docs: Adjust repository to the new packages structure
2024-07-09 13:14:27 +02:00
Faruk AYDIN
402a0fdf3b docs: Adjust repository to the new packages structure 2024-07-08 21:11:25 +02:00
Ali BARIN
078364ffa1 Merge pull request #1951 from automatisch/AUT-1069
fix: set default locale for luxon's  DateTime
2024-07-08 15:19:14 +02:00
kasia.oczkowska
f64d5ec4fc fix: set default locale for luxon's DateTime 2024-07-04 15:14:28 +01:00
kasia.oczkowska
0666174501 fix: fix error for only Create role when publishing flow 2024-07-04 14:08:35 +01:00
Ali BARIN
12194a50e1 Merge pull request #1948 from automatisch/AUT-1076
feat: block access to flow creation for users without permissions
2024-07-04 10:44:15 +02:00
kasia.oczkowska
82ee592699 feat: block access to flow creation for users without permissions 2024-07-04 08:57:03 +01:00
Ali BARIN
1b4fb2ce6e Merge pull request #1947 from automatisch/AUT-1070
feat: use try-catch block for updating flow status
2024-07-03 15:37:02 +02:00
kasia.oczkowska
ebea8d12d1 feat: use try-catch block for updating flow status 2024-07-03 14:04:11 +01:00
Ali BARIN
f842dd77df Merge pull request #1945 from automatisch/remove-access-tokens-in-perm-user-deletion
fix: delete access token before hard deleting user
2024-07-03 13:16:20 +02:00
Ali BARIN
a6ec7a6c99 Merge pull request #1944 from automatisch/fix-reset-password
fix(user): make resetPasswordToken relevant fields nullable
2024-07-03 13:16:12 +02:00
Ali BARIN
369c72282c fix: delete access token before hard deleting user 2024-07-01 11:41:11 +00:00
Ali BARIN
6f30c1a509 fix(user): make resetPasswordToken relevant fields nullable 2024-07-01 10:26:17 +00:00
Ali BARIN
abfd1116c7 Merge pull request #1941 from automatisch/AUT-1038
fix: persist value in ControlledCustomAutocomplete when it depends on other fields
2024-06-21 14:32:09 +02:00
kasia.oczkowska
017854955d fix: persist value in ControlledCustomAutocomplete when it depends on other fields 2024-06-21 10:37:39 +01:00
Ali BARIN
1405cddea1 Merge pull request #1940 from automatisch/AUT-1039
Revert "feat: persist parameters values in FlowSubstep (#1505)"
2024-06-20 15:57:47 +02:00
kasia.oczkowska
00dd3164c9 Revert "feat: persist parameters values in FlowSubstep (#1505)"
This reverts commit 017a881494.
2024-06-20 14:48:51 +01:00
Ali BARIN
d5cbc0f611 Merge pull request #1578 from automatisch/AUT-620
feat: improve UI display depending on user permissions
2024-06-20 12:04:31 +02:00
kasia.oczkowska
5d2e9ccc67 feat: improve UI display depending on user persmissions 2024-06-20 09:52:00 +00:00
kattoczko
017a881494 feat: persist parameters values in FlowSubstep (#1505)
* feat: persist parameters values in FlowSubstep

* feat: add missing import

* feat: use usePrevious hook
2024-06-20 11:13:45 +02:00
Alex Maslakov
52994970e6 fix(DynamicField): display long variables better
* Fix issue #1933

Problem with rendering in DynamicField component with long variables #1933

* Fix issue #1933 (2)

* Fix issue #1933 (3)
2024-06-20 09:55:08 +02:00
Ali BARIN
ebae629e5c Merge pull request #1931 from automatisch/AUT-1010
feat: improve nodes and edges state update
2024-06-19 16:01:24 +02:00
dependabot[bot]
0dea5150a0 chore(deps): bump ws from 7.5.6 to 7.5.10
Bumps [ws](https://github.com/websockets/ws) from 7.5.6 to 7.5.10.
- [Release notes](https://github.com/websockets/ws/releases)
- [Commits](https://github.com/websockets/ws/compare/7.5.6...7.5.10)

---
updated-dependencies:
- dependency-name: ws
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-06-18 08:56:00 +00:00
dependabot[bot]
c99e8d270d chore(deps): bump braces from 3.0.2 to 3.0.3
Bumps [braces](https://github.com/micromatch/braces) from 3.0.2 to 3.0.3.
- [Changelog](https://github.com/micromatch/braces/blob/master/CHANGELOG.md)
- [Commits](https://github.com/micromatch/braces/compare/3.0.2...3.0.3)

---
updated-dependencies:
- dependency-name: braces
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-06-17 12:51:03 +00:00
kasia.oczkowska
4d79220b0c refactor: fix spelling and wording errors 2024-06-14 12:39:42 +01:00
kasia.oczkowska
96fba7fbb8 feat: improve nodes and edges state update 2024-06-14 12:39:42 +01:00
Ali BARIN
e0d610071d Merge pull request #1917 from automatisch/AUT-1009
feat: hide react-flow attribution
2024-06-05 15:35:09 +02:00
kasia.oczkowska
ab0966c005 feat: hide react-flow attribution 2024-06-05 13:39:49 +01:00
dependabot[bot]
145172f486 chore(deps): Bump ip from 1.1.5 to 1.1.9
Bumps [ip](https://github.com/indutny/node-ip) from 1.1.5 to 1.1.9.
- [Commits](https://github.com/indutny/node-ip/compare/v1.1.5...v1.1.9)

---
updated-dependencies:
- dependency-name: ip
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-03-04 14:22:03 +00:00
Rıdvan Akca
e2c5d843df feat(mailerlite): add campaign sent trigger 2024-01-26 18:47:24 +03:00
Rıdvan Akca
557ea55ddf feat(mailerlite): add subscriber unsubscribed trigger 2024-01-26 18:36:49 +03:00
Rıdvan Akca
1591a4edd2 feat(mailerlite): add subscriber created trigger 2024-01-26 18:29:39 +03:00
Rıdvan Akca
b30b5024a8 feat(mailerlite): add spam complaint trigger 2024-01-26 18:22:00 +03:00
Rıdvan Akca
8ef2bd3b8d feat(mailerlite): add mailerlite integration 2024-01-26 13:44:34 +03:00
590 changed files with 16916 additions and 3504 deletions

View File

@@ -71,9 +71,6 @@ jobs:
- name: Migrate database
working-directory: ./packages/backend
run: yarn db:migrate
- name: Seed user
working-directory: ./packages/backend
run: yarn db:seed:user &
- name: Install certutils
run: sudo apt install -y libnss3-tools
- name: Install mkcert

View File

@@ -1,10 +1,10 @@
# syntax=docker/dockerfile:1
FROM node:18-alpine
ENV PORT 3000
ENV PORT=3000
RUN \
apk --no-cache add --virtual build-dependencies python3 build-base git
apk --no-cache add --virtual build-dependencies python3 build-base git make g++
WORKDIR /automatisch

View File

@@ -10,7 +10,7 @@ import process from 'process';
async function fetchAdminRole() {
const role = await Role.query()
.where({
key: 'admin',
name: 'Admin',
})
.limit(1)
.first();

View File

@@ -1,6 +1,6 @@
import { knexSnakeCaseMappers } from 'objection';
import appConfig from './src/config/app.js';
import path from 'path';
import path, { join } from 'path';
import { fileURLToPath } from 'url';
const fileExtension = 'js';
@@ -20,12 +20,12 @@ const knexConfig = {
searchPath: [appConfig.postgresSchema],
pool: { min: 0, max: 20 },
migrations: {
directory: __dirname + '/src/db/migrations',
directory: join(__dirname, '/src/db/migrations'),
extension: fileExtension,
loadExtensions: [`.${fileExtension}`],
},
seeds: {
directory: __dirname + '/src/db/seeds',
directory: join(__dirname, '/src/db/seeds'),
},
...(appConfig.isTest ? knexSnakeCaseMappers() : {}),
};

View File

@@ -5,12 +5,13 @@
"description": "The open source Zapier alternative. Build workflow automation without spending time and money.",
"type": "module",
"scripts": {
"dev": "nodemon --watch 'src/**/*.js' --exec 'node' src/server.js",
"worker": "nodemon --watch 'src/**/*.js' --exec 'node' src/worker.js",
"dev": "nodemon --exec node src/server.js",
"worker": "nodemon --exec node src/worker.js",
"start": "node src/server.js",
"start:worker": "node src/worker.js",
"pretest": "APP_ENV=test node ./test/setup/prepare-test-env.js",
"test": "APP_ENV=test vitest run",
"test:watch": "APP_ENV=test vitest watch",
"lint": "eslint .",
"db:create": "node ./bin/database/create.js",
"db:seed:user": "node ./bin/database/seed-user.js",
@@ -38,7 +39,7 @@
"debug": "~2.6.9",
"dotenv": "^10.0.0",
"express": "~4.18.2",
"express-async-handler": "^1.2.0",
"express-async-errors": "^3.1.1",
"express-basic-auth": "^1.2.1",
"express-graphql": "^0.12.0",
"fast-xml-parser": "^4.0.11",
@@ -49,6 +50,7 @@
"http-errors": "~1.6.3",
"http-proxy-agent": "^7.0.0",
"https-proxy-agent": "^7.0.1",
"isolated-vm": "^5.0.1",
"jsonwebtoken": "^9.0.0",
"knex": "^2.4.0",
"libphonenumber-js": "^1.10.48",
@@ -103,5 +105,11 @@
},
"publishConfig": {
"access": "public"
},
"nodemonConfig": {
"watch": [
"src/"
],
"ext": "js"
}
}

View File

@@ -1,5 +1,6 @@
import createError from 'http-errors';
import express from 'express';
import 'express-async-errors';
import cors from 'cors';
import appConfig from './config/app.js';

View File

@@ -7,7 +7,7 @@ export default defineAction({
'Creates an attachment of a specified object by given parent ID.',
arguments: [
{
label: 'Templete Data',
label: 'Template Data',
key: 'templateData',
type: 'string',
required: true,

View File

@@ -0,0 +1,72 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Create folder',
key: 'createFolder',
description: 'Creates a new folder.',
arguments: [
{
label: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
{
label: 'Space',
key: 'spaceId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.workspaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpaces',
},
{
name: 'parameters.workspaceId',
value: '{parameters.workspaceId}',
},
],
},
},
{
label: 'Folder Name',
key: 'folderName',
type: 'string',
required: true,
description: '',
variables: true,
},
],
async run($) {
const { spaceId, folderName } = $.step.parameters;
const body = {
name: folderName,
};
const { data } = await $.http.post(`/v2/space/${spaceId}/folder`, body);
$.setActionItem({
raw: data,
});
},
});

View File

@@ -0,0 +1,135 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Create list',
key: 'createList',
description: 'Creates a new list.',
arguments: [
{
label: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
{
label: 'Space',
key: 'spaceId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.workspaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpaces',
},
{
name: 'parameters.workspaceId',
value: '{parameters.workspaceId}',
},
],
},
},
{
label: 'Folder',
key: 'folderId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.spaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listFolders',
},
{
name: 'parameters.spaceId',
value: '{parameters.spaceId}',
},
],
},
},
{
label: 'List Name',
key: 'listName',
type: 'string',
required: true,
description: '',
variables: true,
},
{
label: 'List Info',
key: 'listInfo',
type: 'string',
required: false,
description: '',
variables: true,
},
{
label: 'Priority',
key: 'priority',
type: 'dropdown',
required: false,
description: '',
variables: true,
options: [
{ label: 'Urgent', value: 1 },
{ label: 'High', value: 2 },
{ label: 'Normal', value: 3 },
{ label: 'Low', value: 4 },
],
},
{
label: 'Due Date',
key: 'dueDate',
type: 'string',
required: false,
description: 'format: integer <int64>',
variables: true,
},
],
async run($) {
const { folderId, listName, listInfo, priority, dueDate } =
$.step.parameters;
const body = {
name: listName,
content: listInfo,
};
if (priority) {
body.priority = priority;
}
if (dueDate) {
body.due_date = dueDate;
}
const { data } = await $.http.post(`/v2/folder/${folderId}/list`, body);
$.setActionItem({
raw: data,
});
},
});

View File

@@ -0,0 +1,294 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Create task',
key: 'createTask',
description: 'Creates a new task.',
arguments: [
{
label: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
{
label: 'Space',
key: 'spaceId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.workspaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpaces',
},
{
name: 'parameters.workspaceId',
value: '{parameters.workspaceId}',
},
],
},
},
{
label: 'Folder',
key: 'folderId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.spaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listFolders',
},
{
name: 'parameters.spaceId',
value: '{parameters.spaceId}',
},
],
},
},
{
label: 'List',
key: 'listId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.folderId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listLists',
},
{
name: 'parameters.folderId',
value: '{parameters.folderId}',
},
],
},
},
{
label: 'Task Name',
key: 'taskName',
type: 'string',
required: true,
description: '',
variables: true,
},
{
label: 'Task Description',
key: 'taskDescription',
type: 'string',
required: false,
description: '',
variables: true,
},
{
label: 'Markdown Content',
key: 'markdownContent',
type: 'dropdown',
required: false,
description: '',
variables: true,
options: [
{ label: 'False', value: 'false' },
{ label: 'True', value: 'true' },
],
},
{
label: 'Assignees',
key: 'assigneeIds',
type: 'dynamic',
required: false,
description: '',
fields: [
{
label: 'Assignee',
key: 'assigneeId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.listId'],
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listAssignees',
},
{
name: 'parameters.listId',
value: '{parameters.listId}',
},
],
},
},
],
},
{
label: 'Task Status',
key: 'taskStatus',
type: 'dropdown',
required: false,
dependsOn: ['parameters.listId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listStatuses',
},
{
name: 'parameters.listId',
value: '{parameters.listId}',
},
],
},
},
{
label: 'Tags',
key: 'tagIds',
type: 'dynamic',
required: false,
description: '',
fields: [
{
label: 'tag',
key: 'tagId',
type: 'dropdown',
required: false,
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listTags',
},
],
},
},
],
},
{
label: 'Priority',
key: 'priority',
type: 'dropdown',
required: false,
description: '',
variables: true,
options: [
{ label: 'Urgent', value: 1 },
{ label: 'High', value: 2 },
{ label: 'Normal', value: 3 },
{ label: 'Low', value: 4 },
],
},
{
label: 'Due Date',
key: 'dueDate',
type: 'string',
required: false,
description: 'format: integer <int64>',
variables: true,
},
{
label: 'Start Date',
key: 'startDate',
type: 'string',
required: false,
description: 'format: integer <int64>',
variables: true,
},
],
async run($) {
const {
listId,
taskName,
taskDescription,
markdownContent,
assigneeIds,
taskStatus,
tagIds,
priority,
dueDate,
startDate,
} = $.step.parameters;
const tags = tagIds.map((tag) => tag.tagId);
const assignees = assigneeIds.map((assignee) =>
Number(assignee.assigneeId)
);
const body = {
name: taskName,
};
if (assignees.length) {
body.assignees = assignees;
}
if (taskStatus) {
body.status = taskStatus;
}
if (tags.length) {
body.tags = tags;
}
if (priority) {
body.priority = priority;
}
if (dueDate) {
body.due_date = dueDate;
}
if (startDate) {
body.start_date = startDate;
}
if (markdownContent) {
body.markdown_description = taskDescription;
} else {
body.description = taskDescription;
}
const { data } = await $.http.post(`/v2/list/${listId}/task`, body);
$.setActionItem({
raw: data,
});
},
});

View File

@@ -0,0 +1,82 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Find task by id',
key: 'findTaskById',
description: 'Finds a task using id.',
arguments: [
{
label: 'Task ID',
key: 'taskId',
type: 'string',
required: true,
description: '',
variables: true,
},
{
label: 'Use Custom ID',
key: 'useCustomId',
type: 'dropdown',
required: false,
description: '',
variables: true,
options: [
{
label: 'True',
value: true,
},
{
label: 'False',
value: false,
},
],
additionalFields: {
type: 'query',
name: 'getDynamicFields',
arguments: [
{
name: 'key',
value: 'listFieldsWhenUsingCustomId',
},
{
name: 'parameters.useCustomId',
value: '{parameters.useCustomId}',
},
],
},
},
{
label: 'Include Subtasks?',
key: 'includeSubtasks',
type: 'dropdown',
required: false,
description: '',
variables: true,
options: [
{
label: 'True',
value: true,
},
{
label: 'False',
value: false,
},
],
},
],
async run($) {
const { taskId, useCustomId, includeSubtasks } = $.step.parameters;
const params = {
custom_task_ids: useCustomId || false,
include_subtasks: includeSubtasks,
};
const { data } = await $.http.get(`/v2/task/${taskId}`, { params });
$.setActionItem({
raw: data,
});
},
});

View File

@@ -0,0 +1,6 @@
import createFolder from './create-folder/index.js';
import createList from './create-list/index.js';
import createTask from './create-task/index.js';
import findTaskById from './find-task-by-id/index.js';
export default [createFolder, createList, createTask, findTaskById];

View File

@@ -0,0 +1,27 @@
<svg width="185" height="185" viewBox="0 0 185 185" fill="none" xmlns="http://www.w3.org/2000/svg">
<g filter="url(#filter0_d)">
<rect x="30" y="20" width="125" height="125" rx="62.5" fill="white"/>
<rect x="30" y="20" width="125" height="125" rx="62.5" fill="white"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M55.8789 105.714L69.3974 95.3593C76.5762 104.732 84.1998 109.051 92.6948 109.051C101.143 109.051 108.557 104.781 115.414 95.4832L129.119 105.59C119.232 118.996 106.932 126.079 92.6948 126.079C78.5049 126.079 66.0907 119.046 55.8789 105.714Z" fill="url(#paint0_linear)"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M92.6491 60.7078L68.5883 81.4406L57.4727 68.5407L92.6969 38.1885L127.647 68.5644L116.477 81.417L92.6491 60.7078Z" fill="url(#paint1_linear)"/>
</g>
<defs>
<filter id="filter0_d" x="0" y="0" width="185" height="185" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"/>
<feOffset dy="10"/>
<feGaussianBlur stdDeviation="15"/>
<feColorMatrix type="matrix" values="0 0 0 0 0.0627451 0 0 0 0 0.117647 0 0 0 0 0.211765 0 0 0 0.1 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow" result="shape"/>
</filter>
<linearGradient id="paint0_linear" x1="55.8789" y1="116.251" x2="129.119" y2="116.251" gradientUnits="userSpaceOnUse">
<stop stop-color="#8930FD"/>
<stop offset="1" stop-color="#49CCF9"/>
</linearGradient>
<linearGradient id="paint1_linear" x1="57.4727" y1="67.6025" x2="127.647" y2="67.6025" gradientUnits="userSpaceOnUse">
<stop stop-color="#FF02F0"/>
<stop offset="1" stop-color="#FFC800"/>
</linearGradient>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 1.8 KiB

View File

@@ -0,0 +1,21 @@
import { URLSearchParams } from 'url';
export default async function generateAuthUrl($) {
const oauthRedirectUrlField = $.app.auth.fields.find(
(field) => field.key == 'oAuthRedirectUrl'
);
const redirectUri = oauthRedirectUrlField.value;
const state = Math.random().toString();
const searchParams = new URLSearchParams({
client_id: $.auth.data.clientId,
redirect_uri: redirectUri,
state,
});
const url = `https://app.clickup.com/api?${searchParams.toString()}`;
await $.auth.set({
url,
originalState: state,
});
}

View File

@@ -0,0 +1,46 @@
import generateAuthUrl from './generate-auth-url.js';
import verifyCredentials from './verify-credentials.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/clickup/connections/add',
placeholder: null,
description:
'When asked to input a redirect URL in ClickUp, 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,
};

View File

@@ -0,0 +1,8 @@
import getCurrentUser from '../common/get-current-user.js';
const isStillVerified = async ($) => {
const currentUser = await getCurrentUser($);
return !!currentUser.id;
};
export default isStillVerified;

View File

@@ -0,0 +1,31 @@
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.`);
}
const { data } = await $.http.post('/v2/oauth/token', {
client_id: $.auth.data.clientId,
client_secret: $.auth.data.clientSecret,
code: $.auth.data.code,
});
await $.auth.set({
accessToken: data.access_token,
tokenType: data.token_type,
});
const currentUser = await getCurrentUser($);
const screenName = [currentUser.username, currentUser.email]
.filter(Boolean)
.join(' @ ');
await $.auth.set({
clientId: $.auth.data.clientId,
clientSecret: $.auth.data.clientSecret,
screenName,
});
};
export default verifyCredentials;

View File

@@ -0,0 +1,9 @@
const addAuthHeader = ($, requestConfig) => {
if ($.auth.data?.accessToken) {
requestConfig.headers.Authorization = `${$.auth.data.tokenType} ${$.auth.data.accessToken}`;
}
return requestConfig;
};
export default addAuthHeader;

View File

@@ -0,0 +1,6 @@
const getCurrentUser = async ($) => {
const { data } = await $.http.get('/v2/user');
return data.user;
};
export default getCurrentUser;

View File

@@ -0,0 +1,19 @@
import listAssignees from './list-assignees/index.js';
import listFolders from './list-folders/index.js';
import listLists from './list-lists/index.js';
import listSpaces from './list-spaces/index.js';
import listStatuses from './list-statuses/index.js';
import listTags from './list-tags/index.js';
import listTasks from './list-tasks/index.js';
import listWorkspaces from './list-workspaces/index.js';
export default [
listAssignees,
listFolders,
listLists,
listSpaces,
listStatuses,
listTags,
listTasks,
listWorkspaces,
];

View File

@@ -0,0 +1,28 @@
export default {
name: 'List assignees',
key: 'listAssignees',
async run($) {
const assignees = {
data: [],
};
const listId = $.step.parameters.listId;
if (!listId) {
return assignees;
}
const { data } = await $.http.get(`/v2/list/${listId}/member`);
if (data.members) {
for (const member of data.members) {
assignees.data.push({
value: member.id,
name: member.username,
});
}
}
return assignees;
},
};

View File

@@ -0,0 +1,28 @@
export default {
name: 'List folders',
key: 'listFolders',
async run($) {
const folders = {
data: [],
};
const spaceId = $.step.parameters.spaceId;
if (!spaceId) {
return folders;
}
const { data } = await $.http.get(`/v2/space/${spaceId}/folder`);
if (data.folders) {
for (const folder of data.folders) {
folders.data.push({
value: folder.id,
name: folder.name,
});
}
}
return folders;
},
};

View File

@@ -0,0 +1,28 @@
export default {
name: 'List lists',
key: 'listLists',
async run($) {
const lists = {
data: [],
};
const folderId = $.step.parameters.folderId;
if (!folderId) {
return lists;
}
const { data } = await $.http.get(`/v2/folder/${folderId}/list`);
if (data.lists) {
for (const list of data.lists) {
lists.data.push({
value: list.id,
name: list.name,
});
}
}
return lists;
},
};

View File

@@ -0,0 +1,28 @@
export default {
name: 'List spaces',
key: 'listSpaces',
async run($) {
const spaces = {
data: [],
};
const workspaceId = $.step.parameters.workspaceId;
if (!workspaceId) {
return spaces;
}
const { data } = await $.http.get(`/v2/team/${workspaceId}/space`);
if (data.spaces) {
for (const space of data.spaces) {
spaces.data.push({
value: space.id,
name: space.name,
});
}
}
return spaces;
},
};

View File

@@ -0,0 +1,28 @@
export default {
name: 'List statuses',
key: 'listStatuses',
async run($) {
const statuses = {
data: [],
};
const listId = $.step.parameters.listId;
if (!listId) {
return statuses;
}
const { data } = await $.http.get(`/v2/list/${listId}`);
if (data.statuses) {
for (const status of data.statuses) {
statuses.data.push({
value: status.status,
name: status.status,
});
}
}
return statuses;
},
};

View File

@@ -0,0 +1,28 @@
export default {
name: 'List tags',
key: 'listTags',
async run($) {
const tags = {
data: [],
};
const spaceId = $.step.parameters.spaceId;
if (!spaceId) {
return spaceId;
}
const { data } = await $.http.get(`v2/space/${spaceId}/tag`);
if (data.tags) {
for (const tag of data.tags) {
tags.data.push({
value: tag.name,
name: tag.name,
});
}
}
return tags;
},
};

View File

@@ -0,0 +1,41 @@
export default {
name: 'List tasks',
key: 'listTasks',
async run($) {
const tasks = {
data: [],
};
const listId = $.step.parameters.listId;
let next = false;
if (!listId) {
return tasks;
}
const params = {
order_by: 'created',
reverse: true,
};
do {
const { data } = await $.http.get(`/v2/list/${listId}/task`, { params });
if (data.last_page) {
next = false;
} else {
next = true;
}
if (data.tasks) {
for (const task of data.tasks) {
tasks.data.push({
value: task.id,
name: task.name,
});
}
}
} while (next);
return tasks;
},
};

View File

@@ -0,0 +1,23 @@
export default {
name: 'List workspaces',
key: 'listWorkspaces',
async run($) {
const workspaces = {
data: [],
};
const { data } = await $.http.get('/v2/team');
if (data.teams) {
for (const workspace of data.teams) {
workspaces.data.push({
value: workspace.id,
name: workspace.name,
});
}
}
return workspaces;
},
};

View File

@@ -0,0 +1,3 @@
import useCustomId from './use-custom-id/index.js';
export default [useCustomId];

View File

@@ -0,0 +1,29 @@
export default {
name: 'List workspaces when using custom id',
key: 'listFieldsWhenUsingCustomId',
async run($) {
if ($.step.parameters.useCustomId) {
return [
{
label: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
];
}
},
};

View File

@@ -0,0 +1,24 @@
import defineApp from '../../helpers/define-app.js';
import addAuthHeader from './common/add-auth-header.js';
import auth from './auth/index.js';
import triggers from './triggers/index.js';
import dynamicData from './dynamic-data/index.js';
import actions from './actions/index.js';
import dynamicFields from './dynamic-fields/index.js';
export default defineApp({
name: 'ClickUp',
key: 'clickup',
baseUrl: 'https://clickup.com',
apiBaseUrl: 'https://api.clickup.com/api',
iconUrl: '{BASE_URL}/apps/clickup/assets/favicon.svg',
authDocUrl: 'https://automatisch.io/docs/apps/clickup/connection',
primaryColor: 'FD71AF',
supportsConnections: true,
beforeRequest: [addAuthHeader],
auth,
triggers,
dynamicData,
actions,
dynamicFields,
});

View File

@@ -0,0 +1,6 @@
import newFolders from './new-folders/index.js';
import newLists from './new-lists/index.js';
import newTasks from './new-tasks/index.js';
import updatedTask from './updated-task/index.js';
export default [newFolders, newLists, newTasks, updatedTask];

View File

@@ -0,0 +1,105 @@
import Crypto from 'crypto';
import defineTrigger from '../../../../helpers/define-trigger.js';
export default defineTrigger({
name: 'New folders',
key: 'newFolder',
type: 'webhook',
description: 'Triggers when a new folder is created.',
arguments: [
{
label: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
{
label: 'Space',
key: 'spaceId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.workspaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpaces',
},
{
name: 'parameters.workspaceId',
value: '{parameters.workspaceId}',
},
],
},
},
],
async run($) {
const dataItem = {
raw: $.request.body,
meta: {
internalId: $.request.body.folder_id,
},
};
$.pushTriggerItem(dataItem);
},
async testRun($) {
const sampleEventData = {
event: 'folderCreated',
folder_id: '90180382912',
webhook_id: Crypto.randomUUID(),
};
const dataItem = {
raw: sampleEventData,
meta: {
internalId: '',
},
};
$.pushTriggerItem(dataItem);
},
async registerHook($) {
const { workspaceId, spaceId } = $.step.parameters;
const payload = {
name: $.flow.id,
endpoint: $.webhookUrl,
events: ['folderCreated'],
};
if (spaceId) {
payload.space_id = spaceId;
}
const { data } = await $.http.post(
`/v2/team/${workspaceId}/webhook`,
payload
);
await $.flow.setRemoteWebhookId(data.id);
},
async unregisterHook($) {
await $.http.delete(`/v2/webhook/${$.flow.remoteWebhookId}`);
},
});

View File

@@ -0,0 +1,129 @@
import Crypto from 'crypto';
import defineTrigger from '../../../../helpers/define-trigger.js';
export default defineTrigger({
name: 'New lists',
key: 'newLists',
type: 'webhook',
description: 'Triggers when a new list is created.',
arguments: [
{
label: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
{
label: 'Space',
key: 'spaceId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.workspaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpaces',
},
{
name: 'parameters.workspaceId',
value: '{parameters.workspaceId}',
},
],
},
},
{
label: 'Folder',
key: 'folderId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.spaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listFolders',
},
{
name: 'parameters.spaceId',
value: '{parameters.spaceId}',
},
],
},
},
],
async run($) {
const dataItem = {
raw: $.request.body,
meta: {
internalId: $.request.body.list_id,
},
};
$.pushTriggerItem(dataItem);
},
async testRun($) {
const sampleEventData = {
event: 'listCreated',
list_id: '901800588812',
webhook_id: Crypto.randomUUID(),
};
const dataItem = {
raw: sampleEventData,
meta: {
internalId: sampleEventData.webhook_id,
},
};
$.pushTriggerItem(dataItem);
},
async registerHook($) {
const { workspaceId, spaceId, folderId } = $.step.parameters;
const payload = {
name: $.flow.id,
endpoint: $.webhookUrl,
events: ['listCreated'],
space_id: spaceId,
};
if (folderId) {
payload.folder_id = folderId;
}
const { data } = await $.http.post(
`/v2/team/${workspaceId}/webhook`,
payload
);
await $.flow.setRemoteWebhookId(data.id);
},
async unregisterHook($) {
await $.http.delete(`/v2/webhook/${$.flow.remoteWebhookId}`);
},
});

View File

@@ -0,0 +1,186 @@
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: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
{
label: 'Space',
key: 'spaceId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.workspaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpaces',
},
{
name: 'parameters.workspaceId',
value: '{parameters.workspaceId}',
},
],
},
},
{
label: 'Folder',
key: 'folderId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.spaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listFolders',
},
{
name: 'parameters.spaceId',
value: '{parameters.spaceId}',
},
],
},
},
{
label: 'List',
key: 'listId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.folderId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listLists',
},
{
name: 'parameters.folderId',
value: '{parameters.folderId}',
},
],
},
},
{
label: 'Task',
key: 'taskId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.listId'],
description:
'Choose an optional task to determine when this flow should be activated. In this scenario, only subtasks will initiate this flow.',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listTasks',
},
{
name: 'parameters.listId',
value: '{parameters.listId}',
},
],
},
},
],
async run($) {
const dataItem = {
raw: $.request.body,
meta: {
internalId: $.request.body.task_id,
},
};
$.pushTriggerItem(dataItem);
},
async testRun($) {
const sampleEventData = {
event: 'taskCreated',
task_id: '86enn7pg7',
webhook_id: Crypto.randomUUID(),
history_items: [],
};
const dataItem = {
raw: sampleEventData,
meta: {
internalId: sampleEventData.webhook_id,
},
};
$.pushTriggerItem(dataItem);
},
async registerHook($) {
const { workspaceId, spaceId, folderId, listId, taskId } =
$.step.parameters;
const payload = {
name: $.flow.id,
endpoint: $.webhookUrl,
events: ['taskCreated'],
space_id: spaceId,
};
if (folderId) {
payload.folder_id = folderId;
}
if (listId) {
payload.list_id = listId;
}
if (taskId) {
payload.task_id = taskId;
}
const { data } = await $.http.post(
`/v2/team/${workspaceId}/webhook`,
payload
);
await $.flow.setRemoteWebhookId(data.id);
},
async unregisterHook($) {
await $.http.delete(`/v2/webhook/${$.flow.remoteWebhookId}`);
},
});

View File

@@ -0,0 +1,172 @@
import Crypto from 'crypto';
import defineTrigger from '../../../../helpers/define-trigger.js';
export default defineTrigger({
name: 'Updated task',
key: 'updatedTask',
type: 'webhook',
description: 'Triggers when a task is updated.',
arguments: [
{
label: 'Workspace',
key: 'workspaceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listWorkspaces',
},
],
},
},
{
label: 'Space',
key: 'spaceId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.workspaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpaces',
},
{
name: 'parameters.workspaceId',
value: '{parameters.workspaceId}',
},
],
},
},
{
label: 'Folder',
key: 'folderId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.spaceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listFolders',
},
{
name: 'parameters.spaceId',
value: '{parameters.spaceId}',
},
],
},
},
{
label: 'List',
key: 'listId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.folderId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listLists',
},
{
name: 'parameters.folderId',
value: '{parameters.folderId}',
},
],
},
},
{
label: 'What Changed?',
key: 'whatChanged',
type: 'dropdown',
required: false,
variables: true,
options: [
{ label: 'Status', value: 'taskStatusUpdated' },
{ label: 'Assignee Added', value: 'taskAssigneeUpdated' },
{ label: 'Priority', value: 'taskPriorityUpdated' },
{ label: 'Tag Added', value: 'taskTagUpdated' },
],
},
],
async run($) {
const dataItem = {
raw: $.request.body,
meta: {
internalId: Crypto.randomUUID(),
},
};
$.pushTriggerItem(dataItem);
},
async testRun($) {
const sampleEventData = {
event: 'taskUpdated',
task_id: '86enn7pg7',
webhook_id: Crypto.randomUUID(),
history_items: [],
};
const dataItem = {
raw: sampleEventData,
meta: {
internalId: sampleEventData.webhook_id,
},
};
$.pushTriggerItem(dataItem);
},
async registerHook($) {
const { workspaceId, spaceId, folderId, listId, whatChanged } =
$.step.parameters;
const payload = {
name: $.flow.id,
endpoint: $.webhookUrl,
space_id: spaceId,
};
payload.events = [whatChanged || 'taskUpdated'];
if (folderId) {
payload.folder_id = folderId;
}
if (listId) {
payload.list_id = listId;
}
const { data } = await $.http.post(
`/v2/team/${workspaceId}/webhook`,
payload
);
await $.flow.setRemoteWebhookId(data.id);
},
async unregisterHook($) {
await $.http.delete(`/v2/webhook/${$.flow.remoteWebhookId}`);
},
});

View File

@@ -0,0 +1,3 @@
import runJavascript from './run-javascript/index.js';
export default [runJavascript];

View File

@@ -0,0 +1,84 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Run Javascript',
key: 'runJavascript',
description:
'Run browser Javascript code. You can not use NodeJS specific features and npm packages.',
arguments: [
{
label: 'Inputs',
key: 'inputs',
type: 'dynamic',
required: false,
description:
'To be able to use data from previous steps, you need to expose them as input entries. You can access these input values in your code by using the `inputs` argument.',
value: [
{
key: '',
value: '',
},
],
fields: [
{
label: 'Key',
key: 'key',
type: 'string',
required: true,
variables: true,
},
{
label: 'Value',
key: 'value',
type: 'string',
required: true,
variables: true,
valueType: 'parse',
},
],
},
{
label: 'Code Snippet',
key: 'codeSnippet',
type: 'code',
required: true,
variables: false,
value:
'const code = async (inputs) => { \n // E.g. if you have an input called username,\n // you can access its value by calling inputs.username\n // Return value will be used as output of this step.\n\n return true;\n};',
},
],
async run($) {
const { inputs = [], codeSnippet } = $.step.parameters;
const objectifiedInput = {};
for (const input of inputs) {
if (input.key) {
objectifiedInput[input.key] = input.value;
}
}
const ivm = (await import('isolated-vm')).default;
const isolate = new ivm.Isolate({ memoryLimit: 128 });
try {
const context = await isolate.createContext();
await context.global.set(
'inputs',
new ivm.ExternalCopy(objectifiedInput).copyInto()
);
const compiledCodeSnippet = await isolate.compileScript(
`${codeSnippet}; code(inputs);`
);
const codeFunction = await compiledCodeSnippet.run(context, {
reference: true,
promise: true,
});
$.setActionItem({ raw: { output: await codeFunction.copy() } });
} finally {
isolate.dispose();
}
},
});

View File

@@ -0,0 +1,5 @@
<svg xmlns="http://www.w3.org/2000/svg" width="800px" height="800px" viewBox="0 0 512 512">
<polyline points="160 368 32 256 160 144" style="fill:none;stroke:#000000;stroke-linecap:round;stroke-linejoin:round;stroke-width:32px"/>
<polyline points="352 368 480 256 352 144" style="fill:none;stroke:#000000;stroke-linecap:round;stroke-linejoin:round;stroke-width:32px"/>
<line x1="304" y1="96" x2="208" y2="416" style="fill:none;stroke:#000000;stroke-linecap:round;stroke-linejoin:round;stroke-width:32px"/>
</svg>

After

Width:  |  Height:  |  Size: 519 B

View File

@@ -0,0 +1,14 @@
import defineApp from '../../helpers/define-app.js';
import actions from './actions/index.js';
export default defineApp({
name: 'Code',
key: 'code',
baseUrl: '',
apiBaseUrl: '',
iconUrl: '{BASE_URL}/apps/code/assets/favicon.svg',
authDocUrl: '{DOCS_URL}/apps/code/connection',
primaryColor: '000000',
supportsConnections: false,
actions,
});

View File

@@ -0,0 +1,64 @@
import { createHmac } from 'node:crypto';
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Create HMAC',
key: 'createHmac',
description: 'Create a Hash-based Message Authentication Code (HMAC) using the specified algorithm, secret key, and message.',
arguments: [
{
label: 'Algorithm',
key: 'algorithm',
type: 'dropdown',
required: true,
value: 'sha256',
description: 'Specifies the cryptographic hash function to use for HMAC generation.',
options: [
{ label: 'SHA-256', value: 'sha256' },
],
variables: true,
},
{
label: 'Message',
key: 'message',
type: 'string',
required: true,
description: 'The input message to be hashed. This is the value that will be processed to generate the HMAC.',
variables: true,
},
{
label: 'Secret Key',
key: 'secretKey',
type: 'string',
required: true,
description: 'The secret key used to create the HMAC.',
variables: true,
},
{
label: 'Output Encoding',
key: 'outputEncoding',
type: 'dropdown',
required: true,
value: 'hex',
description: 'Specifies the encoding format for the HMAC digest output.',
options: [
{ label: 'base64', value: 'base64' },
{ label: 'base64url', value: 'base64url' },
{ label: 'hex', value: 'hex' },
],
variables: true,
},
],
async run($) {
const hash = createHmac($.step.parameters.algorithm, $.step.parameters.secretKey)
.update($.step.parameters.message)
.digest($.step.parameters.outputEncoding);
$.setActionItem({
raw: {
hash
},
});
},
});

View File

@@ -0,0 +1,65 @@
import crypto from 'node:crypto';
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Create Signature',
key: 'createSignature',
description: 'Create a digital signature using the specified algorithm, secret key, and message.',
arguments: [
{
label: 'Algorithm',
key: 'algorithm',
type: 'dropdown',
required: true,
value: 'RSA-SHA256',
description: 'Specifies the cryptographic hash function to use for HMAC generation.',
options: [
{ label: 'RSA-SHA256', value: 'RSA-SHA256' },
],
variables: true,
},
{
label: 'Message',
key: 'message',
type: 'string',
required: true,
description: 'The input message to be signed.',
variables: true,
},
{
label: 'Private Key',
key: 'privateKey',
type: 'string',
required: true,
description: 'The RSA private key in PEM format used for signing.',
variables: true,
},
{
label: 'Output Encoding',
key: 'outputEncoding',
type: 'dropdown',
required: true,
value: 'hex',
description: 'Specifies the encoding format for the digital signature output. This determines how the generated signature will be represented as a string.',
options: [
{ label: 'base64', value: 'base64' },
{ label: 'base64url', value: 'base64url' },
{ label: 'hex', value: 'hex' },
],
variables: true,
},
],
async run($) {
const signer = crypto.createSign($.step.parameters.algorithm);
signer.update($.step.parameters.message);
signer.end();
const signature = signer.sign($.step.parameters.privateKey, $.step.parameters.outputEncoding);
$.setActionItem({
raw: {
signature
},
});
},
});

View File

@@ -0,0 +1,4 @@
import createHmac from './create-hmac/index.js';
import createRsaSha256Signature from './create-rsa-sha256-signature/index.js';
export default [createHmac, createRsaSha256Signature];

View File

@@ -0,0 +1,3 @@
<svg xmlns="http://www.w3.org/2000/svg" width="100pt" height="100pt" version="1.1" viewBox="0 0 100 100">
<path d="m66.012 33h-3.0117v-11c0-7.1719-5.8281-13-13-13s-13 5.8281-13 13v11h-3.0117c-2.75 0-4.9883 2.2383-4.9883 4.9883v28.012c0 2.75 2.2383 4.9883 4.9883 4.9883h32.012c2.75 0 4.9883-2.2383 4.9883-4.9883v-28.012c0.011719-2.75-2.2266-4.9883-4.9766-4.9883zm-27.012-11c0-6.0703 4.9297-11 11-11s11 4.9297 11 11v11h-22zm30 44.012c0 1.6484-1.3398 2.9883-2.9883 2.9883h-32.023c-1.6484 0-2.9883-1.3398-2.9883-2.9883v-28.023c0-1.6484 1.3398-2.9883 2.9883-2.9883h32.023c1.6484 0 2.9883 1.3398 2.9883 2.9883zm-18 9.9883v14c0 0.55078-0.44922 1-1 1s-1-0.44922-1-1v-14c0-0.55078 0.44922-1 1-1s1 0.44922 1 1zm20 8c0 0.55078-0.44922 1-1 1h-8c-0.55078 0-1-0.44922-1-1v-8c0-0.55078 0.44922-1 1-1s1 0.44922 1 1v7h7c0.55078 0 1 0.44922 1 1zm-32-8v8c0 0.55078-0.44922 1-1 1h-8c-0.55078 0-1-0.44922-1-1s0.44922-1 1-1h7v-7c0-0.55078 0.44922-1 1-1s1 0.44922 1 1zm-14-26c0 0.55078-0.44922 1-1 1h-14c-0.55078 0-1-0.44922-1-1s0.44922-1 1-1h14c0.55078 0 1 0.44922 1 1zm0-12c0 0.55078-0.44922 1-1 1h-8c-0.55078 0-1-0.44922-1-1v-8c0-0.55078 0.44922-1 1-1s1 0.44922 1 1v7h7c0.55078 0 1 0.44922 1 1zm0 24c0 0.55078-0.44922 1-1 1h-7v7c0 0.55078-0.44922 1-1 1s-1-0.44922-1-1v-8c0-0.55078 0.44922-1 1-1h8c0.55078 0 1 0.44922 1 1zm66-12c0 0.55078-0.44922 1-1 1h-14c-0.55078 0-1-0.44922-1-1s0.44922-1 1-1h14c0.55078 0 1 0.44922 1 1zm-16-12c0-0.55078 0.44922-1 1-1h7v-7c0-0.55078 0.44922-1 1-1s1 0.44922 1 1v8c0 0.55078-0.44922 1-1 1h-8c-0.55078 0-1-0.44922-1-1zm10 24v8c0 0.55078-0.44922 1-1 1s-1-0.44922-1-1v-7h-7c-0.55078 0-1-0.44922-1-1s0.44922-1 1-1h8c0.55078 0 1 0.44922 1 1zm-35-17c-2.7617 0-5 2.2383-5 5 0 2.4102 1.7188 4.4297 4 4.8984v5.1016c0 0.55078 0.44922 1 1 1s1-0.44922 1-1v-5.1016c2.2812-0.46094 4-2.4805 4-4.8984 0-2.7617-2.2383-5-5-5zm0 8c-1.6484 0-3-1.3516-3-3s1.3516-3 3-3 3 1.3516 3 3-1.3516 3-3 3z"/>
</svg>

After

Width:  |  Height:  |  Size: 1.9 KiB

View File

@@ -0,0 +1,14 @@
import defineApp from '../../helpers/define-app.js';
import actions from './actions/index.js';
export default defineApp({
name: 'Cryptography',
key: 'cryptography',
iconUrl: '{BASE_URL}/apps/cryptography/assets/favicon.svg',
authDocUrl: '{DOCS_URL}/apps/cryptography/connection',
supportsConnections: false,
baseUrl: '',
apiBaseUrl: '',
primaryColor: '001F52',
actions,
});

View File

@@ -1,8 +1,10 @@
import defineAction from '../../../../helpers/define-action.js';
import formatDateTime from './transformers/format-date-time.js';
import getCurrentTimestamp from './transformers/get-current-timestamp.js';
const transformers = {
formatDateTime,
getCurrentTimestamp,
};
export default defineAction({
@@ -16,7 +18,16 @@ export default defineAction({
type: 'dropdown',
required: true,
variables: true,
options: [{ label: 'Format Date / Time', value: 'formatDateTime' }],
options: [
{
label: 'Get current timestamp',
value: 'getCurrentTimestamp',
},
{
label: 'Format Date / Time',
value: 'formatDateTime',
},
],
additionalFields: {
type: 'query',
name: 'getDynamicFields',

View File

@@ -0,0 +1,5 @@
const getCurrentTimestamp = () => {
return Date.now();
};
export default getCurrentTimestamp;

View File

@@ -14,6 +14,8 @@ import stringToBase64 from './transformers/string-to-base64.js';
import encodeUri from './transformers/encode-uri.js';
import trimWhitespace from './transformers/trim-whitespace.js';
import useDefaultValue from './transformers/use-default-value.js';
import parseStringifiedJson from './transformers/parse-stringified-json.js';
import createUuid from './transformers/create-uuid.js';
const transformers = {
base64ToString,
@@ -30,6 +32,8 @@ const transformers = {
encodeUri,
trimWhitespace,
useDefaultValue,
parseStringifiedJson,
createUuid,
};
export default defineAction({
@@ -47,19 +51,21 @@ export default defineAction({
options: [
{ label: 'Base64 to String', value: 'base64ToString' },
{ label: 'Capitalize', value: 'capitalize' },
{ label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' },
{ label: 'Convert Markdown to HTML', value: 'markdownToHtml' },
{ label: 'Create UUID', value: 'createUuid' },
{ label: 'Encode URI', value: 'encodeUri' },
{
label: 'Encode URI Component',
value: 'encodeUriComponent',
},
{ label: 'Convert HTML to Markdown', value: 'htmlToMarkdown' },
{ label: 'Convert Markdown to HTML', value: 'markdownToHtml' },
{ label: 'Extract Email Address', value: 'extractEmailAddress' },
{ label: 'Extract Number', value: 'extractNumber' },
{ label: 'Lowercase', value: 'lowercase' },
{ label: 'Parse stringified JSON', value: 'parseStringifiedJson' },
{ label: 'Pluralize', value: 'pluralize' },
{ label: 'Replace', value: 'replace' },
{ label: 'String to Base64', value: 'stringToBase64' },
{ label: 'Encode URI', value: 'encodeUri' },
{ label: 'Trim Whitespace', value: 'trimWhitespace' },
{ label: 'Use Default Value', value: 'useDefaultValue' },
],

View File

@@ -0,0 +1,7 @@
import { v4 as uuidv4 } from 'uuid';
const createUuidV4 = () => {
return uuidv4();
};
export default createUuidV4;

View File

@@ -0,0 +1,7 @@
const parseStringifiedJson = ($) => {
const input = $.step.parameters.input;
return JSON.parse(input);
};
export default parseStringifiedJson;

View File

@@ -1,8 +1,26 @@
const replace = ($) => {
const input = $.step.parameters.input;
const find = $.step.parameters.find;
const replace = $.step.parameters.replace;
const useRegex = $.step.parameters.useRegex;
if (useRegex) {
const ignoreCase = $.step.parameters.ignoreCase;
const flags = [ignoreCase && 'i', 'g'].filter(Boolean).join('');
const timeoutId = setTimeout(() => {
$.execution.exit();
}, 100);
const regex = new RegExp(find, flags);
const replacedValue = input.replaceAll(regex, replace);
clearTimeout(timeoutId);
return replacedValue;
}
return input.replaceAll(find, replace);
};

View File

@@ -1,3 +1,4 @@
import listTransformOptions from './list-transform-options/index.js';
import listReplaceRegexOptions from './list-replace-regex-options/index.js';
export default [listTransformOptions];
export default [listTransformOptions, listReplaceRegexOptions];

View File

@@ -0,0 +1,23 @@
export default {
name: 'List replace regex options',
key: 'listReplaceRegexOptions',
async run($) {
if (!$.step.parameters.useRegex) return [];
return [
{
label: 'Ignore case',
key: 'ignoreCase',
type: 'dropdown',
required: true,
description: 'Ignore case sensitivity.',
variables: true,
options: [
{ label: 'Yes', value: true },
{ label: 'No', value: false },
],
},
];
},
};

View File

@@ -12,6 +12,7 @@ import stringToBase64 from './text/string-to-base64.js';
import encodeUri from './text/encode-uri.js';
import trimWhitespace from './text/trim-whitespace.js';
import useDefaultValue from './text/use-default-value.js';
import parseStringifiedJson from './text/parse-stringified-json.js';
import performMathOperation from './numbers/perform-math-operation.js';
import randomNumber from './numbers/random-number.js';
import formatNumber from './numbers/format-number.js';
@@ -38,6 +39,7 @@ const options = {
formatNumber,
formatPhoneNumber,
formatDateTime,
parseStringifiedJson,
};
export default {

View File

@@ -0,0 +1,12 @@
const useDefaultValue = [
{
label: 'Input',
key: 'input',
type: 'string',
required: true,
description: 'Stringified JSON you want to parse.',
variables: true,
},
];
export default useDefaultValue;

View File

@@ -23,6 +23,33 @@ const replace = [
description: 'Text that will replace the found text.',
variables: true,
},
{
label: 'Use Regular Expression',
key: 'useRegex',
type: 'dropdown',
required: true,
description: 'Use regex to search values.',
variables: true,
value: false,
options: [
{ label: 'Yes', value: true },
{ label: 'No', value: false },
],
additionalFields: {
type: 'query',
name: 'getDynamicFields',
arguments: [
{
name: 'key',
value: 'listReplaceRegexOptions',
},
{
name: 'parameters.useRegex',
value: '{parameters.useRegex}',
},
],
},
},
];
export default replace;

View File

@@ -10,7 +10,7 @@ export default defineAction({
label: 'Repo',
key: 'repo',
type: 'dropdown',
required: false,
required: true,
variables: true,
source: {
type: 'query',

View File

@@ -0,0 +1,175 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Find worksheet',
key: 'findWorksheet',
description:
'Finds a worksheet by title. Optionally, create a worksheet if none are found.',
arguments: [
{
label: 'Drive',
key: 'driveId',
type: 'dropdown',
required: false,
description:
'The Google Drive where your spreadsheet resides. If nothing is selected, then your personal Google Drive will be used.',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listDrives',
},
],
},
},
{
label: 'Spreadsheet',
key: 'spreadsheetId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.driveId'],
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSpreadsheets',
},
{
name: 'parameters.driveId',
value: '{parameters.driveId}',
},
],
},
},
{
label: 'Title',
key: 'title',
type: 'string',
required: true,
description:
'The worksheet title needs to match exactly, and the search is case-sensitive.',
variables: true,
},
{
label: 'Create worksheet if none are found.',
key: 'createWorksheet',
type: 'dropdown',
required: false,
options: [
{ label: 'Yes', value: true },
{ label: 'No', value: false },
],
additionalFields: {
type: 'query',
name: 'getDynamicFields',
arguments: [
{
name: 'key',
value: 'listCreateWorksheetFields',
},
{
name: 'parameters.createWorksheet',
value: '{parameters.createWorksheet}',
},
],
},
},
],
async run($) {
const createWorksheet = $.step.parameters.createWorksheet;
async function findWorksheet() {
const {
data: { sheets },
} = await $.http.get(`/v4/spreadsheets/${$.step.parameters.spreadsheetId}`);
const selectedSheet = sheets.find(
(sheet) => sheet.properties.title === $.step.parameters.title
);
return selectedSheet;
}
const selectedSheet = await findWorksheet();
if (selectedSheet) {
$.setActionItem({
raw: selectedSheet,
});
return;
}
if (createWorksheet) {
const headers = $.step.parameters.headers;
const headerValues = headers.map((entry) => entry.header);
const body = {
requests: [
{
addSheet: {
properties: {
title: $.step.parameters.title,
},
},
},
],
};
const { data } = await $.http.post(
`/v4/spreadsheets/${$.step.parameters.spreadsheetId}:batchUpdate`,
body
);
if (headerValues.length) {
const body = {
requests: [
{
updateCells: {
rows: [
{
values: headerValues.map((header) => ({
userEnteredValue: { stringValue: header },
})),
},
],
fields: '*',
start: {
sheetId:
data.replies[data.replies.length - 1].addSheet.properties
.sheetId,
rowIndex: 0,
columnIndex: 0,
},
},
},
],
};
await $.http.post(
`/v4/spreadsheets/${$.step.parameters.spreadsheetId}:batchUpdate`,
body
);
const createdSheet = await findWorksheet();
$.setActionItem({
raw: createdSheet,
});
return;
}
}
$.setActionItem({
raw: null,
});
},
});

View File

@@ -1,5 +1,11 @@
import createSpreadsheet from './create-spreadsheet/index.js';
import createSpreadsheetRow from './create-spreadsheet-row/index.js';
import createWorksheet from './create-worksheet/index.js';
import findWorksheet from './find-worksheet/index.js';
export default [createSpreadsheet, createSpreadsheetRow, createWorksheet];
export default [
createSpreadsheet,
createSpreadsheetRow,
createWorksheet,
findWorksheet,
];

View File

@@ -1,3 +1,4 @@
import listSheetHeaders from './list-sheet-headers/index.js';
import listCreateWorksheetFields from './list-create-worksheet-fields/index.js';
export default [listSheetHeaders];
export default [listSheetHeaders, listCreateWorksheetFields];

View File

@@ -0,0 +1,26 @@
export default {
name: 'List create worksheet fields',
key: 'listCreateWorksheetFields',
async run($) {
if ($.step.parameters.createWorksheet) {
return [
{
label: 'Headers',
key: 'headers',
type: 'dynamic',
required: false,
fields: [
{
label: 'Header',
key: 'header',
type: 'string',
required: true,
variables: true,
},
],
},
];
}
},
};

View File

@@ -145,6 +145,13 @@ export default defineAction({
responseData = Buffer.from(responseData).toString('base64');
}
$.setActionItem({ raw: { data: responseData } });
$.setActionItem({
raw: {
data: responseData,
headers: response.headers,
status: response.status,
statusText: response.statusText
}
});
},
});

View File

@@ -0,0 +1 @@
<svg xmlns:xlink="http://www.w3.org/1999/xlink" class="jff-logo-img" width="53" height="59" viewBox="0 0 53 59" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true"><path d="M14.4509 55.1332C15.5462 56.1951 14.7721 58.0143 13.2168 58.0143H3.4831C1.56265 58.0143 0 56.4995 0 54.6377V45.2017C0 43.6939 1.87664 42.9436 2.97195 44.0054L14.4509 55.1332Z" fill="#0A1551"></path><path d="M29.6655 55.8676C26.7843 53.0052 26.7843 48.3642 29.6655 45.5018L40.0638 35.1713C42.945 32.3089 47.6164 32.3089 50.4976 35.1713C53.3788 38.0338 53.3788 42.6747 50.4976 45.5371L40.0993 55.8676C37.2181 58.73 32.5468 58.73 29.6655 55.8676Z" fill="#FFB629"></path><path d="M2.1968 29.9101C-0.684414 27.0476 -0.684413 22.4067 2.1968 19.5443L19.696 2.14685C22.5772 -0.71559 27.2486 -0.715594 30.1298 2.14685C33.011 5.00929 33.011 9.65022 30.1298 12.5127L12.6306 29.9101C9.74937 32.7725 5.078 32.7725 2.1968 29.9101Z" fill="#0099FF"></path><path d="M16.5015 42.3095C13.6203 39.4471 13.6203 34.8062 16.5015 31.9437L40.1461 8.45322C43.0273 5.59079 47.6986 5.59079 50.5798 8.45322C53.4611 11.3157 53.4611 15.9566 50.5798 18.819L26.9353 42.3095C24.0541 45.1719 19.3827 45.1719 16.5015 42.3095Z" fill="#FF6100"></path></svg>

After

Width:  |  Height:  |  Size: 1.2 KiB

View File

@@ -0,0 +1,30 @@
import verifyCredentials from './verify-credentials.js';
import isStillVerified from './is-still-verified.js';
export default {
fields: [
{
key: 'apiUrl',
label: 'API URL',
type: 'string',
required: false,
readOnly: false,
value: 'https://api.jotform.com',
placeholder: 'https://${subdomain}.jotform.com/api',
clickToCopy: true,
},
{
key: 'apiKey',
label: 'API Key',
type: 'string',
required: true,
readOnly: false,
value: null,
placeholder: null,
clickToCopy: false,
},
],
verifyCredentials,
isStillVerified,
};

View File

@@ -0,0 +1,8 @@
import getCurrentUser from '../common/get-current-user.js';
const isStillVerified = async ($) => {
const user = await getCurrentUser($);
return !!user.username;
};
export default isStillVerified;

View File

@@ -0,0 +1,12 @@
import getCurrentUser from '../common/get-current-user.js';
const verifyCredentials = async ($) => {
const user = await getCurrentUser($);
await $.auth.set({
screenName: user.name,
apiKey: $.auth.data.apiKey,
});
};
export default verifyCredentials;

View File

@@ -0,0 +1,9 @@
const addAuthHeader = ($, requestConfig) => {
if ($.auth.data?.apiKey) {
requestConfig.headers['APIKEY'] = `${$.auth.data.apiKey}`;
}
return requestConfig;
};
export default addAuthHeader;

View File

@@ -0,0 +1,7 @@
const getCurrentUser = async ($) => {
const response = await $.http.get('/user');
const currentUser = response.data.content;
return currentUser;
};
export default getCurrentUser;

View File

@@ -0,0 +1,11 @@
const setBaseUrl = ($, requestConfig) => {
if ($.auth.data.apiUrl) {
requestConfig.baseURL = $.auth.data.apiUrl;
} else if ($.app.apiBaseUrl) {
requestConfig.baseURL = $.app.apiBaseUrl;
}
return requestConfig;
};
export default setBaseUrl;

View File

@@ -0,0 +1,3 @@
import listForms from './list-forms/index.js';
export default [listForms];

View File

@@ -0,0 +1,41 @@
export default {
name: 'List forms',
key: 'listForms',
async run($) {
const forms = {
data: [],
};
let hasMore = false;
const params = {
limit: 1000,
offset: 0,
orderby: 'created_at',
};
do {
const { data } = await $.http.get('/user/forms', { params });
params.offset = params.offset + params.limit;
if (data.content?.length) {
for (const form of data.content) {
if (form.status === 'ENABLED') {
forms.data.push({
value: form.id,
name: form.title,
});
}
}
}
if (data.resultSet.count >= data.resultSet.limit) {
hasMore = true;
} else {
hasMore = false;
}
} while (hasMore);
return forms;
},
};

View File

@@ -0,0 +1,21 @@
import defineApp from '../../helpers/define-app.js';
import addAuthHeader from './common/add-auth-header.js';
import auth from './auth/index.js';
import setBaseUrl from './common/set-base-url.js';
import triggers from './triggers/index.js';
import dynamicData from './dynamic-data/index.js';
export default defineApp({
name: 'Jotform',
key: 'jotform',
iconUrl: '{BASE_URL}/apps/jotform/assets/favicon.svg',
authDocUrl: 'https://automatisch.io/docs/apps/jotform/connection',
supportsConnections: true,
baseUrl: 'https://www.jotform.com',
apiBaseUrl: 'https://api.jotform.com',
primaryColor: 'FF6100',
beforeRequest: [setBaseUrl, addAuthHeader],
auth,
triggers,
dynamicData,
});

View File

@@ -0,0 +1,3 @@
import newSubmissions from './new-submissions/index.js';
export default [newSubmissions];

View File

@@ -0,0 +1,109 @@
import Crypto from 'crypto';
import { URLSearchParams } from 'url';
import defineTrigger from '../../../../helpers/define-trigger.js';
export default defineTrigger({
name: 'New submissions',
key: 'newSubmissions',
type: 'webhook',
description:
'Triggers when a new submission has been added to a specific form.',
arguments: [
{
label: 'Form',
key: 'formId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listForms',
},
],
},
},
],
async run($) {
const dataItem = {
raw: $.request.body,
meta: {
internalId: Crypto.randomUUID(),
},
};
$.pushTriggerItem(dataItem);
},
async testRun($) {
const sampleEventData = {
ip: '127.0.0.1',
type: 'WEB',
appID: '',
event: '',
action: '',
formID: Crypto.randomUUID(),
parent: '',
pretty: 'Name:test, E-mail:user@automatisch.io',
teamID: '',
unread: '',
product: '',
subject: '',
isSilent: '',
username: 'username',
deviceIDs: 'Array',
formTitle: 'Opt-In Form-Get Free Email Updates!',
fromTable: '',
customBody: '',
documentID: '',
rawRequest: '',
webhookURL: '',
customTitle: '',
trackAction: 'Array',
customParams: '',
submissionID: Crypto.randomUUID(),
customBodyParams: 'Array',
customTitleParams: 'Array',
};
const dataItem = {
raw: sampleEventData,
meta: {
internalId: sampleEventData.submissionID,
},
};
$.pushTriggerItem(dataItem);
},
async registerHook($) {
const formId = $.step.parameters.formId;
const params = new URLSearchParams({
webhookURL: $.webhookUrl,
});
const { data } = await $.http.post(
`/form/${formId}/webhooks`,
params.toString()
);
await $.flow.setRemoteWebhookId(data.content[0]);
},
async unregisterHook($) {
const formId = $.step.parameters.formId;
const { data } = await $.http.get(`/form/${formId}/webhooks`);
const webhookURLs = Object.values(data.content);
const webhookId = webhookURLs.findIndex((url) => url === $.webhookUrl);
await $.http.delete(`/form/${formId}/webhooks/${webhookId}`);
},
});

View File

@@ -0,0 +1,180 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Create campaign',
key: 'createCampaign',
description: 'Creates a new campaign draft.',
arguments: [
{
label: 'Campaign Name',
key: 'campaignName',
type: 'string',
required: false,
description: '',
variables: true,
},
{
label: 'Audience',
key: 'audienceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listAudiences',
},
],
},
},
{
label: 'Segment or Tag',
key: 'segmentOrTagId',
type: 'dropdown',
required: false,
dependsOn: ['parameters.audienceId'],
description:
'Choose the specific segment or tag to which you"d like to direct the campaign. If no segment or tag is chosen, the campaign will be distributed to the entire audience previously selected.',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listSegmentsOrTags',
},
{
name: 'parameters.audienceId',
value: '{parameters.audienceId}',
},
],
},
},
{
label: 'Email Subject',
key: 'emailSubject',
type: 'string',
required: true,
description: '',
variables: true,
},
{
label: 'Preview Text',
key: 'previewText',
type: 'string',
required: false,
description:
'The snippet will be visible in the inbox following the subject line.',
variables: true,
},
{
label: 'From Name',
key: 'fromName',
type: 'string',
required: true,
description: 'The "from" name on the campaign (not an email address).',
variables: true,
},
{
label: 'From Email Address',
key: 'fromEmailAddress',
type: 'string',
required: true,
description: 'The reply-to email address for the campaign.',
variables: true,
},
{
label: 'To Name',
key: 'toName',
type: 'string',
required: false,
description:
'Supports *|MERGETAGS|* for recipient name, such as *|FNAME|*, *|LNAME|*, *|FNAME|* *|LNAME|*, etc.',
variables: true,
},
{
label: 'Template',
key: 'templateId',
type: 'dropdown',
required: false,
description:
'Select either a template or provide HTML email content, you cannot provide both. If both fields are left blank, the campaign draft will have no content.',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listTemplates',
},
],
},
},
{
label: 'Email Content (HTML)',
key: 'emailContent',
type: 'string',
required: false,
description:
'Select either a template or provide HTML email content, you cannot provide both. If both fields are left blank, the campaign draft will have no content.',
variables: true,
},
],
async run($) {
const {
campaignName,
audienceId,
segmentOrTagId,
emailSubject,
previewText,
fromName,
fromEmailAddress,
toName,
templateId,
emailContent,
} = $.step.parameters;
const body = {
type: 'regular',
recipients: {
list_id: audienceId,
segment_opts: {
saved_segment_id: Number(segmentOrTagId),
},
},
settings: {
subject_line: emailSubject,
reply_to: fromEmailAddress,
title: campaignName,
preview_text: previewText,
from_name: fromName,
to_name: toName,
},
};
const { data: campaign } = await $.http.post('/3.0/campaigns', body);
const campaignBody = {
template: {
id: Number(templateId),
},
html: emailContent,
};
const { data } = await $.http.put(
`/3.0/campaigns/${campaign.id}/content`,
campaignBody
);
$.setActionItem({
raw: data,
});
},
});

View File

@@ -0,0 +1,4 @@
import createCampaign from './create-campaign/index.js';
import sendCampaign from './send-campaign/index.js';
export default [createCampaign, sendCampaign];

View File

@@ -0,0 +1,39 @@
import defineAction from '../../../../helpers/define-action.js';
export default defineAction({
name: 'Send campaign',
key: 'sendCampaign',
description: 'Sends a campaign draft.',
arguments: [
{
label: 'Campaign',
key: 'campaignId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listCampaigns',
},
],
},
},
],
async run($) {
const campaignId = $.step.parameters.campaignId;
await $.http.post(`/3.0/campaigns/${campaignId}/actions/send`);
$.setActionItem({
raw: {
output: 'sent',
},
});
},
});

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 5.0 KiB

View File

@@ -0,0 +1,19 @@
import { URLSearchParams } from 'url';
export default async function generateAuthUrl($) {
const oauthRedirectUrlField = $.app.auth.fields.find(
(field) => field.key == 'oAuthRedirectUrl'
);
const redirectUri = oauthRedirectUrlField.value;
const searchParams = new URLSearchParams({
response_type: 'code',
client_id: $.auth.data.clientId,
redirect_uri: redirectUri,
});
const url = `https://login.mailchimp.com/oauth2/authorize?${searchParams.toString()}`;
await $.auth.set({
url,
});
}

View File

@@ -0,0 +1,46 @@
import generateAuthUrl from './generate-auth-url.js';
import verifyCredentials from './verify-credentials.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/mailchimp/connections/add',
placeholder: null,
description:
'When asked to input a redirect URL in Mailchimp, 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,
};

View File

@@ -0,0 +1,8 @@
import getCurrentUser from '../common/get-current-user.js';
const isStillVerified = async ($) => {
const currentUser = await getCurrentUser($);
return !!currentUser.user_id;
};
export default isStillVerified;

View File

@@ -0,0 +1,40 @@
import getCurrentUser from '../common/get-current-user.js';
const verifyCredentials = async ($) => {
const oauthRedirectUrlField = $.app.auth.fields.find(
(field) => field.key == 'oAuthRedirectUrl'
);
const redirectUri = oauthRedirectUrlField.value;
const params = new URLSearchParams({
grant_type: 'authorization_code',
client_id: $.auth.data.clientId,
client_secret: $.auth.data.clientSecret,
redirect_uri: redirectUri,
code: $.auth.data.code,
});
const { data } = await $.http.post(
'https://login.mailchimp.com/oauth2/token',
params.toString()
);
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,
idToken: data.id_token,
expiresIn: data.expires_in,
refreshToken: data.refresh_token,
serverPrefix: currentUser.dc,
screenName: currentUser.login.login_name,
});
};
export default verifyCredentials;

View File

@@ -0,0 +1,12 @@
const addAuthHeader = ($, requestConfig) => {
if (
!requestConfig.additionalProperties?.skipAddingAuthHeader &&
$.auth.data?.accessToken
) {
requestConfig.headers.Authorization = `Bearer ${$.auth.data.accessToken}`;
}
return requestConfig;
};
export default addAuthHeader;

View File

@@ -0,0 +1,18 @@
const getCurrentUser = async ($) => {
const { data: currentUser } = await $.http.get(
'https://login.mailchimp.com/oauth2/metadata',
{
headers: {
Authorization: `OAuth ${$.auth.data.accessToken}`,
},
additionalProperties: {
skipAddingAuthHeader: true,
skipAddingBaseUrl: true,
},
}
);
return currentUser;
};
export default getCurrentUser;

View File

@@ -0,0 +1,10 @@
const setBaseUrl = ($, requestConfig) => {
const serverPrefix = $.auth.data.serverPrefix;
if (!requestConfig.additionalProperties?.skipAddingBaseUrl && serverPrefix) {
requestConfig.baseURL = `https://${serverPrefix}.api.mailchimp.com`;
}
return requestConfig;
};
export default setBaseUrl;

View File

@@ -0,0 +1,6 @@
import listAudiences from './list-audiences/index.js';
import listCampaigns from './list-campaigns/index.js';
import listTags from './list-segments-or-tags/index.js';
import listTemplates from './list-templates/index.js';
export default [listAudiences, listCampaigns, listTags, listTemplates];

View File

@@ -0,0 +1,40 @@
export default {
name: 'List audiences',
key: 'listAudiences',
async run($) {
const audiences = {
data: [],
};
let hasMore = false;
const params = {
sort_field: 'date_created',
sort_dir: 'DESC',
count: 1000,
offset: 0,
};
do {
const { data } = await $.http.get('/3.0/lists', { params });
params.offset = params.offset + params.count;
if (data?.lists) {
for (const audience of data.lists) {
audiences.data.push({
value: audience.id,
name: audience.name,
});
}
}
if (data.total_items > params.offset) {
hasMore = true;
} else {
hasMore = false;
}
} while (hasMore);
return audiences;
},
};

View File

@@ -0,0 +1,42 @@
export default {
name: 'List campaigns',
key: 'listCampaigns',
async run($) {
const campaigns = {
data: [],
};
let hasMore = false;
const audienceId = $.step.parameters.audienceId;
const params = {
list_id: audienceId,
sort_field: 'create_time',
sort_dir: 'DESC',
count: 1000,
offset: 0,
};
do {
const { data } = await $.http.get('/3.0/campaigns', { params });
params.offset = params.offset + params.count;
if (data?.campaigns) {
for (const campaign of data.campaigns) {
campaigns.data.push({
value: campaign.id,
name: campaign.settings.title || campaign.settings.subject_line || 'Unnamed campaign',
});
}
}
if (data.total_items > params.offset) {
hasMore = true;
} else {
hasMore = false;
}
} while (hasMore);
return campaigns;
},
};

View File

@@ -0,0 +1,44 @@
export default {
name: 'List segments or tags',
key: 'listSegmentsOrTags',
async run($) {
const segmentsOrTags = {
data: [],
};
const audienceId = $.step.parameters.audienceId;
if (!audienceId) {
return segmentsOrTags;
}
const {
data: { tags: allTags },
} = await $.http.get(`/3.0/lists/${audienceId}/tag-search`);
const {
data: { segments },
} = await $.http.get(`/3.0/lists/${audienceId}/segments`);
const mergedArray = [...allTags, ...segments].reduce(
(accumulator, current) => {
if (!accumulator.some((item) => item.id === current.id)) {
accumulator.push(current);
}
return accumulator;
},
[]
);
if (mergedArray?.length) {
for (const tagOrSegment of mergedArray) {
segmentsOrTags.data.push({
value: tagOrSegment.id,
name: tagOrSegment.name,
});
}
}
return segmentsOrTags;
},
};

View File

@@ -0,0 +1,30 @@
export default {
name: 'List templates',
key: 'listTemplates',
async run($) {
const templates = {
data: [],
};
const params = {
sort_field: 'date_created',
sort_dir: 'DESC',
count: 1000,
offset: 0,
};
const { data } = await $.http.get('/3.0/templates', { params });
if (data?.templates) {
for (const template of data.templates) {
templates.data.push({
value: template.id,
name: template.name,
});
}
}
return templates;
},
};

View File

@@ -0,0 +1,23 @@
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';
import actions from './actions/index.js';
export default defineApp({
name: 'Mailchimp',
key: 'mailchimp',
baseUrl: 'https://mailchimp.com',
apiBaseUrl: '',
iconUrl: '{BASE_URL}/apps/mailchimp/assets/favicon.svg',
authDocUrl: 'https://automatisch.io/docs/apps/mailchimp/connection',
primaryColor: '000000',
supportsConnections: true,
beforeRequest: [setBaseUrl, addAuthHeader],
auth,
triggers,
dynamicData,
actions,
});

View File

@@ -0,0 +1,101 @@
import defineTrigger from '../../../../helpers/define-trigger.js';
export default defineTrigger({
name: 'Email opened',
key: 'emailOpened',
pollInterval: 15,
description:
'Triggers when a recipient opens an email as part of a particular campaign.',
arguments: [
{
label: 'Audience',
key: 'audienceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listAudiences',
},
],
},
},
{
label: 'Campaign Type',
key: 'campaignType',
type: 'dropdown',
required: true,
description: '',
variables: true,
options: [
{
label: 'Campaign',
value: 'campaign',
},
],
},
{
label: 'Campaign',
key: 'campaignId',
type: 'dropdown',
required: true,
dependsOn: ['parameters.audienceId'],
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listCampaigns',
},
{
name: 'parameters.audienceId',
value: '{parameters.audienceId}',
},
],
},
},
],
async run($) {
const campaignId = $.step.parameters.campaignId;
let hasMore = false;
const params = {
count: 1000,
offset: 0,
};
do {
const { data } = await $.http.get(
`/3.0/reports/${campaignId}/open-details`,
{ params }
);
params.offset = params.offset + params.count;
if (data.members?.length) {
for (const member of data.members) {
$.pushTriggerItem({
raw: member,
meta: {
internalId: member.email_id,
},
});
}
}
if (data.total_items > params.offset) {
hasMore = true;
} else {
hasMore = false;
}
} while (hasMore);
},
});

View File

@@ -0,0 +1,5 @@
import emailOpened from './email-opened/index.js';
import newSubscribers from './new-subscribers/index.js';
import newUnsubscribers from './new-unsubscribers/index.js';
export default [emailOpened, newSubscribers, newUnsubscribers];

View File

@@ -0,0 +1,105 @@
import Crypto from 'crypto';
import defineTrigger from '../../../../helpers/define-trigger.js';
export default defineTrigger({
name: 'New subscribers',
key: 'newSubscribers',
type: 'webhook',
description: 'Triggers when a new subscriber is appended to an audience.',
arguments: [
{
label: 'Audience',
key: 'audienceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listAudiences',
},
],
},
},
],
async run($) {
const dataItem = {
raw: $.request.body,
meta: {
internalId: Crypto.randomUUID(),
},
};
$.pushTriggerItem(dataItem);
},
async testRun($) {
const audienceId = $.step.parameters.audienceId;
const computedWebhookEvent = {
data: {
id: Crypto.randomUUID(),
email: 'user@automatisch.io',
ip_opt: '127.0.0.1',
merges: {
EMAIL: 'user@automatisch.io',
FNAME: 'FNAME',
LNAME: 'LNAME',
PHONE: '',
ADDRESS: '',
BIRTHDAY: '',
},
web_id: Crypto.randomUUID(),
list_id: audienceId,
email_type: 'html',
},
type: 'subscribe',
fired_at: new Date().toLocaleString(),
};
const dataItem = {
raw: computedWebhookEvent,
meta: {
internalId: '',
},
};
$.pushTriggerItem(dataItem);
},
async registerHook($) {
const audienceId = $.step.parameters.audienceId;
const payload = {
url: $.webhookUrl,
events: {
subscribe: true,
},
sources: {
user: true,
admin: true,
api: true,
},
};
const response = await $.http.post(
`/3.0/lists/${audienceId}/webhooks`,
payload
);
await $.flow.setRemoteWebhookId(response.data.id);
},
async unregisterHook($) {
const audienceId = $.step.parameters.audienceId;
await $.http.delete(
`/3.0/lists/${audienceId}/webhooks/${$.flow.remoteWebhookId}`
);
},
});

View File

@@ -0,0 +1,108 @@
import Crypto from 'crypto';
import defineTrigger from '../../../../helpers/define-trigger.js';
export default defineTrigger({
name: 'New unsubscribers',
key: 'newUnsubscribers',
type: 'webhook',
description: 'Triggers when any existing subscriber opts out of an audience.',
arguments: [
{
label: 'Audience',
key: 'audienceId',
type: 'dropdown',
required: true,
description: '',
variables: true,
source: {
type: 'query',
name: 'getDynamicData',
arguments: [
{
name: 'key',
value: 'listAudiences',
},
],
},
},
],
async run($) {
const dataItem = {
raw: $.request.body,
meta: {
internalId: Crypto.randomUUID(),
},
};
$.pushTriggerItem(dataItem);
},
async testRun($) {
const audienceId = $.step.parameters.audienceId;
const computedWebhookEvent = {
data: {
id: Crypto.randomUUID(),
email: 'user@automatisch.io',
action: 'unsub',
ip_opt: '127.0.0.1',
merges: {
EMAIL: 'user@automatisch.io',
FNAME: 'FNAME',
LNAME: 'LNAME',
PHONE: '',
ADDRESS: '',
BIRTHDAY: '',
},
reason: 'manual',
web_id: Crypto.randomUUID(),
list_id: audienceId,
email_type: 'html',
campaign_id: Crypto.randomUUID(),
},
type: 'unsubscribe',
fired_at: new Date().toLocaleString(),
};
const dataItem = {
raw: computedWebhookEvent,
meta: {
internalId: '',
},
};
$.pushTriggerItem(dataItem);
},
async registerHook($) {
const audienceId = $.step.parameters.audienceId;
const payload = {
url: $.webhookUrl,
events: {
unsubscribe: true,
},
sources: {
user: true,
admin: true,
api: true,
},
};
const response = await $.http.post(
`/3.0/lists/${audienceId}/webhooks`,
payload
);
await $.flow.setRemoteWebhookId(response.data.id);
},
async unregisterHook($) {
const audienceId = $.step.parameters.audienceId;
await $.http.delete(
`/3.0/lists/${audienceId}/webhooks/${$.flow.remoteWebhookId}`
);
},
});

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 9.0 KiB

View File

@@ -0,0 +1,33 @@
import verifyCredentials from './verify-credentials.js';
import isStillVerified from './is-still-verified.js';
export default {
fields: [
{
key: 'screenName',
label: 'Screen Name',
type: 'string',
required: true,
readOnly: false,
value: null,
placeholder: null,
description:
'Screen name of your connection to be used on Automatisch UI.',
clickToCopy: false,
},
{
key: 'apiKey',
label: 'API Key',
type: 'string',
required: true,
readOnly: false,
value: null,
placeholder: null,
description: 'MailerLite API key of your account.',
clickToCopy: false,
},
],
verifyCredentials,
isStillVerified,
};

View File

@@ -0,0 +1,8 @@
import verifyCredentials from './verify-credentials.js';
const isStillVerified = async ($) => {
await verifyCredentials($);
return true;
};
export default isStillVerified;

View File

@@ -0,0 +1,10 @@
const verifyCredentials = async ($) => {
await $.http.get('/campaigns');
await $.auth.set({
screenName: $.auth.data.screenName,
apiKey: $.auth.data.apiKey,
});
};
export default verifyCredentials;

View File

@@ -0,0 +1,9 @@
const addAuthHeader = ($, requestConfig) => {
if ($.auth.data?.apiKey) {
requestConfig.headers.Authorization = `Bearer ${$.auth.data.apiKey}`;
}
return requestConfig;
};
export default addAuthHeader;

Some files were not shown because too many files have changed in this diff Show More