forked from Ivasoft/mattermost-mobile
258 lines
6.8 KiB
JavaScript
258 lines
6.8 KiB
JavaScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import {Platform} from 'react-native';
|
|
import RNFetchBlob from 'rn-fetch-blob';
|
|
import mimeDB from 'mime-db';
|
|
|
|
import {lookupMimeType} from 'mattermost-redux/utils/file_utils';
|
|
|
|
import {DeviceTypes} from 'app/constants/';
|
|
|
|
const EXTRACT_TYPE_REGEXP = /^\s*([^;\s]*)(?:;|\s|$)/;
|
|
const CONTENT_DISPOSITION_REGEXP = /inline;filename=".*\.([a-z]+)";/i;
|
|
const {DOCUMENTS_PATH, IMAGES_PATH, VIDEOS_PATH} = DeviceTypes;
|
|
const DEFAULT_SERVER_MAX_FILE_SIZE = 50 * 1024 * 1024;// 50 Mb
|
|
|
|
export const SUPPORTED_DOCS_FORMAT = [
|
|
'application/json',
|
|
'application/msword',
|
|
'application/pdf',
|
|
'application/rtf',
|
|
'application/vnd.ms-excel',
|
|
'application/vnd.ms-powerpoint',
|
|
'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
|
|
'application/vnd.openxmlformats-officedocument.presentationml.presentation',
|
|
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
|
|
'application/x-x509-ca-cert',
|
|
'application/xml',
|
|
'text/csv',
|
|
'text/plain',
|
|
];
|
|
|
|
const SUPPORTED_VIDEO_FORMAT = Platform.select({
|
|
ios: ['video/mp4', 'video/x-m4v', 'video/quicktime'],
|
|
android: ['video/3gpp', 'video/x-matroska', 'video/mp4', 'video/webm'],
|
|
});
|
|
|
|
const types = {};
|
|
const extensions = {};
|
|
|
|
export function generateId() {
|
|
// Implementation taken from http://stackoverflow.com/a/2117523
|
|
let id = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx';
|
|
|
|
id = id.replace(/[xy]/g, (c) => {
|
|
const r = Math.floor(Math.random() * 16);
|
|
|
|
let v;
|
|
if (c === 'x') {
|
|
v = r;
|
|
} else {
|
|
v = (r & 0x3) | 0x8;
|
|
}
|
|
|
|
return v.toString(16);
|
|
});
|
|
|
|
return 'uid' + id;
|
|
}
|
|
|
|
export async function getFileCacheSize() {
|
|
const isDocsDir = await RNFetchBlob.fs.isDir(DOCUMENTS_PATH);
|
|
const isImagesDir = await RNFetchBlob.fs.isDir(IMAGES_PATH);
|
|
const isVideosDir = await RNFetchBlob.fs.isDir(VIDEOS_PATH);
|
|
let size = 0;
|
|
|
|
if (isDocsDir) {
|
|
const docsStats = await RNFetchBlob.fs.lstat(DOCUMENTS_PATH);
|
|
size = docsStats.reduce((accumulator, stat) => {
|
|
return accumulator + parseInt(stat.size, 10);
|
|
}, size);
|
|
}
|
|
|
|
if (isImagesDir) {
|
|
const imagesStats = await RNFetchBlob.fs.lstat(IMAGES_PATH);
|
|
size = imagesStats.reduce((accumulator, stat) => {
|
|
return accumulator + parseInt(stat.size, 10);
|
|
}, size);
|
|
}
|
|
|
|
if (isVideosDir) {
|
|
const videoStats = await RNFetchBlob.fs.lstat(VIDEOS_PATH);
|
|
size = videoStats.reduce((accumulator, stat) => {
|
|
return accumulator + parseInt(stat.size, 10);
|
|
}, size);
|
|
}
|
|
|
|
return size;
|
|
}
|
|
|
|
export async function deleteFileCache() {
|
|
const isDocsDir = await RNFetchBlob.fs.isDir(DOCUMENTS_PATH);
|
|
const isImagesDir = await RNFetchBlob.fs.isDir(IMAGES_PATH);
|
|
const isVideosDir = await RNFetchBlob.fs.isDir(VIDEOS_PATH);
|
|
if (isDocsDir) {
|
|
await RNFetchBlob.fs.unlink(DOCUMENTS_PATH);
|
|
}
|
|
|
|
if (isImagesDir) {
|
|
await RNFetchBlob.fs.unlink(IMAGES_PATH);
|
|
}
|
|
|
|
if (isVideosDir) {
|
|
await RNFetchBlob.fs.unlink(VIDEOS_PATH);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
export function buildFileUploadData(file) {
|
|
const re = /heic/i;
|
|
const uri = file.uri;
|
|
let name = file.fileName || file.name || file.path || file.uri;
|
|
let mimeType = lookupMimeType(name.toLowerCase());
|
|
let extension = name.split('.').pop().replace('.', '');
|
|
|
|
if (re.test(extension)) {
|
|
extension = 'JPG';
|
|
name = name.replace(re, 'jpg');
|
|
mimeType = 'image/jpeg';
|
|
}
|
|
|
|
return {
|
|
uri,
|
|
name,
|
|
type: mimeType,
|
|
extension,
|
|
};
|
|
}
|
|
|
|
export const encodeHeaderURIStringToUTF8 = (string) => {
|
|
return encodeURIComponent(string) + '"; filename*="utf-8\'\'' + encodeURIComponent(string);
|
|
};
|
|
|
|
export const getAllowedServerMaxFileSize = (config) => {
|
|
return config && config.MaxFileSize ? parseInt(config.MaxFileSize, 10) : DEFAULT_SERVER_MAX_FILE_SIZE;
|
|
};
|
|
|
|
export const isGif = (file) => {
|
|
let mime = file.mime_type || file.type || '';
|
|
if (mime && mime.includes(';')) {
|
|
mime = mime.split(';')[0];
|
|
}
|
|
|
|
return mime === 'image/gif';
|
|
};
|
|
|
|
export const isDocument = (file) => {
|
|
let mime = file.mime_type || file.type || '';
|
|
if (mime && mime.includes(';')) {
|
|
mime = mime.split(';')[0];
|
|
}
|
|
|
|
return SUPPORTED_DOCS_FORMAT.includes(mime);
|
|
};
|
|
|
|
export const isVideo = (file) => {
|
|
let mime = file.mime_type || file.type || '';
|
|
if (mime && mime.includes(';')) {
|
|
mime = mime.split(';')[0];
|
|
}
|
|
|
|
return SUPPORTED_VIDEO_FORMAT.includes(mime);
|
|
};
|
|
|
|
/**
|
|
* Get the default extension for a MIME type.
|
|
*
|
|
* @param {string} type
|
|
* @return {boolean|string}
|
|
*/
|
|
|
|
export function getExtensionFromMime(type) {
|
|
if (!Object.keys(extensions).length) {
|
|
populateMaps();
|
|
}
|
|
|
|
if (!type || typeof type !== 'string') {
|
|
return false;
|
|
}
|
|
|
|
// TODO: use media-typer
|
|
const match = EXTRACT_TYPE_REGEXP.exec(type);
|
|
|
|
// get extensions
|
|
const exts = match && extensions[match[1].toLowerCase()];
|
|
|
|
if (!exts || !exts.length) {
|
|
return false;
|
|
}
|
|
|
|
return exts[0];
|
|
}
|
|
|
|
/**
|
|
* Populate the extensions and types maps.
|
|
* @private
|
|
*/
|
|
|
|
function populateMaps() {
|
|
// source preference (least -> most)
|
|
const preference = ['nginx', 'apache', undefined, 'iana']; //eslint-disable-line no-undefined
|
|
|
|
Object.keys(mimeDB).forEach((type) => {
|
|
const mime = mimeDB[type];
|
|
const exts = mime.extensions;
|
|
|
|
if (!exts || !exts.length) {
|
|
return;
|
|
}
|
|
|
|
extensions[type] = exts;
|
|
|
|
for (let i = 0; i < exts.length; i++) {
|
|
const extension = exts[i];
|
|
|
|
if (types[extension]) {
|
|
const from = preference.indexOf(mimeDB[types[extension]].source);
|
|
const to = preference.indexOf(mime.source);
|
|
|
|
if (types[extension] !== 'application/octet-stream' &&
|
|
(from > to || (from === to && types[extension].substr(0, 12) === 'application/'))) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
types[extension] = type;
|
|
}
|
|
});
|
|
}
|
|
|
|
export function getLocalFilePathFromFile(dir, file) {
|
|
if (dir && file && file.data && file.data.id && file.data.extension) {
|
|
return `${dir}/${file.data.id}.${file.data.extension}`;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
export function getExtensionFromContentDisposition(contentDisposition) {
|
|
const match = CONTENT_DISPOSITION_REGEXP.exec(contentDisposition);
|
|
let extension = match && match[1];
|
|
if (extension) {
|
|
if (!Object.keys(types).length) {
|
|
populateMaps();
|
|
}
|
|
|
|
extension = extension.toLowerCase();
|
|
if (types[extension]) {
|
|
return extension;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
return null;
|
|
}
|