@ibm-aspera/sdk 0.2.2

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.
Files changed (68) hide show
  1. package/.editorconfig +13 -0
  2. package/.eslintrc.js +128 -0
  3. package/.github/CODE_OF_CONDUCT.md +128 -0
  4. package/.github/CONTRIBUTING.md +147 -0
  5. package/.github/workflows/ci.yml +36 -0
  6. package/.github/workflows/documentation.yml +43 -0
  7. package/.github/workflows/npm_upload.yml +30 -0
  8. package/.husky/pre-commit +4 -0
  9. package/CHANGELOG.md +124 -0
  10. package/LICENSE +201 -0
  11. package/README.md +25 -0
  12. package/dist/commonjs/app/core.d.ts +219 -0
  13. package/dist/commonjs/app/core.js +546 -0
  14. package/dist/commonjs/app/installer.d.ts +9 -0
  15. package/dist/commonjs/app/installer.js +50 -0
  16. package/dist/commonjs/constants/constants.d.ts +6 -0
  17. package/dist/commonjs/constants/constants.js +9 -0
  18. package/dist/commonjs/constants/messages.d.ts +29 -0
  19. package/dist/commonjs/constants/messages.js +32 -0
  20. package/dist/commonjs/helpers/client/client.d.ts +5 -0
  21. package/dist/commonjs/helpers/client/client.js +7 -0
  22. package/dist/commonjs/helpers/client/http-client.d.ts +42 -0
  23. package/dist/commonjs/helpers/client/http-client.js +84 -0
  24. package/dist/commonjs/helpers/client/safari-client.d.ts +99 -0
  25. package/dist/commonjs/helpers/client/safari-client.js +252 -0
  26. package/dist/commonjs/helpers/helpers.d.ts +84 -0
  27. package/dist/commonjs/helpers/helpers.js +197 -0
  28. package/dist/commonjs/helpers/http.d.ts +16 -0
  29. package/dist/commonjs/helpers/http.js +42 -0
  30. package/dist/commonjs/helpers/ws.d.ts +62 -0
  31. package/dist/commonjs/helpers/ws.js +182 -0
  32. package/dist/commonjs/index.d.ts +41 -0
  33. package/dist/commonjs/index.js +99 -0
  34. package/dist/commonjs/models/aspera-sdk.model.d.ts +213 -0
  35. package/dist/commonjs/models/aspera-sdk.model.js +288 -0
  36. package/dist/commonjs/models/models.d.ts +640 -0
  37. package/dist/commonjs/models/models.js +2 -0
  38. package/dist/js/aspera-sdk.js +3 -0
  39. package/dist/js/aspera-sdk.js.LICENSE.txt +7 -0
  40. package/dist/js/aspera-sdk.js.map +1 -0
  41. package/docs/DEVELOPMENT.md +38 -0
  42. package/jest.config.js +15 -0
  43. package/jest.setup.js +0 -0
  44. package/package.json +50 -0
  45. package/src/app/core.ts +610 -0
  46. package/src/app/installer.ts +53 -0
  47. package/src/constants/constants.ts +16 -0
  48. package/src/constants/messages.ts +29 -0
  49. package/src/helpers/client/client.ts +11 -0
  50. package/src/helpers/client/http-client.ts +92 -0
  51. package/src/helpers/client/safari-client.ts +318 -0
  52. package/src/helpers/helpers.ts +200 -0
  53. package/src/helpers/http.ts +39 -0
  54. package/src/helpers/ws.ts +215 -0
  55. package/src/index.html +404 -0
  56. package/src/index.ts +104 -0
  57. package/src/models/aspera-sdk.model.ts +360 -0
  58. package/src/models/models.ts +669 -0
  59. package/tests/client.spec.ts +52 -0
  60. package/tests/core.spec.ts +13 -0
  61. package/tests/helpers.spec.ts +124 -0
  62. package/tests/http.spec.ts +14 -0
  63. package/tests/installer.spec.ts +135 -0
  64. package/tests/mocks.ts +11 -0
  65. package/tsconfig.json +10 -0
  66. package/tsconfig.module.json +15 -0
  67. package/typedoc.js +17 -0
  68. package/webpack.config.js +53 -0
package/src/index.ts ADDED
@@ -0,0 +1,104 @@
1
+ import {AsperaSdk} from './models/aspera-sdk.model';
2
+ import {createDropzone, deregisterActivityCallback, deregisterRemovedCallback, deregisterSafariExtensionStatusCallback, deregisterStatusCallback, getAllTransfers, getInfo, getTransfer, init, initDragDrop, modifyTransfer, registerActivityCallback, registerRemovedCallback, registerSafariExtensionStatusCallback, registerStatusCallback, removeDropzone, removeTransfer, resumeTransfer, setBranding, showDirectory, showPreferences, showSelectFileDialog, showSelectFolderDialog, startTransfer, stopTransfer, testConnection,} from './app/core';
3
+ import {getInstallerInfo} from './app/installer';
4
+ import {isSafari} from './helpers/helpers';
5
+
6
+ export const asperaSdk: AsperaSdk = new AsperaSdk();
7
+
8
+ asperaSdk.init = init;
9
+ asperaSdk.testConnection = testConnection;
10
+ asperaSdk.startTransfer = startTransfer;
11
+ asperaSdk.registerActivityCallback = registerActivityCallback;
12
+ asperaSdk.deregisterActivityCallback = deregisterActivityCallback;
13
+ asperaSdk.removeTransfer = removeTransfer;
14
+ asperaSdk.showDirectory = showDirectory;
15
+ asperaSdk.stopTransfer = stopTransfer;
16
+ asperaSdk.resumeTransfer = resumeTransfer;
17
+ asperaSdk.getAllTransfers = getAllTransfers;
18
+ asperaSdk.getTransfer = getTransfer;
19
+ asperaSdk.registerRemovedCallback = registerRemovedCallback;
20
+ asperaSdk.deregisterRemovedCallback = deregisterRemovedCallback;
21
+ asperaSdk.showSelectFileDialog = showSelectFileDialog;
22
+ asperaSdk.showSelectFolderDialog = showSelectFolderDialog;
23
+ asperaSdk.showPreferences = showPreferences;
24
+ asperaSdk.modifyTransfer = modifyTransfer;
25
+ asperaSdk.createDropzone = createDropzone;
26
+ asperaSdk.removeDropzone = removeDropzone;
27
+ asperaSdk.getInstallerInfo = getInstallerInfo;
28
+ asperaSdk.registerStatusCallback = registerStatusCallback;
29
+ asperaSdk.deregisterStatusCallback = deregisterStatusCallback;
30
+ asperaSdk.registerSafariExtensionStatusCallback = registerSafariExtensionStatusCallback;
31
+ asperaSdk.deregisterSafariExtensionStatusCallback = deregisterSafariExtensionStatusCallback;
32
+ asperaSdk.initDragDrop = initDragDrop;
33
+ asperaSdk.setBranding = setBranding;
34
+ asperaSdk.getInfo = getInfo;
35
+ asperaSdk.isSafari = isSafari;
36
+
37
+ const launch = asperaSdk.globals.launch;
38
+ asperaSdk.launch = launch;
39
+
40
+ if (typeof (<any>window) === 'object') {
41
+ (<any>window).asperaSdk = asperaSdk;
42
+ }
43
+
44
+ export {
45
+ isSafari,
46
+ init,
47
+ testConnection,
48
+ startTransfer,
49
+ launch,
50
+ registerActivityCallback,
51
+ deregisterActivityCallback,
52
+ removeTransfer,
53
+ showDirectory,
54
+ stopTransfer,
55
+ resumeTransfer,
56
+ getAllTransfers,
57
+ getTransfer,
58
+ registerRemovedCallback,
59
+ deregisterRemovedCallback,
60
+ showSelectFileDialog,
61
+ showSelectFolderDialog,
62
+ showPreferences,
63
+ modifyTransfer,
64
+ createDropzone,
65
+ removeDropzone,
66
+ getInstallerInfo,
67
+ registerStatusCallback,
68
+ deregisterStatusCallback,
69
+ registerSafariExtensionStatusCallback,
70
+ deregisterSafariExtensionStatusCallback,
71
+ setBranding,
72
+ getInfo,
73
+ };
74
+
75
+ export default {
76
+ isSafari,
77
+ init,
78
+ testConnection,
79
+ startTransfer,
80
+ launch,
81
+ registerActivityCallback,
82
+ deregisterActivityCallback,
83
+ removeTransfer,
84
+ showDirectory,
85
+ stopTransfer,
86
+ resumeTransfer,
87
+ getAllTransfers,
88
+ getTransfer,
89
+ registerRemovedCallback,
90
+ deregisterRemovedCallback,
91
+ showSelectFileDialog,
92
+ showSelectFolderDialog,
93
+ showPreferences,
94
+ modifyTransfer,
95
+ createDropzone,
96
+ removeDropzone,
97
+ getInstallerInfo,
98
+ registerStatusCallback,
99
+ deregisterStatusCallback,
100
+ registerSafariExtensionStatusCallback,
101
+ deregisterSafariExtensionStatusCallback,
102
+ setBranding,
103
+ getInfo,
104
+ };
@@ -0,0 +1,360 @@
1
+ import {CustomBrandingOptions, DataTransferResponse, AsperaSdkSpec, AsperaSdkTransfer, FileDialogOptions, FolderDialogOptions, InitOptions, InstallerInfoResponse, InstallerOptions, ModifyTransferOptions, ResumeTransferOptions, SafariExtensionEvent, TransferSpec, WebsocketEvent} from './models';
2
+ import {hiddenStyleList, installerUrl, protocol} from '../constants/constants';
3
+ import {messages} from '../constants/messages';
4
+ import {safariClient} from '../helpers/client/safari-client';
5
+ import {errorLog, isSafari} from '../helpers/helpers';
6
+ import {websocketService} from '../helpers/ws';
7
+ import {asperaSdk} from '../index';
8
+
9
+ class AsperaSdkGlobals {
10
+ /** The URL of the IBM Aspera HTTP server to use with the SDK */
11
+ asperaAppUrl = 'http://127.0.0.1';
12
+ /** The URL of the IBM Aspera Desktop HTTP server to use with the SDK */
13
+ rpcPort = 33024;
14
+ /** The default URL to check for latest Aspera installers */
15
+ installerUrl = installerUrl;
16
+ /** Aspera SDK info */
17
+ AsperaSdkInfo: AsperaSdkInfo;
18
+ /** Indication that the server has been verified as working */
19
+ asperaAppVerified = false;
20
+ /** The unique ID for the website */
21
+ appId: string;
22
+ /** The session ID for the current user */
23
+ sessionId?: string;
24
+ /** Map of drop zones created by querySelector */
25
+ dropZonesCreated: Map<string, {event: string; callback: (event: any) => void}[]> = new Map();
26
+
27
+ backupLaunchMethod(url: string): void {
28
+ window.alert(messages.loadingProtocol);
29
+ window.location.href = url;
30
+ }
31
+
32
+ /**
33
+ * Launch the IBM Aspera App via protocol url. By default, a hidden IFRAME attempts to
34
+ * open the app but if that fails a fallback of opening a new window happens.
35
+ */
36
+ launch(): void {
37
+ try {
38
+ const element = document.createElement('iframe');
39
+ element.src = protocol;
40
+ element.onerror = error => {
41
+ errorLog(messages.failedToGenerateIframe, error);
42
+ this.backupLaunchMethod(protocol);
43
+ };
44
+ element.setAttribute('style', hiddenStyleList);
45
+ document.body.appendChild(element);
46
+ } catch (error) {
47
+ errorLog(messages.failedToGenerateIframe, error);
48
+ this.backupLaunchMethod(protocol);
49
+ }
50
+ }
51
+ }
52
+
53
+ export class AsperaSdkInfo {
54
+ /** The version of IBM Aspera SDK */
55
+ version: string;
56
+ /** The public key corresponding to the SSH private key generated by IBM Aspera */
57
+ client_pubkey?: string;
58
+ }
59
+
60
+ export class TransferResponse {
61
+ transfers: AsperaSdkTransfer[];
62
+ }
63
+
64
+ export type ActivityMessageTypes = 'transferUpdated'|'transferRemoved';
65
+
66
+ export class ActivityMessage {
67
+ type: ActivityMessageTypes;
68
+ data: unknown;
69
+ }
70
+
71
+ export class ActivityTracking {
72
+ /** Map of callbacks that receive transfer update events */
73
+ private activity_callbacks: Map<string, Function> = new Map();
74
+ /** Map of callbacks that received removed transfer events */
75
+ private removed_callbacks: Map<string, Function> = new Map();
76
+ /** Map of callbacks that receive connection events */
77
+ private event_callbacks: Map<string, Function> = new Map();
78
+ /** Map of callbacks that receive Safari extension events */
79
+ private safari_extension_callbacks: Map<string, Function> = new Map();
80
+
81
+ /** Keep track of the last WebSocket event **/
82
+ private lastWebSocketEvent: WebsocketEvent = 'CLOSED';
83
+ /** Keep track of the last Safari extension event **/
84
+ private lastSafariExtensionEvent: SafariExtensionEvent = 'DISABLED';
85
+
86
+ /**
87
+ * Notify all consumers when a message is received from the websocket
88
+ *
89
+ * @param message the message received from the websocket
90
+ */
91
+ handleTransferActivity(message: ActivityMessage): void {
92
+ if (message.type === 'transferUpdated') {
93
+ this.activity_callbacks.forEach(callback => {
94
+ if (typeof callback === 'function') {
95
+ callback(message.data);
96
+ }
97
+ });
98
+ }
99
+
100
+ if (message.type === 'transferRemoved') {
101
+ this.removed_callbacks.forEach(callback => {
102
+ if (typeof callback === 'function') {
103
+ callback(message.data);
104
+ }
105
+ });
106
+ }
107
+ }
108
+
109
+ /**
110
+ * Notify all consumers when a connection webSocketEvent occurs. For example, when the SDK
111
+ * websocket connection to IBM Aspera App is closed or reconnected.
112
+ *
113
+ * @param webSocketEvent the event type.
114
+ */
115
+ handleWebSocketEvents(webSocketEvent: WebsocketEvent): void {
116
+ if (this.lastWebSocketEvent === webSocketEvent) {
117
+ return;
118
+ }
119
+
120
+ this.event_callbacks.forEach(callback => {
121
+ if (typeof callback === 'function') {
122
+ callback(webSocketEvent);
123
+ }
124
+ });
125
+
126
+ this.lastWebSocketEvent = webSocketEvent;
127
+ }
128
+
129
+ /**
130
+ * Notify all consumers when a Safari extension safariExtensionEvent occurs (enabled/disabled).
131
+ *
132
+ * @param safariExtensionEvent the event type.
133
+ */
134
+ handleSafariExtensionEvents(safariExtensionEvent: SafariExtensionEvent): void {
135
+ if (this.lastSafariExtensionEvent === safariExtensionEvent) {
136
+ return;
137
+ }
138
+
139
+ this.safari_extension_callbacks.forEach(callback => {
140
+ if (typeof callback === 'function') {
141
+ callback(safariExtensionEvent);
142
+ }
143
+ });
144
+
145
+ this.lastSafariExtensionEvent = safariExtensionEvent;
146
+ }
147
+
148
+ /**
149
+ * Set up the activity tracking with IBM Aspera.
150
+ *
151
+ * @returns a promise that resolves when the websocket connection is established.
152
+ * Currently, this promise does not reject.
153
+ */
154
+ setup(): Promise<unknown> {
155
+ if (asperaSdk.globals.sessionId) {
156
+ this.registerDesktopAppSession();
157
+ }
158
+
159
+ if (isSafari()) {
160
+ return safariClient.monitorTransferActivity();
161
+ }
162
+
163
+ return websocketService.init()
164
+ .then(() => {
165
+ websocketService.registerMessage('transfer_activity', (data: ActivityMessage) => this.handleTransferActivity(data));
166
+ websocketService.registerEvent((status: 'CLOSED'|'RECONNECT') => this.handleWebSocketEvents(status));
167
+ });
168
+ }
169
+
170
+ /**
171
+ * Register a callback for getting transfers back to the consumer
172
+ *
173
+ * @param callback the function to call with the array of transfers
174
+ *
175
+ * @returns the ID of the callback index
176
+ */
177
+ setCallback(callback: (transfers: TransferResponse) => void): string {
178
+ if (typeof callback !== 'function') {
179
+ errorLog(messages.callbackIsNotFunction);
180
+ return;
181
+ }
182
+ const id = `callback-${this.activity_callbacks.size + 1}`;
183
+ this.activity_callbacks.set(id, callback);
184
+ return id;
185
+ }
186
+
187
+ /**
188
+ * Remove the callback (deregister) from the list of callbacks
189
+ *
190
+ * @param id the string of the callback to remove
191
+ */
192
+ removeCallback(id: string): void {
193
+ this.activity_callbacks.delete(id);
194
+ }
195
+
196
+ /**
197
+ * Register a callback for getting transfers back to the consumer
198
+ *
199
+ * @param callback the function to call with the array of transfers
200
+ *
201
+ * @returns the ID of the callback index
202
+ */
203
+ setRemovedCallback(callback: (transfer: AsperaSdkTransfer) => void): string {
204
+ if (typeof callback !== 'function') {
205
+ errorLog(messages.callbackIsNotFunction);
206
+ return;
207
+ }
208
+ const id = `callback-${this.removed_callbacks.size + 1}`;
209
+ this.removed_callbacks.set(id, callback);
210
+ return id;
211
+ }
212
+
213
+ /**
214
+ * Remove the callback (deregister) from the list of callbacks
215
+ *
216
+ * @param id the string of the callback to remove
217
+ */
218
+ removeRemovedCallback(id: string): void {
219
+ this.removed_callbacks.delete(id);
220
+ }
221
+
222
+ /**
223
+ * Register a callback for getting websocket events back to the consumer
224
+ *
225
+ * @param callback the function to call with the websocket event
226
+ *
227
+ * @returns the ID of the callback index
228
+ */
229
+ setWebSocketEventCallback(callback: (status: WebsocketEvent) => void): string {
230
+ if (typeof callback !== 'function') {
231
+ errorLog(messages.callbackIsNotFunction);
232
+ return;
233
+ }
234
+ const id = `callback-${this.event_callbacks.size + 1}`;
235
+ this.event_callbacks.set(id, callback);
236
+ callback(this.lastWebSocketEvent);
237
+ return id;
238
+ }
239
+
240
+ /**
241
+ * Remove the callback (deregister) from the list of callbacks
242
+ *
243
+ * @param id the string of the callback to remove
244
+ */
245
+ removeWebSocketEventCallback(id: string): void {
246
+ this.event_callbacks.delete(id);
247
+ }
248
+
249
+ /**
250
+ * Register a callback for getting Safari extension events back to the consumer
251
+ *
252
+ * @param callback the function to call with the websocket event
253
+ *
254
+ * @returns the ID of the callback index
255
+ */
256
+ setSafariExtensionEventCallback(callback: (status: SafariExtensionEvent) => void): string {
257
+ if (typeof callback !== 'function') {
258
+ errorLog(messages.callbackIsNotFunction);
259
+ return;
260
+ }
261
+ const id = `callback-${this.safari_extension_callbacks.size + 1}`;
262
+ this.safari_extension_callbacks.set(id, callback);
263
+ callback(this.lastSafariExtensionEvent);
264
+ return id;
265
+ }
266
+
267
+ /**
268
+ * Remove the callback (deregister) from the list of callbacks
269
+ *
270
+ * @param id the string of the callback to remove
271
+ */
272
+ removeSafariExtensionEventCallback(id: string): void {
273
+ this.safari_extension_callbacks.delete(id);
274
+ }
275
+
276
+ private registerDesktopAppSession(): void {
277
+ const iframe = document.createElement('iframe');
278
+ iframe.style.display = 'none';
279
+ iframe.src = `aspera://initialize?app_id=${asperaSdk.globals.appId}&session_id=${asperaSdk.globals.sessionId}`;
280
+ document.body.appendChild(iframe);
281
+
282
+ setTimeout(() => {
283
+ document.body.removeChild(iframe);
284
+ }, 1000);
285
+ }
286
+ }
287
+
288
+ export class AsperaSdk {
289
+ /** Global information about IBM Aspera */
290
+ globals: AsperaSdkGlobals = new AsperaSdkGlobals();
291
+ /** Activity tracking for watching transfers */
292
+ activityTracking: ActivityTracking = new ActivityTracking();
293
+ /** Function to initialize IBM Aspera */
294
+ init: (options: InitOptions) => Promise<any>;
295
+ /** Function to test the IBM Aspera status */
296
+ testConnection: () => Promise<any>;
297
+ /** Function to initiate a transfer */
298
+ startTransfer: (transferSpec: TransferSpec, AsperaSdkSpec: AsperaSdkSpec) => Promise<AsperaSdkTransfer>;
299
+ /** Function to launch IBM Aspera */
300
+ launch: () => void;
301
+ /** Register callback for the transfer activity monitor */
302
+ registerActivityCallback: (callback: (transfers: TransferResponse) => void) => string;
303
+ /** Deregister callback to remove it from the callbacks getting transfer data */
304
+ deregisterActivityCallback: (id: string) => void;
305
+ /** Register callback for removed transfers from the app */
306
+ registerRemovedCallback: (callback: (transfer: AsperaSdkTransfer) => void) => string;
307
+ /** Deregister callback to remove it from the callbacks getting removed transfer data */
308
+ deregisterRemovedCallback: (id: string) => void;
309
+ /** Register callback for connection status events from the app */
310
+ registerStatusCallback: (callback: (status: WebsocketEvent) => void) => string;
311
+ /** Deregister callback to remove it from the callbacks getting connection events */
312
+ deregisterStatusCallback: (id: string) => void;
313
+ /** Register callback for Safari extension status events */
314
+ registerSafariExtensionStatusCallback: (callback: (status: SafariExtensionEvent) => void) => string;
315
+ /** Deregister callback to remove it from the callbacks getting Safari extension events */
316
+ deregisterSafariExtensionStatusCallback: (id: string) => void;
317
+ /** Function to remove a transfer */
318
+ removeTransfer: (transferId: string) => Promise<any>;
319
+ /** Function to show the transfer's download directory in Finder or Windows Explorer */
320
+ showDirectory: (transferId: string) => Promise<any>;
321
+ /** Function to stop a transfer */
322
+ stopTransfer: (transferId: string) => Promise<any>;
323
+ /** Function to resume a transfer */
324
+ resumeTransfer: (transferId: string, options?: ResumeTransferOptions) => Promise<AsperaSdkTransfer>;
325
+ /** Function to get a list of all transfers */
326
+ getAllTransfers: () => Promise<AsperaSdkTransfer[]>;
327
+ /** Function to get information for a specific transfer */
328
+ getTransfer: (transferId: string) => Promise<AsperaSdkTransfer>;
329
+ /** Function to display a file dialog for the user to select files. */
330
+ showSelectFileDialog: (options?: FileDialogOptions) => Promise<DataTransferResponse>;
331
+ /** Function to display a folder dialog for the user to select folders. */
332
+ showSelectFolderDialog: (options?: FolderDialogOptions) => Promise<DataTransferResponse>;
333
+ /** Function to display the IBM Aspera preferences page */
334
+ showPreferences: () => Promise<any>;
335
+ /** Function to modify a running transfer */
336
+ modifyTransfer: (transferId: string, options: ModifyTransferOptions) => Promise<AsperaSdkTransfer>;
337
+ /** Function to set custom branding for IBM Aspera */
338
+ setBranding: (id: string, options: CustomBrandingOptions) => Promise<any>;
339
+ /** Create dropzone for drop events of files */
340
+ createDropzone: (callback: (data: {event: any; files: DataTransferResponse}) => void, elementSelector: string) => void;
341
+ /** Remove dropzone for drop events of files */
342
+ removeDropzone: (elementSelector: string) => void;
343
+ /** Function to get latest installer information */
344
+ getInstallerInfo: (options: InstallerOptions) => Promise<InstallerInfoResponse>;
345
+ /** Initialize drag and drop */
346
+ initDragDrop: () => Promise<any>;
347
+ /** Function to get information about the IBM Aspera instance */
348
+ getInfo: () => Promise<AsperaSdkInfo>;
349
+ /** Function to get whether IBM Aspera is running on Safari */
350
+ isSafari: () => boolean;
351
+
352
+ /**
353
+ * Check if IBM Aspera is ready to be used and has been verified.
354
+ *
355
+ * @returns a boolean indicating if SDK can be used for requests
356
+ */
357
+ get isReady(): boolean {
358
+ return this.globals.asperaAppVerified && this.globals.appId !== '';
359
+ }
360
+ }