@kaspernj/api-maker 1.0.2121 → 1.0.2123

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,192 @@
1
+ import CustomError from "./custom-error.js";
2
+ import Logger from "./logger.js";
3
+ import channelsConsumer from "./channels-consumer.js";
4
+ const logger = new Logger({ name: "ApiMaker / WebsocketRequestClient" });
5
+ const shared = {};
6
+ /** Shared websocket request client for ApiMaker command/service execution. */
7
+ export default class ApiMakerWebsocketRequestClient {
8
+ /** @returns {ApiMakerWebsocketRequestClient} */
9
+ static current() {
10
+ if (!shared.currentApiMakerWebsocketRequestClient) {
11
+ shared.currentApiMakerWebsocketRequestClient = new ApiMakerWebsocketRequestClient();
12
+ }
13
+ return shared.currentApiMakerWebsocketRequestClient;
14
+ }
15
+ /** Constructor. */
16
+ constructor() {
17
+ this.currentRequestId = 1;
18
+ this.pendingRequests = {};
19
+ this.pendingRequestsByFingerprint = {};
20
+ this.responseCache = {};
21
+ this.subscriptionState = "new";
22
+ }
23
+ /**
24
+ * @param {object} args
25
+ * @param {boolean} [args.cacheResponse]
26
+ * @param {Record<string, any>} [args.global]
27
+ * @param {(value: string) => void} [args.onLog]
28
+ * @param {(value: Record<string, any>) => void} [args.onProgress]
29
+ * @param {(value: Record<string, any>) => void} [args.onReceived]
30
+ * @param {Record<string, any>} args.request
31
+ * @returns {Promise<Record<string, any>>}
32
+ */
33
+ perform({ cacheResponse, global, onLog, onProgress, onReceived, request }) {
34
+ const fingerprint = JSON.stringify({ global, request });
35
+ if (cacheResponse && this.responseCache[fingerprint]) {
36
+ return Promise.resolve(this.responseCache[fingerprint]);
37
+ }
38
+ if (this.pendingRequestsByFingerprint[fingerprint]) {
39
+ const pendingRequestData = this.pendingRequestsByFingerprint[fingerprint];
40
+ const pendingRequest = this.pendingRequests[pendingRequestData.requestId];
41
+ if (onLog)
42
+ pendingRequest?.onLogCallbacks.push(onLog);
43
+ if (onProgress)
44
+ pendingRequest?.onProgressCallbacks.push(onProgress);
45
+ if (onReceived)
46
+ pendingRequest?.onReceivedCallbacks.push(onReceived);
47
+ return pendingRequestData.promise;
48
+ }
49
+ const promise = new Promise((resolve, reject) => {
50
+ const requestId = this.currentRequestId;
51
+ this.currentRequestId += 1;
52
+ this.pendingRequests[requestId] = {
53
+ cacheResponse,
54
+ fingerprint,
55
+ onLogCallbacks: onLog ? [onLog] : [],
56
+ onProgressCallbacks: onProgress ? [onProgress] : [],
57
+ onReceivedCallbacks: onReceived ? [onReceived] : [],
58
+ reject,
59
+ resolve
60
+ };
61
+ this.waitForSubscription().then(() => {
62
+ this.ensureSubscription().perform("execute", {
63
+ cache_response: cacheResponse,
64
+ global,
65
+ request,
66
+ request_id: requestId
67
+ });
68
+ })
69
+ .catch((error) => {
70
+ delete this.pendingRequests[requestId];
71
+ reject(error);
72
+ });
73
+ });
74
+ this.pendingRequestsByFingerprint[fingerprint] = { promise, requestId: this.currentRequestId - 1 };
75
+ promise.then(() => {
76
+ delete this.pendingRequestsByFingerprint[fingerprint];
77
+ }, () => {
78
+ delete this.pendingRequestsByFingerprint[fingerprint];
79
+ });
80
+ return promise;
81
+ }
82
+ /** @returns {any} */
83
+ ensureSubscription() {
84
+ if (!this.subscription) {
85
+ logger.debug("Creating websocket request subscription");
86
+ this.subscriptionState = "connecting";
87
+ this.resetSubscriptionReadyPromise();
88
+ this.subscription = channelsConsumer().subscriptions.create({ channel: "ApiMaker::RequestsChannel" }, {
89
+ connected: this.onConnected,
90
+ disconnected: this.onDisconnected,
91
+ received: this.onReceived,
92
+ rejected: this.onRejected
93
+ });
94
+ }
95
+ return this.subscription;
96
+ }
97
+ /** @returns {Promise<void>} */
98
+ waitForSubscription() {
99
+ this.ensureSubscription();
100
+ if (this.subscriptionState == "connected") {
101
+ return Promise.resolve();
102
+ }
103
+ if (!this.subscriptionReadyPromise) {
104
+ this.resetSubscriptionReadyPromise();
105
+ }
106
+ return this.subscriptionReadyPromise;
107
+ }
108
+ /** @returns {void} */
109
+ resetSubscriptionReadyPromise() {
110
+ this.subscriptionReadyPromise = new Promise((resolve, reject) => {
111
+ this.resolveSubscriptionReadyPromise = resolve;
112
+ this.rejectSubscriptionReadyPromise = reject;
113
+ });
114
+ }
115
+ /** @returns {void} */
116
+ onConnected = () => {
117
+ logger.debug("Websocket request subscription connected");
118
+ this.subscriptionState = "connected";
119
+ this.resolveSubscriptionReadyPromise?.();
120
+ };
121
+ /** @returns {void} */
122
+ onDisconnected = () => {
123
+ logger.debug("Websocket request subscription disconnected");
124
+ this.rejectPendingRequests(new Error("Websocket request subscription disconnected"));
125
+ this.subscriptionState = "disconnected";
126
+ this.subscription = null;
127
+ this.resetSubscriptionReadyPromise();
128
+ };
129
+ /** @returns {void} */
130
+ onRejected = () => {
131
+ const error = new Error("Websocket request subscription was rejected");
132
+ logger.error(error);
133
+ this.rejectPendingRequests(error);
134
+ this.subscriptionState = "rejected";
135
+ this.subscription = null;
136
+ this.rejectSubscriptionReadyPromise?.(error);
137
+ this.resetSubscriptionReadyPromise();
138
+ };
139
+ /**
140
+ * @param {Record<string, any>} data
141
+ * @returns {void}
142
+ */
143
+ onReceived = (data) => {
144
+ const pendingRequest = this.pendingRequests[data.request_id];
145
+ if (!pendingRequest) {
146
+ logger.debug(() => ["Ignoring websocket response without a pending request", { data }]);
147
+ return;
148
+ }
149
+ if (data.type == "api_maker_command_log") {
150
+ pendingRequest.onLogCallbacks.forEach((callback) => callback(data.message));
151
+ }
152
+ else if (data.type == "api_maker_command_progress") {
153
+ const progressData = {
154
+ count: data.count,
155
+ progress: data.progress,
156
+ total: data.total
157
+ };
158
+ pendingRequest.onProgressCallbacks.forEach((callback) => callback(progressData));
159
+ }
160
+ else if (data.type == "api_maker_request_received") {
161
+ pendingRequest.onReceivedCallbacks.forEach((callback) => callback(data));
162
+ }
163
+ else if (data.type == "api_maker_request_response") {
164
+ delete this.pendingRequests[data.request_id];
165
+ if (pendingRequest.cacheResponse) {
166
+ this.responseCache[pendingRequest.fingerprint] = data.response;
167
+ }
168
+ pendingRequest.resolve(data.response);
169
+ }
170
+ else if (data.type == "api_maker_request_error") {
171
+ delete this.pendingRequests[data.request_id];
172
+ pendingRequest.reject(new CustomError("Websocket request failed", { response: data.response }));
173
+ }
174
+ else {
175
+ delete this.pendingRequests[data.request_id];
176
+ pendingRequest.reject(new Error(`Unknown websocket request response type: ${data.type}`));
177
+ }
178
+ };
179
+ /**
180
+ * @param {Error} error
181
+ * @returns {void}
182
+ */
183
+ rejectPendingRequests(error) {
184
+ const pendingRequests = this.pendingRequests;
185
+ this.pendingRequests = {};
186
+ this.pendingRequestsByFingerprint = {};
187
+ Object.values(pendingRequests).forEach((pendingRequest) => {
188
+ queueMicrotask(() => pendingRequest.reject(error));
189
+ });
190
+ }
191
+ }
192
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-request-client.js","sourceRoot":"/src/","sources":["websocket-request-client.js"],"names":[],"mappings":"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","sourcesContent":["import CustomError from \"./custom-error.js\"\nimport Logger from \"./logger.js\"\nimport channelsConsumer from \"./channels-consumer.js\"\n\nconst logger = new Logger({name: \"ApiMaker / WebsocketRequestClient\"})\nconst shared = {}\n\n/** Shared websocket request client for ApiMaker command/service execution. */\nexport default class ApiMakerWebsocketRequestClient {\n  /** @returns {ApiMakerWebsocketRequestClient} */\n  static current () {\n    if (!shared.currentApiMakerWebsocketRequestClient) {\n      shared.currentApiMakerWebsocketRequestClient = new ApiMakerWebsocketRequestClient()\n    }\n\n    return shared.currentApiMakerWebsocketRequestClient\n  }\n\n  /** Constructor. */\n  constructor () {\n    this.currentRequestId = 1\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n    this.responseCache = {}\n    this.subscriptionState = \"new\"\n  }\n\n  /**\n   * @param {object} args\n   * @param {boolean} [args.cacheResponse]\n   * @param {Record<string, any>} [args.global]\n   * @param {(value: string) => void} [args.onLog]\n   * @param {(value: Record<string, any>) => void} [args.onProgress]\n   * @param {(value: Record<string, any>) => void} [args.onReceived]\n   * @param {Record<string, any>} args.request\n   * @returns {Promise<Record<string, any>>}\n   */\n  perform ({cacheResponse, global, onLog, onProgress, onReceived, request}) {\n    const fingerprint = JSON.stringify({global, request})\n\n    if (cacheResponse && this.responseCache[fingerprint]) {\n      return Promise.resolve(this.responseCache[fingerprint])\n    }\n\n    if (this.pendingRequestsByFingerprint[fingerprint]) {\n      const pendingRequestData = this.pendingRequestsByFingerprint[fingerprint]\n      const pendingRequest = this.pendingRequests[pendingRequestData.requestId]\n\n      if (onLog) pendingRequest?.onLogCallbacks.push(onLog)\n      if (onProgress) pendingRequest?.onProgressCallbacks.push(onProgress)\n      if (onReceived) pendingRequest?.onReceivedCallbacks.push(onReceived)\n\n      return pendingRequestData.promise\n    }\n\n    const promise = new Promise((resolve, reject) => {\n      const requestId = this.currentRequestId\n      this.currentRequestId += 1\n\n      this.pendingRequests[requestId] = {\n        cacheResponse,\n        fingerprint,\n        onLogCallbacks: onLog ? [onLog] : [],\n        onProgressCallbacks: onProgress ? [onProgress] : [],\n        onReceivedCallbacks: onReceived ? [onReceived] : [],\n        reject,\n        resolve\n      }\n\n      this.waitForSubscription().then(() => {\n        this.ensureSubscription().perform(\"execute\", {\n          cache_response: cacheResponse,\n          global,\n          request,\n          request_id: requestId\n        })\n      })\n        .catch((error) => {\n          delete this.pendingRequests[requestId]\n          reject(error)\n        })\n    })\n\n    this.pendingRequestsByFingerprint[fingerprint] = {promise, requestId: this.currentRequestId - 1}\n\n    promise.then(() => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    }, () => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    })\n\n    return promise\n  }\n\n  /** @returns {any} */\n  ensureSubscription () {\n    if (!this.subscription) {\n      logger.debug(\"Creating websocket request subscription\")\n      this.subscriptionState = \"connecting\"\n      this.resetSubscriptionReadyPromise()\n\n      this.subscription = channelsConsumer().subscriptions.create(\n        {channel: \"ApiMaker::RequestsChannel\"},\n        {\n          connected: this.onConnected,\n          disconnected: this.onDisconnected,\n          received: this.onReceived,\n          rejected: this.onRejected\n        }\n      )\n    }\n\n    return this.subscription\n  }\n\n  /** @returns {Promise<void>} */\n  waitForSubscription () {\n    this.ensureSubscription()\n\n    if (this.subscriptionState == \"connected\") {\n      return Promise.resolve()\n    }\n\n    if (!this.subscriptionReadyPromise) {\n      this.resetSubscriptionReadyPromise()\n    }\n\n    return this.subscriptionReadyPromise\n  }\n\n  /** @returns {void} */\n  resetSubscriptionReadyPromise () {\n    this.subscriptionReadyPromise = new Promise((resolve, reject) => {\n      this.resolveSubscriptionReadyPromise = resolve\n      this.rejectSubscriptionReadyPromise = reject\n    })\n  }\n\n  /** @returns {void} */\n  onConnected = () => {\n    logger.debug(\"Websocket request subscription connected\")\n    this.subscriptionState = \"connected\"\n    this.resolveSubscriptionReadyPromise?.()\n  }\n\n  /** @returns {void} */\n  onDisconnected = () => {\n    logger.debug(\"Websocket request subscription disconnected\")\n    this.rejectPendingRequests(new Error(\"Websocket request subscription disconnected\"))\n    this.subscriptionState = \"disconnected\"\n    this.subscription = null\n    this.resetSubscriptionReadyPromise()\n  }\n\n  /** @returns {void} */\n  onRejected = () => {\n    const error = new Error(\"Websocket request subscription was rejected\")\n\n    logger.error(error)\n    this.rejectPendingRequests(error)\n    this.subscriptionState = \"rejected\"\n    this.subscription = null\n    this.rejectSubscriptionReadyPromise?.(error)\n    this.resetSubscriptionReadyPromise()\n  }\n\n  /**\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  onReceived = (data) => {\n    const pendingRequest = this.pendingRequests[data.request_id]\n\n    if (!pendingRequest) {\n      logger.debug(() => [\"Ignoring websocket response without a pending request\", {data}])\n      return\n    }\n\n    if (data.type == \"api_maker_command_log\") {\n      pendingRequest.onLogCallbacks.forEach((callback) => callback(data.message))\n    } else if (data.type == \"api_maker_command_progress\") {\n      const progressData = {\n        count: data.count,\n        progress: data.progress,\n        total: data.total\n      }\n\n      pendingRequest.onProgressCallbacks.forEach((callback) => callback(progressData))\n    } else if (data.type == \"api_maker_request_received\") {\n      pendingRequest.onReceivedCallbacks.forEach((callback) => callback(data))\n    } else if (data.type == \"api_maker_request_response\") {\n      delete this.pendingRequests[data.request_id]\n\n      if (pendingRequest.cacheResponse) {\n        this.responseCache[pendingRequest.fingerprint] = data.response\n      }\n\n      pendingRequest.resolve(data.response)\n    } else if (data.type == \"api_maker_request_error\") {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new CustomError(\"Websocket request failed\", {response: data.response}))\n    } else {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new Error(`Unknown websocket request response type: ${data.type}`))\n    }\n  }\n\n  /**\n   * @param {Error} error\n   * @returns {void}\n   */\n  rejectPendingRequests (error) {\n    const pendingRequests = this.pendingRequests\n\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n\n    Object.values(pendingRequests).forEach((pendingRequest) => {\n      queueMicrotask(() => pendingRequest.reject(error))\n    })\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@kaspernj/api-maker",
3
3
  "type": "module",
4
- "version": "1.0.2121",
4
+ "version": "1.0.2123",
5
5
  "description": "My new module",
6
6
  "files": [
7
7
  "build/**"
@@ -41,6 +41,7 @@
41
41
  "license": "MIT",
42
42
  "homepage": "https://github.com/kaspernj/api_maker#readme",
43
43
  "dependencies": {
44
+ "classnames": "^2.5.1",
44
45
  "clone-deep": ">= 4.0.1",
45
46
  "debounce": ">= 2.0.0",
46
47
  "diggerize": ">= 1.0.10",
@@ -57,6 +58,7 @@
57
58
  "numberable": ">= 1.0.0",
58
59
  "object-to-formdata": ">= 4.1.0",
59
60
  "on-location-changed": ">= 1.0.18",
61
+ "prop-types-exact": "^1.2.7",
60
62
  "qs": ">= 6.9.3",
61
63
  "replaceall": ">= 0.1.6",
62
64
  "responsive-breakpoints": ">= 0.1.3",
@@ -71,22 +73,26 @@
71
73
  "ya-use-event-listener": ">= 0.1.1"
72
74
  },
73
75
  "devDependencies": {
74
- "@babel/eslint-parser": "^7.16.3",
76
+ "@eslint/compat": "^2.0.3",
75
77
  "@babel/preset-env": "^7.12.11",
76
78
  "@babel/preset-react": "^7.26.3",
77
79
  "@jest/globals": "~30.3.0",
78
80
  "@types/react": "~19.0.10",
79
81
  "babel-jest": "~30.3.0",
80
- "eslint": "^9.23.0",
82
+ "eslint": "^10.0.3",
81
83
  "eslint-config-expo": "~9.2.0",
82
- "eslint-plugin-jest": "^29.0.1",
83
- "eslint-plugin-jsdoc": "^62.0.0",
84
+ "eslint-plugin-jest": "^29.15.0",
85
+ "eslint-plugin-jsdoc": "^62.8.0",
84
86
  "eslint-plugin-react": "^7.23.2",
85
87
  "expo": "~53.0.27",
86
88
  "expo-module-scripts": "^55.0.2",
89
+ "flash-notifications": "^0.0.42",
90
+ "history": "^5.3.0",
91
+ "i18n-on-steroids": "^1.0.21",
87
92
  "jest": "~29.7.0",
88
93
  "react": "19.0.0",
89
94
  "react-native": "0.79.6",
95
+ "react-native-vector-icons": "^10.3.0",
90
96
  "typescript": "~5.8.3",
91
97
  "velocious": "^1.0.175"
92
98
  },
@@ -107,8 +113,12 @@
107
113
  "metro-resolver": "^0.82.0"
108
114
  },
109
115
  "overrides": {
116
+ "@typescript-eslint/eslint-plugin": "^8.57.1",
117
+ "@typescript-eslint/parser": "^8.57.1",
118
+ "@typescript-eslint/utils": "^8.57.1",
110
119
  "@expo/config-plugins": "~10.1.1",
111
120
  "@expo/metro-config": "~0.20.18",
121
+ "eslint-plugin-expo": "^1.0.0",
112
122
  "metro": "^0.82.0",
113
123
  "metro-config": "^0.82.0",
114
124
  "metro-resolver": "^0.82.0"