2020-10-02 15:10:00 +00:00
|
|
|
/**
|
|
|
|
* This module contains several ways to create an upload on the server. In all cases, an
|
|
|
|
* UploadResult is returned, with an uploadId which may be used in other server calls to identify
|
|
|
|
* this upload.
|
|
|
|
*
|
|
|
|
* TODO: another proposed source for files is uploadUrl(url) which would fetch a file from URL and
|
|
|
|
* upload, and if that fails due to CORS, would fetch the file on the server side instead.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import {DocComm} from 'app/client/components/DocComm';
|
|
|
|
import {UserError} from 'app/client/models/errors';
|
|
|
|
import {FileDialogOptions, openFilePicker} from 'app/client/ui/FileDialog';
|
2020-10-15 21:51:30 +00:00
|
|
|
import {BaseAPI} from 'app/common/BaseAPI';
|
2020-10-02 15:10:00 +00:00
|
|
|
import {GristLoadConfig} from 'app/common/gristUrls';
|
|
|
|
import {byteString, safeJsonParse} from 'app/common/gutil';
|
|
|
|
import {UPLOAD_URL_PATH, UploadResult} from 'app/common/uploads';
|
|
|
|
import {docUrl} from 'app/common/urlUtils';
|
|
|
|
import {OpenDialogOptions} from 'electron';
|
|
|
|
import noop = require('lodash/noop');
|
|
|
|
import trimStart = require('lodash/trimStart');
|
|
|
|
import {basename} from 'path'; // made available by webpack using path-browserify module.
|
|
|
|
|
|
|
|
type ProgressCB = (percent: number) => void;
|
|
|
|
|
|
|
|
export interface UploadOptions {
|
|
|
|
docWorkerUrl?: string;
|
|
|
|
sizeLimit?: 'import'|'attachment';
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface SelectFileOptions extends UploadOptions {
|
|
|
|
multiple?: boolean; // Whether multiple files may be selected.
|
|
|
|
extensions?: string[]; // Comma-separated list of extensions (with a leading period),
|
|
|
|
// e.g. [".jpg", ".png"]
|
|
|
|
}
|
|
|
|
|
|
|
|
export const IMPORTABLE_EXTENSIONS = [".grist", ".csv", ".tsv", ".txt", ".xls", ".xlsx", ".xlsm"];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows the file-picker dialog with the given options, and uploads the selected files. If under
|
|
|
|
* electron, shows the native file-picker instead.
|
|
|
|
*
|
|
|
|
* If given, onProgress() callback will be called with 0 on initial call, and will go up to 100
|
|
|
|
* after files are selected to indicate percentage of data uploaded.
|
|
|
|
*/
|
|
|
|
export async function selectFiles(options: SelectFileOptions,
|
|
|
|
onProgress: ProgressCB = noop): Promise<UploadResult|null> {
|
|
|
|
onProgress(0);
|
|
|
|
let result: UploadResult|null = null;
|
|
|
|
const electronSelectFiles: any = (window as any).electronSelectFiles;
|
|
|
|
if (typeof electronSelectFiles === 'function') {
|
|
|
|
result = await electronSelectFiles(getElectronOptions(options));
|
|
|
|
} else {
|
|
|
|
const files: File[] = await openFilePicker(getFileDialogOptions(options));
|
|
|
|
result = await uploadFiles(files, options, onProgress);
|
|
|
|
}
|
|
|
|
onProgress(100);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper to convert SelectFileOptions to the browser's FileDialogOptions.
|
|
|
|
function getFileDialogOptions(options: SelectFileOptions): FileDialogOptions {
|
|
|
|
const resOptions: FileDialogOptions = {};
|
|
|
|
if (options.multiple) {
|
|
|
|
resOptions.multiple = options.multiple;
|
|
|
|
}
|
|
|
|
if (options.extensions) {
|
|
|
|
resOptions.accept = options.extensions.join(",");
|
|
|
|
}
|
|
|
|
return resOptions;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper to convert SelectFileOptions to electron's OpenDialogOptions.
|
|
|
|
function getElectronOptions(options: SelectFileOptions): OpenDialogOptions {
|
|
|
|
const resOptions: OpenDialogOptions = {
|
|
|
|
filters: [],
|
|
|
|
properties: ['openFile'],
|
|
|
|
};
|
|
|
|
if (options.extensions) {
|
|
|
|
// Electron does not expect leading period.
|
|
|
|
const extensions = options.extensions.map(e => trimStart(e, '.'));
|
|
|
|
resOptions.filters!.push({name: 'Select files', extensions});
|
|
|
|
}
|
|
|
|
if (options.multiple) {
|
|
|
|
resOptions.properties!.push('multiSelections');
|
|
|
|
}
|
|
|
|
return resOptions;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Uploads a list of File objects to the server.
|
|
|
|
*/
|
|
|
|
export async function uploadFiles(
|
|
|
|
fileList: File[], options: UploadOptions, onProgress: ProgressCB = noop
|
|
|
|
): Promise<UploadResult|null> {
|
|
|
|
if (!fileList.length) { return null; }
|
|
|
|
|
|
|
|
const formData = new FormData();
|
|
|
|
for (const file of fileList) {
|
|
|
|
formData.append('upload', file);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for upload limits.
|
|
|
|
const gristConfig: GristLoadConfig = (window as any).gristConfig || {};
|
|
|
|
const {maxUploadSizeImport, maxUploadSizeAttachment} = gristConfig;
|
|
|
|
if (options.sizeLimit === 'import' && maxUploadSizeImport) {
|
|
|
|
// For imports, we limit the total upload size, but exempt .grist files from the upload limit.
|
|
|
|
// Grist docs can be uploaded to make copies or restore from backup, and may legitimately be
|
|
|
|
// very large (e.g. contain many attachments or on-demand tables).
|
|
|
|
const totalSize = fileList.reduce((acc, f) => acc + (f.name.endsWith(".grist") ? 0 : f.size), 0);
|
|
|
|
if (totalSize > maxUploadSizeImport) {
|
|
|
|
throw new UserError(`Imported files may not exceed ${byteString(maxUploadSizeImport)}`);
|
|
|
|
}
|
|
|
|
} else if (options.sizeLimit === 'attachment' && maxUploadSizeAttachment) {
|
|
|
|
// For attachments, we limit the size of each attachment.
|
|
|
|
if (fileList.some((f) => (f.size > maxUploadSizeAttachment))) {
|
|
|
|
throw new UserError(`Attachments may not exceed ${byteString(maxUploadSizeAttachment)}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise<UploadResult>((resolve, reject) => {
|
|
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
xhr.open('post', docUrl(options.docWorkerUrl, UPLOAD_URL_PATH), true);
|
2020-10-08 13:28:39 +00:00
|
|
|
xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
|
2020-10-02 15:10:00 +00:00
|
|
|
xhr.withCredentials = true;
|
|
|
|
xhr.upload.addEventListener('progress', (e) => {
|
|
|
|
if (e.lengthComputable) {
|
|
|
|
onProgress(e.loaded / e.total * 100); // percentage complete
|
|
|
|
}
|
|
|
|
});
|
|
|
|
xhr.addEventListener('error', (e: ProgressEvent) => {
|
|
|
|
console.warn("Upload error", e); // tslint:disable-line:no-console
|
|
|
|
// The event does not seem to have any helpful info in it, to add to the message.
|
|
|
|
reject(new Error('Upload error'));
|
|
|
|
});
|
|
|
|
xhr.addEventListener('load', () => {
|
|
|
|
if (xhr.status !== 200) {
|
|
|
|
// tslint:disable-next-line:no-console
|
|
|
|
console.warn("Upload failed", xhr.status, xhr.responseText);
|
|
|
|
const err = safeJsonParse(xhr.responseText, null);
|
|
|
|
reject(new UserError('Upload failed: ' + (err && err.error || xhr.status)));
|
|
|
|
} else {
|
|
|
|
resolve(JSON.parse(xhr.responseText));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
xhr.send(formData);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetches ressource from a url and returns an UploadResult. Tries to fetch from the client and
|
|
|
|
* upload the file to the server. If unsuccessful, tries to fetch directly from the server. In both
|
|
|
|
* case, it guesses the name of the file based on the response's content-type and the url.
|
|
|
|
*/
|
|
|
|
export async function fetchURL(
|
|
|
|
docComm: DocComm, url: string, onProgress: ProgressCB = noop
|
|
|
|
): Promise<UploadResult> {
|
|
|
|
let response: Response;
|
|
|
|
try {
|
|
|
|
response = await window.fetch(url);
|
|
|
|
} catch (err) {
|
|
|
|
console.log( // tslint:disable-line:no-console
|
|
|
|
`Could not fetch ${url} on the Client, falling back to server fetch: ${err.message}`
|
|
|
|
);
|
|
|
|
return docComm.fetchURL(url);
|
|
|
|
}
|
|
|
|
// TODO: We should probably parse response.headers.get('content-disposition') when available
|
|
|
|
// (see content-disposition npm module).
|
|
|
|
const fileName = basename(url);
|
|
|
|
const mimeType = response.headers.get('content-type');
|
|
|
|
const options = mimeType ? { type: mimeType } : {};
|
|
|
|
const fileObj = new File([await response.blob()], fileName, options);
|
|
|
|
const res = await uploadFiles([fileObj], {docWorkerUrl: docComm.docWorkerUrl}, onProgress);
|
|
|
|
return res!;
|
|
|
|
}
|
2020-10-08 13:28:39 +00:00
|
|
|
|
2020-10-15 21:51:30 +00:00
|
|
|
/**
|
|
|
|
* Convert a form to a JSON-stringifiable object, ignoring any File fields.
|
|
|
|
*/
|
|
|
|
export function formDataToObj(formElem: HTMLFormElement): {[key: string]: string} {
|
|
|
|
// Use FormData to collect values (rather than e.g. finding <input> elements) to ensure we get
|
|
|
|
// values from all form items correctly (e.g. checkboxes and textareas).
|
|
|
|
const formData = new FormData(formElem);
|
|
|
|
const data: {[key: string]: string} = {};
|
|
|
|
for (const [name, value] of formData.entries()) {
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
data[name] = value;
|
2020-10-08 13:28:39 +00:00
|
|
|
}
|
2020-10-15 21:51:30 +00:00
|
|
|
}
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Submit a form using BaseAPI. Send inputs as JSON, and interpret any reply as JSON.
|
|
|
|
export async function submitForm(form: HTMLFormElement): Promise<any> {
|
|
|
|
const data = formDataToObj(form);
|
|
|
|
return BaseAPI.requestJson(form.action, {method: 'POST', body: JSON.stringify(data)});
|
2020-10-08 13:28:39 +00:00
|
|
|
}
|