Files
mattermost-mobile/app/utils/image_cache_manager.test.js
Miguel Alatzar ebe8d45711 [MM-16816] Parse source URI (#3026)
* Parse source URI

* Revert "Parse source URI"

This reverts commit 1cf421c9b9.

* Pass imageUri instead of defaultSource to ProgressiveImage

* Parse source URI in ImageCacheManager
2019-07-24 07:59:06 -07:00

314 lines
14 KiB
JavaScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import RNFetchBlob from 'rn-fetch-blob';
import ImageCacheManager, {getCacheFile, hashCode} from 'app/utils/image_cache_manager';
import {emptyFunction} from 'app/utils/general';
import * as fileUtils from 'app/utils/file';
import mattermostBucket from 'app/mattermost_bucket';
fileUtils.getExtensionFromMime = jest.fn();
describe('getCacheFile', () => {
it('should return a path with correct extension for a non-cached file using file name', async () => {
RNFetchBlob.fs.exists.mockReturnValue(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValue(null);
const extensions = ['.pdf', '.png', '.bmp', '.jpg', '.jpeg'];
const fileUri = 'https://file-uri';
for (const ext of extensions) {
const fileName = `file${ext}`;
const {exists, path} = await getCacheFile(fileName, fileUri); // eslint-disable-line no-await-in-loop
expect(exists).toEqual(false);
expect(path.endsWith(ext)).toEqual(true);
}
});
it('should return a path with correct extension for a non-cached file using file uri', async () => {
RNFetchBlob.fs.exists.mockReturnValue(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValue(null);
const extensions = ['.pdf', '.png', '.bmp', '.jpg', '.jpeg'];
const fileName = '';
for (const ext of extensions) {
const fileUri = `https://file-uri/file${ext}`;
const {exists, path} = await getCacheFile(fileName, fileUri); // eslint-disable-line no-await-in-loop
expect(exists).toEqual(false);
expect(path.endsWith(ext)).toEqual(true);
}
});
it('should return a path with correct extension for a cached file using file name', async () => {
RNFetchBlob.fs.exists.mockReturnValue(true);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValue(null);
const extensions = ['.pdf', '.png', '.bmp', '.jpg', '.jpeg'];
const fileUri = 'https://file-uri';
for (const ext of extensions) {
const fileName = `file${ext}`;
const {exists, path} = await getCacheFile(fileName, fileUri); // eslint-disable-line no-await-in-loop
expect(exists).toEqual(true);
expect(path.endsWith(ext)).toEqual(true);
}
});
it('should return a path with correct extension for a cached file using file uri', async () => {
RNFetchBlob.fs.exists.mockReturnValue(true);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValue(null);
const extensions = ['.pdf', '.png', '.bmp', '.jpg', '.jpeg'];
const fileName = '';
for (const ext of extensions) {
const fileUri = `https://file-uri/file${ext}`;
const {exists, path} = await getCacheFile(fileName, fileUri); // eslint-disable-line no-await-in-loop
expect(exists).toEqual(true);
expect(path.endsWith(ext)).toEqual(true);
}
});
it('should return a path with default extension for a file and uri without an extension', async () => {
RNFetchBlob.fs.exists.mockReturnValue(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValue(null);
const defaultExt = 'png';
fileUtils.getExtensionFromMime.mockReturnValue(defaultExt);
const fileNames = ['', 'file'];
const fileUris = ['', 'https://file-uri/file'];
for (const fileName of fileNames) {
for (const fileUri of fileUris) {
const {exists, path} = await getCacheFile(fileName, fileUri); // eslint-disable-line no-await-in-loop
expect(exists).toEqual(false);
expect(path.endsWith(`.${defaultExt}`)).toEqual(true);
}
}
});
it('should return a path with a different extension for a file cached with the different extension', async () => {
const pathWithDiffExt = '/path/to/file.jpeg';
RNFetchBlob.fs.exists.mockReturnValue(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValue(pathWithDiffExt);
const fileNamesUris = [
{fileName: '', fileUri: 'https://file-uri/file.png'},
{fileName: 'file.png', fileUri: ''},
];
for (const {fileName, fileUri} of fileNamesUris) {
const {exists, path} = await getCacheFile(fileName, fileUri); // eslint-disable-line no-await-in-loop
expect(exists).toEqual(true);
expect(path).toEqual(pathWithDiffExt);
}
});
});
describe('ImageCacheManager.cache', () => {
const imageCacheManagerUtils = require('app/utils/image_cache_manager');
imageCacheManagerUtils.isDownloading = jest.fn();
RNFetchBlob.config.mockReturnValue(RNFetchBlob);
mattermostBucket.getPreference = jest.fn().mockReturnValue({});
beforeEach(() => {
ImageCacheManager.listeners = {};
});
it('should cache a file from an http uri', async () => {
const fileName = '';
const fileUris = ['http://file-uri', 'https://file-uri'];
for (const fileUri of fileUris) {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: '', headers: {}}});
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
expect(path).not.toEqual(null);
}
});
it('should get the extension from the content disposition', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
const ext = '.bmp';
const headers = {'Content-Disposition': `inline;filename="file${ext}"; filename*=UTF-8''file${ext}`};
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: '', headers}});
const fileName = '';
const fileUri = 'https://file-uri';
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
expect(path.endsWith(ext)).toEqual(true);
});
it('should get the extension from the content type', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
const ext = 'jpg';
const headers = {'Content-Type': 'image/jpeg'};
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: '', headers}});
fileUtils.getExtensionFromMime.
mockReturnValueOnce(ext). // first call in getCacheFile
mockReturnValueOnce(ext); // seconds call in cache using MIME type from header
const fileName = '';
const fileUri = 'https://file-uri';
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
expect(path.endsWith(`.${ext}`)).toEqual(true);
});
it('should default to .png', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
const ext = 'png';
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: '', headers: {}}});
fileUtils.getExtensionFromMime.
mockReturnValueOnce(ext). // first call in getCacheFile
mockReturnValueOnce(null). // second call in cache using MIME type from header
mockReturnValueOnce(ext); // third call in cache using DEFAULT_MIME_TYPE
const fileName = '';
const fileUri = 'https://file-uri';
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
expect(path.endsWith(`.${ext}`)).toEqual(true);
});
it('should move file if path extension and extracted extension don\'t match', async () => {
const oldExt = 'png';
const fileNameUris = [
{fileName: '', fileUri: `https://file-uri/file.${oldExt}`},
{fileName: `file${oldExt}`, fileUri: `https://file-uri/file.${oldExt}`},
];
for (const {fileName, fileUri} of fileNameUris) {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
const ext = 'bmp';
const headers = {'Content-Type': 'image/bmp'};
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: '', headers}});
fileUtils.getExtensionFromMime.
mockReturnValueOnce(ext). // first call in getCacheFile
mockReturnValueOnce(ext); // second call in cache using MIME type from header
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
expect(path.endsWith(`.${ext}`)).toEqual(true);
const oldPath = path.replace(ext, oldExt);
expect(RNFetchBlob.fs.mv).toHaveBeenCalledWith(oldPath, path);
}
});
it('should return cached file when it exists', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(true);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
const ext = '.png';
const fileName = `file${ext}`;
const fileUri = `http://file-uri/file${ext}`;
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
expect(path.endsWith(ext)).toEqual(true);
});
it('should not cache the file when respInfo.respType === "text"', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: 'text', headers: {}}});
const fileName = '';
const fileUri = 'https://file-uri';
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
expect(path).toEqual(null);
expect(RNFetchBlob.fs.unlink).toHaveBeenCalled();
});
it('should add the listener if it is downloading', async () => {
imageCacheManagerUtils.isDownloading.mockReturnValueOnce(true);
const fileName = 'file.png';
const fileUri = 'http://file-uri/file.png';
await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
const expectedListeners = {[fileUri]: [emptyFunction]};
expect(ImageCacheManager.listeners).toMatchObject(expectedListeners);
});
it('should call listener with path if file exists', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(true);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
const fileName = 'file.png';
const fileUri = 'http://file-uri/file.png';
const listener = jest.fn();
const path = await ImageCacheManager.cache(fileName, fileUri, listener); // eslint-disable-line no-await-in-loop
expect(ImageCacheManager.listeners).toMatchObject({});
expect(listener).toHaveBeenCalledWith(path);
});
it('should call all listeners with path after downloading file then remove listeners', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: '', headers: {}}});
// Ensure isDownloading returns false so that we can proceed to
// download the file even if there are existing listeners.
imageCacheManagerUtils.isDownloading.mockReturnValueOnce(false);
const fileName = 'file.png';
const fileUri = 'http://file-uri/file.png';
const existingListener = jest.fn();
ImageCacheManager.listeners = {[fileUri]: [existingListener]};
const newListener = jest.fn();
const path = await ImageCacheManager.cache(fileName, fileUri, newListener); // eslint-disable-line no-await-in-loop
expect(ImageCacheManager.listeners).toMatchObject({});
expect(existingListener).toHaveBeenCalledWith(path);
expect(newListener).toHaveBeenCalledWith(path);
});
it('should call all listeners with the local file', async () => {
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
// Ensure isDownloading returns false so that we can proceed to
// download the file even if there are existing listeners.
imageCacheManagerUtils.isDownloading.mockReturnValueOnce(false);
const fileName = 'file.png';
const fileUri = 'file://file-uri/file.png';
const existingListener = jest.fn();
ImageCacheManager.listeners = {[fileUri]: [existingListener]};
const newListener = jest.fn();
await ImageCacheManager.cache(fileName, fileUri, newListener); // eslint-disable-line no-await-in-loop
expect(ImageCacheManager.listeners).toMatchObject({});
expect(existingListener).toHaveBeenCalledWith(fileUri);
expect(newListener).toHaveBeenCalledWith(fileUri);
});
it('should parse the uri to get the correct protocol', async () => {
const fileUri = 'HTTPS://file-uri/ABC123';
const expectedFileUri = 'https://file-uri/ABC123';
const fileName = null;
const incorrectHash = hashCode(fileUri);
const expectedHash = hashCode(expectedFileUri);
expect(incorrectHash).not.toBe(expectedHash);
RNFetchBlob.fs.exists.mockReturnValueOnce(false);
RNFetchBlob.fs.existsWithDiffExt.mockReturnValueOnce(null);
RNFetchBlob.fetch.mockReturnValueOnce({respInfo: {respType: '', headers: {}}});
const path = await ImageCacheManager.cache(fileName, fileUri, emptyFunction); // eslint-disable-line no-await-in-loop
const localFilename = path.substring(path.lastIndexOf('/') + 1, path.length);
expect(localFilename).toEqual(`${expectedHash}.png`);
});
});