mirror of
https://github.com/balena-io/etcher.git
synced 2025-04-23 23:07:17 +00:00

`image-stream` returns image objects that look like this: ```js { stream: <readable stream>, transform: <transform stream>, size: { original: <number>, final: { value: <number>, estimation: <boolean> } }, ... } ``` While the GUI handles image objects that look like this: ```sh { path: <string>, size: <number>, ... } ``` It looks like we should share a common structure between both, so we can use `image-stream` images in `drive-constraints`, for example. Turns out that we actually transform `image-stream` image objects to GUI image objects when the user selects an image using the image selector dialog, which is another indicator that we should normalise this situation. As a solution, this commit does the following: - Add `path` to `image-stream` image object - Reuse `image-stream` image objects in the GUI, given they are a superset of GUI image objects See: https://github.com/resin-io/etcher/pull/1223#discussion_r108165110 Fixes: https://github.com/resin-io/etcher/issues/1232 Signed-off-by: Juan Cruz Viotti <jviotti@openmailbox.org>
212 lines
6.0 KiB
JavaScript
212 lines
6.0 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 path = require('path');
|
|
const Bluebird = require('bluebird');
|
|
const _ = require('lodash');
|
|
const PassThroughStream = require('stream').PassThrough;
|
|
const supportedFileTypes = require('./supported');
|
|
const utils = require('./utils');
|
|
const errors = require('../shared/errors');
|
|
|
|
/**
|
|
* @summary Archive metadata base path
|
|
* @constant
|
|
* @private
|
|
* @type {String}
|
|
*/
|
|
const ARCHIVE_METADATA_BASE_PATH = '.meta';
|
|
|
|
/**
|
|
* @summary Image extensions
|
|
* @constant
|
|
* @private
|
|
* @type {String[]}
|
|
*/
|
|
const IMAGE_EXTENSIONS = _.reduce(supportedFileTypes, (accumulator, file) => {
|
|
if (file.type === 'image') {
|
|
accumulator.push(file.extension);
|
|
}
|
|
|
|
return accumulator;
|
|
}, []);
|
|
|
|
/**
|
|
* @summary Extract entry by path
|
|
* @function
|
|
* @private
|
|
*
|
|
* @param {String} archive - archive
|
|
* @param {String} filePath - entry file path
|
|
* @param {Object} options - options
|
|
* @param {Object} options.hooks - archive hooks
|
|
* @param {Object[]} options.entries - archive entries
|
|
* @param {*} [options.default] - entry default value
|
|
* @fulfil {*} contents
|
|
* @returns {Promise}
|
|
*
|
|
* extractEntryByPath('my/archive.zip', '_info/logo.svg', {
|
|
* hooks: { ... },
|
|
* entries: [ ... ],
|
|
* default: ''
|
|
* }).then((contents) => {
|
|
* console.log(contents);
|
|
* });
|
|
*/
|
|
const extractEntryByPath = (archive, filePath, options) => {
|
|
const fileEntry = _.find(options.entries, (entry) => {
|
|
return _.chain(entry.name)
|
|
.split('/')
|
|
.tail()
|
|
.join('/')
|
|
.value() === filePath;
|
|
});
|
|
|
|
if (!fileEntry) {
|
|
return Bluebird.resolve(options.default);
|
|
}
|
|
|
|
return options.hooks.extractFile(archive, options.entries, fileEntry.name)
|
|
.then(utils.extractStream);
|
|
};
|
|
|
|
/**
|
|
* @summary Extract archive metadata
|
|
* @function
|
|
* @private
|
|
*
|
|
* @param {String} archive - archive
|
|
* @param {String} basePath - metadata base path
|
|
* @param {Object} options - options
|
|
* @param {Object[]} options.entries - archive entries
|
|
* @param {Object} options.hooks - archive hooks
|
|
* @fulfil {Object} - metadata
|
|
* @returns {Promise}
|
|
*
|
|
* extractArchiveMetadata('my/archive.zip', '.meta', {
|
|
* hooks: { ... },
|
|
* entries: [ ... ]
|
|
* }).then((metadata) => {
|
|
* console.log(metadata);
|
|
* });
|
|
*/
|
|
const extractArchiveMetadata = (archive, basePath, options) => {
|
|
return Bluebird.props({
|
|
logo: extractEntryByPath(archive, `${basePath}/logo.svg`, options),
|
|
instructions: extractEntryByPath(archive, `${basePath}/instructions.markdown`, options),
|
|
bmap: extractEntryByPath(archive, `${basePath}/image.bmap`, options),
|
|
manifest: _.attempt(() => {
|
|
return extractEntryByPath(archive, `${basePath}/manifest.json`, {
|
|
hooks: options.hooks,
|
|
entries: options.entries,
|
|
default: '{}'
|
|
}).then((manifest) => {
|
|
try {
|
|
return JSON.parse(manifest);
|
|
} catch (parseError) {
|
|
throw errors.createUserError(
|
|
'Invalid archive manifest.json',
|
|
'The archive manifest.json file is not valid JSON'
|
|
);
|
|
}
|
|
});
|
|
})
|
|
}).then((results) => {
|
|
return {
|
|
name: results.manifest.name,
|
|
version: results.manifest.version,
|
|
url: results.manifest.url,
|
|
supportUrl: results.manifest.supportUrl,
|
|
releaseNotesUrl: results.manifest.releaseNotesUrl,
|
|
checksumType: results.manifest.checksumType,
|
|
checksum: results.manifest.checksum,
|
|
bytesToZeroOutFromTheBeginning: results.manifest.bytesToZeroOutFromTheBeginning,
|
|
recommendedDriveSize: results.manifest.recommendedDriveSize,
|
|
logo: _.invoke(results.logo, [ 'toString' ]),
|
|
bmap: _.invoke(results.bmap, [ 'toString' ]),
|
|
instructions: _.invoke(results.instructions, [ 'toString' ])
|
|
};
|
|
});
|
|
};
|
|
|
|
/**
|
|
* @summary Extract image from archive
|
|
* @function
|
|
* @public
|
|
*
|
|
* @param {String} archive - archive path
|
|
* @param {Object} hooks - archive hooks
|
|
* @param {Function} hooks.getEntries - get entries hook
|
|
* @param {Function} hooks.extractFile - extract file hook
|
|
* @fulfil {Object} image metadata
|
|
* @returns {Promise}
|
|
*
|
|
* @example
|
|
* archive.extractImage('path/to/my/archive.zip', {
|
|
* getEntries: (archive) => {
|
|
* return [ ..., ..., ... ];
|
|
* },
|
|
* extractFile: (archive, entries, file) => {
|
|
* ...
|
|
* }
|
|
* }).then((image) => {
|
|
* image.stream.pipe(image.transform).pipe(...);
|
|
* });
|
|
*/
|
|
exports.extractImage = (archive, hooks) => {
|
|
return hooks.getEntries(archive).then((entries) => {
|
|
|
|
const imageEntries = _.filter(entries, (entry) => {
|
|
const extension = _.toLower(_.replace(path.extname(entry.name), '.', ''));
|
|
return _.includes(IMAGE_EXTENSIONS, extension);
|
|
});
|
|
|
|
const VALID_NUMBER_OF_IMAGE_ENTRIES = 1;
|
|
if (imageEntries.length !== VALID_NUMBER_OF_IMAGE_ENTRIES) {
|
|
throw errors.createUserError(
|
|
'Invalid archive image',
|
|
'The archive image should contain one and only one top image file'
|
|
);
|
|
}
|
|
|
|
const imageEntry = _.first(imageEntries);
|
|
|
|
return Bluebird.props({
|
|
imageStream: hooks.extractFile(archive, entries, imageEntry.name),
|
|
metadata: extractArchiveMetadata(archive, ARCHIVE_METADATA_BASE_PATH, {
|
|
entries,
|
|
hooks
|
|
})
|
|
}).then((results) => {
|
|
results.metadata.stream = results.imageStream;
|
|
results.metadata.transform = new PassThroughStream();
|
|
results.metadata.path = archive;
|
|
|
|
results.metadata.size = {
|
|
original: imageEntry.size,
|
|
final: {
|
|
estimation: false,
|
|
value: imageEntry.size
|
|
}
|
|
};
|
|
|
|
return results.metadata;
|
|
});
|
|
});
|
|
};
|