-
-
Notifications
You must be signed in to change notification settings - Fork 169
/
client.js
312 lines (275 loc) · 13.1 KB
/
client.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
import { DDP } from 'meteor/ddp-client';
import { Mongo } from 'meteor/mongo';
import { Meteor } from 'meteor/meteor';
import { Cookies } from 'meteor/ostrio:cookies';
import { check, Match } from 'meteor/check';
import { UploadInstance } from './upload.js';
import FilesCollectionCore from './core.js';
import { formatFleURL, helpers } from './lib.js';
const NOOP = () => { };
const allowedParams = ['allowClientCode', 'allowQueryStringCookies', 'chunkSize', 'collection', 'collectionName', 'ddp', 'debug', 'disableSetTokenCookie', 'disableUpload', 'downloadRoute', 'namingFunction', 'onBeforeUpload', 'onbeforeunloadMessage', 'public', 'sanitize', 'schema'];
/**
* @locus Anywhere
* @class FilesCollection
* @param config {Object} - [Both] Configuration object with next properties:
* @param config.debug {Boolean} - [Both] Turn on/of debugging and extra logging
* @param config.ddp {Object} - [Client] Custom DDP connection. Object returned form `DDP.connect()`
* @param config.schema {Object} - [Both] Collection Schema
* @param config.public {Boolean} - [Both] Store files in folder accessible for proxy servers, for limits, and more - read docs`
* @param config.chunkSize {Number} - [Both] Upload chunk size, default: 524288 bytes (0,5 Mb)
* @param config.downloadRoute {String} - [Both] Server Route used to retrieve files
* @param config.collection {Mongo.Collection} - [Both] Mongo Collection Instance
* @param config.collectionName {String} - [Both] Collection name
* @param config.namingFunction {Function}- [Both] Function which returns `String`
* @param config.onBeforeUpload {Function}- [Both] Function which executes on server after receiving each chunk and on client right before beginning upload. Function context is `File` - so you are able to check for extension, mime-type, size and etc.
* return `true` to continue
* return `false` or `String` to abort upload
* @param config.allowClientCode {Boolean} - [Both] Allow to run `remove` from client
* @param config.onbeforeunloadMessage {String|Function} - [Client] Message shown to user when closing browser's window or tab while upload process is running
* @param config.disableUpload {Boolean} - Disable file upload, useful for server only solutions
* @param config.disableSetTokenCookie {Boolean} - Disable cookie setting. Useful when you use multiple file collections or when you want to implement your own authorization.
* @param config.allowQueryStringCookies {Boolean} - Allow passing Cookies in a query string (in URL). Primary should be used only in Cordova environment. Note: this option will be used only on Cordova. Default: `false`
* @param config.sanitize {Function} - Override default sanitize function
* @summary Create new instance of FilesCollection
*/
class FilesCollection extends FilesCollectionCore {
constructor(config) {
super();
if (config) {
Object.keys(config).forEach((param) => {
if (allowedParams.includes(param)) {
this[param] = config[param];
}
});
}
const self = this;
const cookie = new Cookies({
allowQueryStringCookies: this.allowQueryStringCookies
});
if (!helpers.isBoolean(this.debug)) {
this.debug = false;
}
if (!helpers.isBoolean(this.public)) {
this.public = false;
}
if (!this.chunkSize) {
this.chunkSize = 1024 * 512;
}
this.chunkSize = Math.floor(this.chunkSize / 8) * 8;
if (!helpers.isString(this.collectionName) && !this.collection) {
this.collectionName = 'MeteorUploadFiles';
}
if (!this.collection) {
this.collection = new Mongo.Collection(this.collectionName);
} else {
this.collectionName = this.collection._name;
}
this.collection.filesCollection = this;
check(this.collectionName, String);
if (this.public && !this.downloadRoute) {
throw new Meteor.Error(500, `[FilesCollection.${this.collectionName}]: "downloadRoute" must be precisely provided on "public" collections! Note: "downloadRoute" must be equal or be inside of your web/proxy-server (relative) root.`);
}
if (!helpers.isBoolean(this.disableUpload)) {
this.disableUpload = false;
}
if (!helpers.isBoolean(this.disableSetTokenCookie)) {
this.disableSetTokenCookie = false;
}
if (!helpers.isString(this.downloadRoute)) {
this.downloadRoute = '/cdn/storage';
}
this.downloadRoute = this.downloadRoute.replace(/\/$/, '');
if (!helpers.isFunction(this.namingFunction)) {
this.namingFunction = false;
}
if (!helpers.isFunction(this.onBeforeUpload)) {
this.onBeforeUpload = false;
}
if (!helpers.isBoolean(this.allowClientCode)) {
this.allowClientCode = true;
}
if (!this.ddp) {
this.ddp = Meteor;
}
if (!this.onbeforeunloadMessage) {
this.onbeforeunloadMessage = 'Upload in a progress... Do you want to abort?';
}
if (!config.disableSetTokenCookie) {
const setTokenCookie = () => {
if (Meteor.connection._lastSessionId) {
cookie.set('x_mtok', Meteor.connection._lastSessionId, { path: '/', sameSite: 'Lax' });
if (Meteor.isCordova && this.allowQueryStringCookies) {
cookie.send();
}
}
};
const _accounts = (Package && Package['accounts-base'] && Package['accounts-base'].Accounts) ? Package['accounts-base'].Accounts : undefined;
if (_accounts) {
DDP.onReconnect((conn) => {
conn.onReconnect = setTokenCookie;
});
Meteor.startup(setTokenCookie);
_accounts.onLogin(setTokenCookie);
}
}
check(this.onbeforeunloadMessage, Match.OneOf(String, Function));
try {
const _URL = window.URL || window.webkitURL || window.mozURL || window.msURL || window.oURL || false;
if (window.Worker && window.Blob && _URL && helpers.isFunction(_URL.createObjectURL)) {
this._supportWebWorker = true;
this._webWorkerUrl = _URL.createObjectURL(new window.Blob(['!function(a){"use strict";a.onmessage=function(b){var c=b.data.f.slice(b.data.cs*(b.data.cc-1),b.data.cs*b.data.cc);if(b.data.ib===!0)postMessage({bin:c,chunkId:b.data.cc});else{var d;a.FileReader?(d=new FileReader,d.onloadend=function(a){postMessage({bin:(d.result||a.srcElement||a.target).split(",")[1],chunkId:b.data.cc,s:b.data.s})},d.onerror=function(a){throw(a.target||a.srcElement).error},d.readAsDataURL(c)):a.FileReaderSync?(d=new FileReaderSync,postMessage({bin:d.readAsDataURL(c).split(",")[1],chunkId:b.data.cc})):postMessage({bin:null,chunkId:b.data.cc,error:"File API is not supported in WebWorker!"})}}}(this);'], { type: 'application/javascript' }));
} else if (window.Worker) {
this._supportWebWorker = true;
this._webWorkerUrl = Meteor.absoluteUrl('packages/ostrio_files/worker.min.js');
} else {
this._supportWebWorker = false;
}
} catch (e) {
self._debug('[FilesCollection] [Check WebWorker Availability] Error:', e);
this._supportWebWorker = false;
}
if (!this.schema) {
this.schema = FilesCollectionCore.schema;
}
check(this.debug, Boolean);
check(this.schema, Object);
check(this.public, Boolean);
check(this.chunkSize, Number);
check(this.downloadRoute, String);
check(this.disableUpload, Boolean);
check(this.namingFunction, Match.OneOf(false, Function));
check(this.onBeforeUpload, Match.OneOf(false, Function));
check(this.allowClientCode, Boolean);
check(this.ddp, Match.Any);
this._methodNames = {
_Abort: `_FilesCollectionAbort_${this.collectionName}`,
_Write: `_FilesCollectionWrite_${this.collectionName}`,
_Start: `_FilesCollectionStart_${this.collectionName}`,
_Remove: `_FilesCollectionRemove_${this.collectionName}`
};
}
/**
* @locus Anywhere
* @memberOf FilesCollection
* @name _getMimeType
* @param {Object} fileData - File Object
* @summary Returns file's mime-type
* @returns {String}
*/
_getMimeType(fileData) {
let mime;
check(fileData, Object);
if (helpers.isObject(fileData)) {
mime = fileData.type;
}
if (!mime || !helpers.isString(mime)) {
mime = 'application/octet-stream';
}
return mime;
}
/**
* @locus Anywhere
* @memberOf FilesCollection
* @name _getUser
* @summary Returns object with `userId` and `user()` method which return user's object
* @returns {Object}
*/
_getUser() {
const result = {
user() {
return null;
},
userId: null
};
if (helpers.isFunction(Meteor.userId)) {
result.user = () => Meteor.user();
result.userId = Meteor.userId();
}
return result;
}
/**
* @locus Client
* @memberOf FilesCollection
* @name insert
* @see https://developer.mozilla.org/en-US/docs/Web/API/FileReader
* @param {Object} config - Configuration object with next properties:
* {File|Object} file - HTML5 `files` item, like in change event: `e.currentTarget.files[0]`
* {String} fileId - Optional `fileId` used at insert
* {Object} meta - Additional data as object, use later for search
* {Boolean} allowWebWorkers- Allow/Deny WebWorkers usage
* {Number|dynamic} chunkSize - Chunk size for upload
* {String} transport - Upload transport `http` or `ddp`
* {Object} ddp - Custom DDP connection. Object returned form `DDP.connect()`
* {Function} onUploaded - Callback triggered when upload is finished, with two arguments `error` and `fileRef`
* {Function} onStart - Callback triggered when upload is started after all successful validations, with two arguments `error` (always null) and `fileRef`
* {Function} onError - Callback triggered on error in upload and/or FileReader, with two arguments `error` and `fileData`
* {Function} onProgress - Callback triggered when chunk is sent, with only argument `progress`
* {Function} onBeforeUpload - Callback triggered right before upload is started:
* return true to continue
* return false to abort upload
* @param {Boolean} autoStart - Start upload immediately. If set to false, you need manually call .start() method on returned class. Useful to set EventListeners.
* @summary Upload file to server over DDP or HTTP
* @returns {UploadInstance} Instance. UploadInstance has next properties:
* {ReactiveVar} onPause - Is upload process on the pause?
* {ReactiveVar} state - active|paused|aborted|completed
* {ReactiveVar} progress - Current progress in percentage
* {Function} pause - Pause upload process
* {Function} continue - Continue paused upload process
* {Function} toggle - Toggle continue/pause if upload process
* {Function} abort - Abort upload
* {Function} readAsDataURL - Current file as data URL, use to create image preview and etc. Be aware of big files, may lead to browser crash
*/
insert(config, autoStart = true) {
if (this.disableUpload) {
Meteor._debug('[FilesCollection] [insert()] Upload is disabled with [disableUpload]!');
return {};
}
return (new UploadInstance(config, this))[autoStart ? 'start' : 'manual']();
}
/**
* @locus Anywhere
* @memberOf FilesCollection
* @name remove
* @param {String|Object} selector - Mongo-Style selector (http://docs.meteor.com/api/collections.html#selectors)
* @param {Function} callback - Callback with one `error` argument
* @summary Remove documents from the collection
* @returns {FilesCollection} Instance
*/
remove(selector = {}, callback) {
this._debug(`[FilesCollection] [remove(${JSON.stringify(selector)})]`);
check(selector, Match.OneOf(Object, String));
check(callback, Match.Optional(Function));
if (this.allowClientCode) {
this.ddp.call(this._methodNames._Remove, selector, (callback || NOOP));
} else {
callback && callback(new Meteor.Error(401, '[FilesCollection] [remove] Run code from client is not allowed!'));
this._debug('[FilesCollection] [remove] Run code from client is not allowed!');
}
return this;
}
}
/*
* @locus Client
* @TemplateHelper
* @name fileURL
* @param {Object} fileRef - File reference object
* @param {String} version - [Optional] Version of file you would like to request
* @param {String} uriBase - [Optional] URI base, see - https://github.com/veliovgroup/Meteor-Files/issues/626
* @summary Get download URL for file by fileRef, even without subscription
* @example {{fileURL fileRef}}
* @returns {String}
*/
Meteor.startup(() => {
const _template = (Package && Package.templating && Package.templating.Template) ? Package.templating.Template : undefined;
if (_template) {
_template.registerHelper('fileURL', (fileRef, _version = 'original', _uriBase) => {
if (!helpers.isObject(fileRef)) {
return '';
}
const version = (!helpers.isString(_version)) ? 'original' : _version;
const uriBase = (!helpers.isString(_uriBase)) ? void 0 : _uriBase;
return formatFleURL(fileRef, version, uriBase);
});
}
});
export { FilesCollection, helpers };