workflow.js 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. import { defineStore } from 'pinia';
  2. import { nanoid } from 'nanoid';
  3. import defu from 'defu';
  4. import deepmerge from 'lodash.merge';
  5. import browser from 'webextension-polyfill';
  6. import dayjs from 'dayjs';
  7. import { fetchApi } from '@/utils/api';
  8. import { tasks } from '@/utils/shared';
  9. import firstWorkflows from '@/utils/firstWorkflows';
  10. import { cleanWorkflowTriggers } from '@/utils/workflowTrigger';
  11. import { parseJSON } from '@/utils/helper';
  12. import { useUserStore } from './user';
  13. const defaultWorkflow = (data = null, options = {}) => {
  14. let workflowData = {
  15. id: nanoid(),
  16. name: '',
  17. icon: 'riGlobalLine',
  18. folderId: null,
  19. connectedTable: null,
  20. drawflow: {
  21. edges: [],
  22. position: { zoom: 1 },
  23. nodes: [
  24. {
  25. position: {
  26. x: 100,
  27. y: window.innerHeight / 2,
  28. },
  29. id: nanoid(),
  30. label: 'trigger',
  31. data: tasks.trigger.data,
  32. type: tasks.trigger.component,
  33. },
  34. ],
  35. },
  36. table: [],
  37. dataColumns: [],
  38. description: '',
  39. trigger: null,
  40. createdAt: Date.now(),
  41. updatedAt: Date.now(),
  42. isDisabled: false,
  43. settings: {
  44. publicId: '',
  45. blockDelay: 0,
  46. saveLog: true,
  47. debugMode: false,
  48. restartTimes: 3,
  49. notification: true,
  50. reuseLastState: false,
  51. inputAutocomplete: true,
  52. onError: 'stop-workflow',
  53. executedBlockOnWeb: false,
  54. insertDefaultColumn: true,
  55. defaultColumnName: 'column',
  56. },
  57. version: browser.runtime.getManifest().version,
  58. globalData: '{\n\t"key": "value"\n}',
  59. };
  60. if (data) {
  61. if (options.duplicateId && data.id) {
  62. delete workflowData.id;
  63. }
  64. if (data.drawflow?.nodes?.length > 0) {
  65. workflowData.drawflow.nodes = [];
  66. }
  67. workflowData = defu(data, workflowData);
  68. }
  69. return workflowData;
  70. };
  71. function convertWorkflowsToObject(workflows) {
  72. if (Array.isArray(workflows)) {
  73. return workflows.reduce((acc, workflow) => {
  74. acc[workflow.id] = workflow;
  75. return acc;
  76. }, {});
  77. }
  78. return workflows;
  79. }
  80. export const useWorkflowStore = defineStore('workflow', {
  81. storageMap: {
  82. workflows: 'workflows',
  83. },
  84. state: () => ({
  85. states: [],
  86. workflows: {},
  87. retrieved: false,
  88. }),
  89. getters: {
  90. getById: (state) => (id) => state.workflows[id],
  91. getWorkflows: (state) => Object.values(state.workflows),
  92. getWorkflowStates: (state) => (id) =>
  93. state.states.filter(({ workflowId }) => workflowId === id),
  94. },
  95. actions: {
  96. async loadData() {
  97. const { workflows, isFirstTime } = await browser.storage.local.get([
  98. 'workflows',
  99. 'isFirstTime',
  100. ]);
  101. let localWorkflows = workflows;
  102. if (isFirstTime) {
  103. localWorkflows = firstWorkflows.map((workflow) =>
  104. defaultWorkflow(workflow)
  105. );
  106. await browser.storage.local.set({ isFirstTime: false });
  107. }
  108. this.workflows = convertWorkflowsToObject(localWorkflows);
  109. const storedStates = localStorage.getItem('workflowState') || '{}';
  110. const states = parseJSON(storedStates, {});
  111. this.states = Object.values(states).filter(
  112. ({ isDestroyed }) => !isDestroyed
  113. );
  114. if (isFirstTime) {
  115. await this.saveToStorage('workflows');
  116. }
  117. this.retrieved = true;
  118. },
  119. async insert(data = {}, options = {}) {
  120. const insertedWorkflows = {};
  121. if (Array.isArray(data)) {
  122. data.forEach((item) => {
  123. if (!options.duplicateId) {
  124. delete item.id;
  125. }
  126. const workflow = defaultWorkflow(item, options);
  127. this.workflows[workflow.id] = workflow;
  128. insertedWorkflows[workflow.id] = workflow;
  129. });
  130. } else {
  131. if (!options.duplicateId) {
  132. delete data.id;
  133. }
  134. const workflow = defaultWorkflow(data, options);
  135. this.workflows[workflow.id] = workflow;
  136. insertedWorkflows[workflow.id] = workflow;
  137. }
  138. await this.saveToStorage('workflows');
  139. return insertedWorkflows;
  140. },
  141. async update({ id, data = {}, deep = false }) {
  142. const isFunction = typeof id === 'function';
  143. if (!isFunction && !this.workflows[id]) return null;
  144. const updatedWorkflows = {};
  145. const workflowUpdater = (workflowId) => {
  146. if (deep) {
  147. this.workflows[workflowId] = deepmerge(
  148. this.workflows[workflowId],
  149. data
  150. );
  151. } else {
  152. Object.assign(this.workflows[workflowId], data);
  153. }
  154. this.workflows[workflowId].updatedAt = Date.now();
  155. updatedWorkflows[workflowId] = this.workflows[workflowId];
  156. };
  157. if (isFunction) {
  158. this.getWorkflows.forEach((workflow) => {
  159. const isMatch = id(workflow) ?? false;
  160. if (isMatch) workflowUpdater(workflow.id);
  161. });
  162. } else {
  163. workflowUpdater(id);
  164. }
  165. await this.saveToStorage('workflows');
  166. return updatedWorkflows;
  167. },
  168. async insertOrUpdate(data = [], { checkUpdateDate = false } = {}) {
  169. const insertedData = {};
  170. data.forEach((item) => {
  171. const currentWorkflow = this.workflows[item.id];
  172. if (currentWorkflow) {
  173. let insert = true;
  174. if (checkUpdateDate && currentWorkflow.createdAt && item.updatedAt) {
  175. insert = dayjs(currentWorkflow.updatedAt).isBefore(item.updatedAt);
  176. }
  177. if (insert) {
  178. Object.assign(this.workflows[item.id], item);
  179. insertedData[item.id] = this.workflows[item.id];
  180. }
  181. } else {
  182. const workflow = defaultWorkflow(item);
  183. this.workflows[workflow.id] = workflow;
  184. insertedData[workflow.id] = workflow;
  185. }
  186. });
  187. await this.saveToStorage('workflows');
  188. return insertedData;
  189. },
  190. async delete(id) {
  191. if (Array.isArray(id)) {
  192. id.forEach((workflowId) => {
  193. delete this.workflows[workflowId];
  194. });
  195. } else {
  196. delete this.workflows[id];
  197. }
  198. await cleanWorkflowTriggers(id);
  199. const userStore = useUserStore();
  200. const hostedWorkflow = userStore.hostedWorkflows[id];
  201. const backupIndex = userStore.backupIds.indexOf(id);
  202. if (hostedWorkflow || backupIndex !== -1) {
  203. const response = await fetchApi(`/me/workflows?id=${id}`, {
  204. method: 'DELETE',
  205. });
  206. const result = await response.json();
  207. if (!response.ok) {
  208. throw new Error(result.message);
  209. }
  210. if (backupIndex !== -1) {
  211. userStore.backupIds.splice(backupIndex, 1);
  212. await browser.storage.local.set({ backupIds: userStore.backupIds });
  213. }
  214. }
  215. await browser.storage.local.remove(`state:${id}`);
  216. await this.saveToStorage('workflows');
  217. return id;
  218. },
  219. },
  220. });