mirror of
https://github.com/balena-io/etcher.git
synced 2025-07-18 16:56:32 +00:00
refactor(image-stream): extract part of utils.js into mime.js (#1594)
This is a small refactoring commit that extracts the MIME related function from utils.js into a new file called mime.js, and stores the default image MIME type there as a constant, to avoid duplicating it in multiple parts of the code. Change-Type: patch Signed-off-by: Juan Cruz Viotti <jv@jviotti.com>
This commit is contained in:
parent
a772877ae1
commit
37b7ea3b0a
2
.gitattributes
vendored
2
.gitattributes
vendored
@ -39,3 +39,5 @@ Makefile text
|
|||||||
*.zip binary diff=hex
|
*.zip binary diff=hex
|
||||||
*.dmg binary diff=hex
|
*.dmg binary diff=hex
|
||||||
*.rpi-sdcard binary diff=hex
|
*.rpi-sdcard binary diff=hex
|
||||||
|
*.foo binary diff=hex
|
||||||
|
xz-without-extension binary diff=hex
|
||||||
|
@ -20,7 +20,7 @@ const _ = require('lodash');
|
|||||||
const Bluebird = require('bluebird');
|
const Bluebird = require('bluebird');
|
||||||
const fs = Bluebird.promisifyAll(require('fs'));
|
const fs = Bluebird.promisifyAll(require('fs'));
|
||||||
const stream = require('stream');
|
const stream = require('stream');
|
||||||
const utils = require('./utils');
|
const mime = require('./mime');
|
||||||
const handlers = require('./handlers');
|
const handlers = require('./handlers');
|
||||||
const supportedFileTypes = require('./supported');
|
const supportedFileTypes = require('./supported');
|
||||||
const errors = require('../shared/errors');
|
const errors = require('../shared/errors');
|
||||||
@ -76,9 +76,8 @@ exports.getFromFilePath = (file) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
const MIME_TYPE_RAW_IMAGE = 'application/octet-stream';
|
const mimeType = _.has(handlers, type) ? type : mime.DEFAULT_MIME_TYPE;
|
||||||
const mimeType = _.has(handlers, type) ? type : MIME_TYPE_RAW_IMAGE;
|
|
||||||
return _.invoke(handlers, mimeType, file, {
|
return _.invoke(handlers, mimeType, file, {
|
||||||
size: fileStats.size
|
size: fileStats.size
|
||||||
});
|
});
|
||||||
|
65
lib/image-stream/mime.js
Normal file
65
lib/image-stream/mime.js
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 resin.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const _ = require('lodash');
|
||||||
|
const Bluebird = require('bluebird');
|
||||||
|
const fileType = require('file-type');
|
||||||
|
const mime = require('mime-types');
|
||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary The default MIME type
|
||||||
|
* @type {String}
|
||||||
|
* @constant
|
||||||
|
*/
|
||||||
|
exports.DEFAULT_MIME_TYPE = 'application/octet-stream';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Get file's mime type, by reading the initial 262 bytes if necessary
|
||||||
|
* @function
|
||||||
|
* @public
|
||||||
|
*
|
||||||
|
* @param {String} filename - file path
|
||||||
|
* @fulfil {String} - mime type
|
||||||
|
* @returns {Promise}
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* mime.getMimeTypeFromFileName('path/to/raspberrypi.img.gz').then((mimeType) => {
|
||||||
|
* console.log(mimeType);
|
||||||
|
* });
|
||||||
|
*/
|
||||||
|
exports.getMimeTypeFromFileName = (filename) => {
|
||||||
|
const mimeType = mime.lookup(filename);
|
||||||
|
|
||||||
|
if (mimeType) {
|
||||||
|
return Bluebird.resolve(mimeType);
|
||||||
|
}
|
||||||
|
|
||||||
|
const FILE_TYPE_ID_BYTES = 262;
|
||||||
|
|
||||||
|
return Bluebird.using(fs.openAsync(filename, 'r').disposer((fileDescriptor) => {
|
||||||
|
return fs.closeAsync(fileDescriptor);
|
||||||
|
}), (fileDescriptor) => {
|
||||||
|
const BUFFER_START = 0;
|
||||||
|
const buffer = Buffer.alloc(FILE_TYPE_ID_BYTES);
|
||||||
|
|
||||||
|
return fs.readAsync(fileDescriptor, buffer, BUFFER_START, FILE_TYPE_ID_BYTES, null).then(() => {
|
||||||
|
return _.get(fileType(buffer), [ 'mime' ], exports.DEFAULT_MIME_TYPE);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
@ -16,48 +16,7 @@
|
|||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
const _ = require('lodash');
|
|
||||||
const Bluebird = require('bluebird');
|
const Bluebird = require('bluebird');
|
||||||
const fileType = require('file-type');
|
|
||||||
const mime = require('mime-types');
|
|
||||||
const fs = require('fs');
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Get archive mime type
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {String} filename - file path
|
|
||||||
* @fulfil {String} - mime type
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* utils.getArchiveMimeType('path/to/raspberrypi.img.gz').then((mimeType) => {
|
|
||||||
* console.log(mimeType);
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
exports.getArchiveMimeType = (filename) => {
|
|
||||||
const MIME_TYPE_RAW_IMAGE = 'application/octet-stream';
|
|
||||||
const mimeType = mime.lookup(filename);
|
|
||||||
|
|
||||||
if (mimeType) {
|
|
||||||
return Bluebird.resolve(mimeType);
|
|
||||||
}
|
|
||||||
|
|
||||||
const FILE_TYPE_ID_BYTES = 262;
|
|
||||||
|
|
||||||
return Bluebird.using(fs.openAsync(filename, 'r').disposer((fileDescriptor) => {
|
|
||||||
return fs.closeAsync(fileDescriptor);
|
|
||||||
}), (fileDescriptor) => {
|
|
||||||
const BUFFER_START = 0;
|
|
||||||
const buffer = Buffer.alloc(FILE_TYPE_ID_BYTES);
|
|
||||||
|
|
||||||
return fs.readAsync(fileDescriptor, buffer, BUFFER_START, FILE_TYPE_ID_BYTES, null).then(() => {
|
|
||||||
return _.get(fileType(buffer), [ 'mime' ], MIME_TYPE_RAW_IMAGE);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @summary Extract the data of a readable stream
|
* @summary Extract the data of a readable stream
|
||||||
|
Binary file not shown.
BIN
tests/image-stream/data/unrecognized/xz-without-extension
Normal file
BIN
tests/image-stream/data/unrecognized/xz-without-extension
Normal file
Binary file not shown.
107
tests/image-stream/mime.spec.js
Normal file
107
tests/image-stream/mime.spec.js
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 resin.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const m = require('mochainon');
|
||||||
|
const path = require('path');
|
||||||
|
const DATA_PATH = path.join(__dirname, 'data');
|
||||||
|
const mime = require('../../lib/image-stream/mime');
|
||||||
|
|
||||||
|
describe('ImageStream: MIME', function() {
|
||||||
|
|
||||||
|
describe('.getMimeTypeFromFileName()', function() {
|
||||||
|
|
||||||
|
it('should resolve application/x-bzip2 for a bz2 archive', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'bz2', 'etcher-test.img.bz2');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/x-bzip2');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/x-xz for a xz archive', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'xz', 'etcher-test.img.xz');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/x-xz');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/gzip for a gz archive', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'gz', 'etcher-test.img.gz');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/gzip');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/zip for a zip archive', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'zip', 'zip-directory-etcher-only.zip');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/zip');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/octet-stream for an uncompressed image', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'images', 'etcher-test.img');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/octet-stream');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/x-iso9660-image for an uncompressed iso', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'images', 'etcher-test.iso');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/x-iso9660-image');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/x-apple-diskimage for a compressed Apple disk image', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'dmg', 'etcher-test-zlib.dmg');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/x-apple-diskimage');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/x-apple-diskimage for an uncompressed Apple disk image', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'dmg', 'etcher-test-raw.dmg');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/x-apple-diskimage');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve application/octet-stream for an unrecognized file type', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'unrecognized', 'random.rpi-sdcard');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/octet-stream');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve the correct MIME type given an invalid extension', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'unrecognized', 'xz-with-invalid-extension.foo');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/x-xz');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should resolve the correct MIME type given no extension', function() {
|
||||||
|
const file = path.join(DATA_PATH, 'unrecognized', 'xz-without-extension');
|
||||||
|
return mime.getMimeTypeFromFileName(file).then((type) => {
|
||||||
|
m.chai.expect(type).to.equal('application/x-xz');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
@ -17,80 +17,11 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
const m = require('mochainon');
|
const m = require('mochainon');
|
||||||
const path = require('path');
|
|
||||||
const StreamReadable = require('stream').Readable;
|
const StreamReadable = require('stream').Readable;
|
||||||
const DATA_PATH = path.join(__dirname, 'data');
|
|
||||||
const utils = require('../../lib/image-stream/utils');
|
const utils = require('../../lib/image-stream/utils');
|
||||||
|
|
||||||
describe('ImageStream: Utils', function() {
|
describe('ImageStream: Utils', function() {
|
||||||
|
|
||||||
describe('.getArchiveMimeType()', function() {
|
|
||||||
|
|
||||||
it('should resolve application/x-bzip2 for a bz2 archive', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'bz2', 'etcher-test.img.bz2');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/x-bzip2');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/x-xz for a xz archive', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'xz', 'etcher-test.img.xz');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/x-xz');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/gzip for a gz archive', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'gz', 'etcher-test.img.gz');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/gzip');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/zip for a zip archive', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'zip', 'zip-directory-etcher-only.zip');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/zip');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/octet-stream for an uncompressed image', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'images', 'etcher-test.img');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/octet-stream');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/x-iso9660-image for an uncompressed iso', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'images', 'etcher-test.iso');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/x-iso9660-image');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/x-apple-diskimage for a compressed Apple disk image', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'dmg', 'etcher-test-zlib.dmg');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/x-apple-diskimage');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/x-apple-diskimage for an uncompressed Apple disk image', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'dmg', 'etcher-test-raw.dmg');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/x-apple-diskimage');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve application/octet-stream for an unrecognized file type', function() {
|
|
||||||
const file = path.join(DATA_PATH, 'unrecognized', 'random.rpi-sdcard');
|
|
||||||
return utils.getArchiveMimeType(file).then((type) => {
|
|
||||||
m.chai.expect(type).to.equal('application/octet-stream');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('.extractStream()', function() {
|
describe('.extractStream()', function() {
|
||||||
|
|
||||||
describe('given a stream that emits data', function() {
|
describe('given a stream that emits data', function() {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user