Jonas Hermsmeier becca2d05e fix(image-stream): Remove inability to fallback to octet-stream (#1363)
Using `mime-types` in that place made it impossible to use other
other file extensions like `.sdcard` and have them treated as `application/octet-stream`
when drag and dropping images into Etcher.

This moves the fallback logic in `lib/image-stream` to `.getFromFilePath()`,
to still facilitate proper MIME type detection, while allowing it to fall back
to the octet-stream handler, if there's no available handler for a specific type.

Change-Type: patch
Changelog-Entry: Fix not treating unknown images as octet-stream
2017-04-27 18:00:53 +02:00

96 lines
2.5 KiB
JavaScript

/*
* Copyright 2016 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 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
* @function
* @public
*
* @description
* You should be careful when using this function, since you can only
* extract files that are not bigger than the available computer memory.
*
* @param {StreamReadable} stream - stream
* @fulfil {Buffer} - data
* @returns {Promise}
*
* @example
* const stream = fs.createReadStream('./foo/bar');
*
* utils.extractStream(stream).then((data) => {
* console.log(data.toString());
* });
*/
exports.extractStream = (stream) => {
return new Bluebird((resolve, reject) => {
const chunks = [];
stream.on('data', (chunk) => {
chunks.push(chunk);
});
stream.on('error', reject);
stream.on('end', () => {
resolve(Buffer.concat(chunks));
});
});
};