[react-packager] implement /assets endpoint to serve assets
This commit is contained in:
parent
fd8bc3b5f6
commit
fb7036eaac
|
@ -0,0 +1,85 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
jest
|
||||||
|
.autoMockOff()
|
||||||
|
.mock('../../lib/declareOpts')
|
||||||
|
.mock('fs');
|
||||||
|
|
||||||
|
var fs = require('fs');
|
||||||
|
var AssetServer = require('../');
|
||||||
|
var Promise = require('bluebird');
|
||||||
|
|
||||||
|
describe('AssetServer', function() {
|
||||||
|
pit('should work for the simple case', function() {
|
||||||
|
var server = new AssetServer({
|
||||||
|
projectRoots: ['/root'],
|
||||||
|
assetExts: ['png'],
|
||||||
|
});
|
||||||
|
|
||||||
|
fs.__setMockFilesystem({
|
||||||
|
'root': {
|
||||||
|
imgs: {
|
||||||
|
'b.png': 'b image',
|
||||||
|
'b@2x.png': 'b2 image',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return Promise.all([
|
||||||
|
server.get('imgs/b.png'),
|
||||||
|
server.get('imgs/b@1x.png'),
|
||||||
|
]).then(function(resp) {
|
||||||
|
resp.forEach(function(data) {
|
||||||
|
expect(data).toBe('b image');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
pit.only('should pick the bigger one', function() {
|
||||||
|
var server = new AssetServer({
|
||||||
|
projectRoots: ['/root'],
|
||||||
|
assetExts: ['png'],
|
||||||
|
});
|
||||||
|
|
||||||
|
fs.__setMockFilesystem({
|
||||||
|
'root': {
|
||||||
|
imgs: {
|
||||||
|
'b@1x.png': 'b1 image',
|
||||||
|
'b@2x.png': 'b2 image',
|
||||||
|
'b@4x.png': 'b4 image',
|
||||||
|
'b@4.5x.png': 'b4.5 image',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return server.get('imgs/b@3x.png').then(function(data) {
|
||||||
|
expect(data).toBe('b4 image');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
pit('should support multiple project roots', function() {
|
||||||
|
var server = new AssetServer({
|
||||||
|
projectRoots: ['/root'],
|
||||||
|
assetExts: ['png'],
|
||||||
|
});
|
||||||
|
|
||||||
|
fs.__setMockFilesystem({
|
||||||
|
'root': {
|
||||||
|
imgs: {
|
||||||
|
'b.png': 'b image',
|
||||||
|
},
|
||||||
|
'root2': {
|
||||||
|
'newImages': {
|
||||||
|
'imgs': {
|
||||||
|
'b@1x.png': 'b1 image',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return server.get('newImages/imgs/b.png').then(function(data) {
|
||||||
|
expect(data).toBe('b1 image');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,132 @@
|
||||||
|
/**
|
||||||
|
* Copyright (c) 2015-present, Facebook, Inc.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* This source code is licensed under the BSD-style license found in the
|
||||||
|
* LICENSE file in the root directory of this source tree. An additional grant
|
||||||
|
* of patent rights can be found in the PATENTS file in the same directory.
|
||||||
|
*/
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
var declareOpts = require('../lib/declareOpts');
|
||||||
|
var extractAssetResolution = require('../lib/extractAssetResolution');
|
||||||
|
var path = require('path');
|
||||||
|
var Promise = require('bluebird');
|
||||||
|
var fs = require('fs');
|
||||||
|
|
||||||
|
var lstat = Promise.promisify(fs.lstat);
|
||||||
|
var readDir = Promise.promisify(fs.readdir);
|
||||||
|
var readFile = Promise.promisify(fs.readFile);
|
||||||
|
|
||||||
|
module.exports = AssetServer;
|
||||||
|
|
||||||
|
var validateOpts = declareOpts({
|
||||||
|
projectRoots: {
|
||||||
|
type: 'array',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
assetExts: {
|
||||||
|
type: 'array',
|
||||||
|
default: ['png'],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
function AssetServer(options) {
|
||||||
|
var opts = validateOpts(options);
|
||||||
|
this._roots = opts.projectRoots;
|
||||||
|
this._assetExts = opts.assetExts;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a request for an image by path. That could contain a resolution
|
||||||
|
* postfix, we need to find that image (or the closest one to it's resolution)
|
||||||
|
* in one of the project roots:
|
||||||
|
*
|
||||||
|
* 1. We first parse the directory of the asset
|
||||||
|
* 2. We check to find a matching directory in one of the project roots
|
||||||
|
* 3. We then build a map of all assets and their resolutions in this directory
|
||||||
|
* 4. Then pick the closest resolution (rounding up) to the requested one
|
||||||
|
*/
|
||||||
|
|
||||||
|
AssetServer.prototype.get = function(assetPath) {
|
||||||
|
var filename = path.basename(assetPath);
|
||||||
|
|
||||||
|
return findRoot(
|
||||||
|
this._roots,
|
||||||
|
path.dirname(assetPath)
|
||||||
|
).then(function(dir) {
|
||||||
|
return [
|
||||||
|
dir,
|
||||||
|
readDir(dir),
|
||||||
|
];
|
||||||
|
}).spread(function(dir, files) {
|
||||||
|
// Easy case. File exactly what the client requested.
|
||||||
|
var index = files.indexOf(filename);
|
||||||
|
if (index > -1) {
|
||||||
|
return readFile(path.join(dir, filename));
|
||||||
|
}
|
||||||
|
|
||||||
|
var assetData = extractAssetResolution(filename);
|
||||||
|
var map = buildAssetMap(dir, files);
|
||||||
|
var record = map[assetData.assetName];
|
||||||
|
|
||||||
|
if (!record) {
|
||||||
|
throw new Error('Asset not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var i = 0; i < record.resolutions.length; i++) {
|
||||||
|
if (record.resolutions[i] >= assetData.resolution) {
|
||||||
|
return readFile(record.files[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return readFile(record.files[record.files.length - 1]);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
function findRoot(roots, dir) {
|
||||||
|
return Promise.some(
|
||||||
|
roots.map(function(root) {
|
||||||
|
var absPath = path.join(root, dir);
|
||||||
|
return lstat(absPath).then(function(stat) {
|
||||||
|
if (!stat.isDirectory()) {
|
||||||
|
throw new Error('Looking for dirs');
|
||||||
|
}
|
||||||
|
stat._path = absPath;
|
||||||
|
return stat;
|
||||||
|
});
|
||||||
|
}),
|
||||||
|
1
|
||||||
|
).spread(
|
||||||
|
function(stat) {
|
||||||
|
return stat._path;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildAssetMap(dir, files) {
|
||||||
|
var assets = files.map(extractAssetResolution);
|
||||||
|
var map = Object.create(null);
|
||||||
|
assets.forEach(function(asset, i) {
|
||||||
|
var file = files[i];
|
||||||
|
var record = map[asset.assetName];
|
||||||
|
if (!record) {
|
||||||
|
record = map[asset.assetName] = {
|
||||||
|
resolutions: [],
|
||||||
|
files: [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
var insertIndex;
|
||||||
|
var length = record.resolutions.length;
|
||||||
|
for (insertIndex = 0; insertIndex < length; insertIndex++) {
|
||||||
|
if (asset.resolution < record.resolutions[insertIndex]) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
record.resolutions.splice(insertIndex, 0, asset.resolution);
|
||||||
|
record.files.splice(insertIndex, 0, path.join(dir, file));
|
||||||
|
});
|
||||||
|
|
||||||
|
return map;
|
||||||
|
}
|
|
@ -14,6 +14,7 @@ jest
|
||||||
.dontMock('absolute-path')
|
.dontMock('absolute-path')
|
||||||
.dontMock('../docblock')
|
.dontMock('../docblock')
|
||||||
.dontMock('../../replacePatterns')
|
.dontMock('../../replacePatterns')
|
||||||
|
.dontMock('../../../../lib/extractAssetResolution')
|
||||||
.setMock('../../../ModuleDescriptor', function(data) {return data;});
|
.setMock('../../../ModuleDescriptor', function(data) {return data;});
|
||||||
|
|
||||||
describe('DependencyGraph', function() {
|
describe('DependencyGraph', function() {
|
||||||
|
|
|
@ -18,6 +18,7 @@ var isAbsolutePath = require('absolute-path');
|
||||||
var debug = require('debug')('DependecyGraph');
|
var debug = require('debug')('DependecyGraph');
|
||||||
var util = require('util');
|
var util = require('util');
|
||||||
var declareOpts = require('../../../lib/declareOpts');
|
var declareOpts = require('../../../lib/declareOpts');
|
||||||
|
var extractAssetResolution = require('../../../lib/extractAssetResolution');
|
||||||
|
|
||||||
var readFile = Promise.promisify(fs.readFile);
|
var readFile = Promise.promisify(fs.readFile);
|
||||||
var readDir = Promise.promisify(fs.readdir);
|
var readDir = Promise.promisify(fs.readdir);
|
||||||
|
@ -421,7 +422,7 @@ DependecyGraph.prototype._processModule = function(modulePath) {
|
||||||
var module;
|
var module;
|
||||||
|
|
||||||
if (this._assetExts.indexOf(extname(modulePath)) > -1) {
|
if (this._assetExts.indexOf(extname(modulePath)) > -1) {
|
||||||
var assetData = extractResolutionPostfix(this._lookupName(modulePath));
|
var assetData = extractAssetResolution(this._lookupName(modulePath));
|
||||||
moduleData.id = assetData.assetName;
|
moduleData.id = assetData.assetName;
|
||||||
moduleData.resolution = assetData.resolution;
|
moduleData.resolution = assetData.resolution;
|
||||||
moduleData.isAsset = true;
|
moduleData.isAsset = true;
|
||||||
|
@ -772,28 +773,6 @@ function extname(name) {
|
||||||
return path.extname(name).replace(/^\./, '');
|
return path.extname(name).replace(/^\./, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
function extractResolutionPostfix(filename) {
|
|
||||||
var ext = extname(filename);
|
|
||||||
var re = new RegExp('@([\\d\\.]+)x\\.' + ext + '$');
|
|
||||||
|
|
||||||
var match = filename.match(re);
|
|
||||||
var resolution;
|
|
||||||
|
|
||||||
if (!(match && match[1])) {
|
|
||||||
resolution = 1;
|
|
||||||
} else {
|
|
||||||
resolution = parseFloat(match[1], 10);
|
|
||||||
if (isNaN(resolution)) {
|
|
||||||
resolution = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
resolution: resolution,
|
|
||||||
assetName: match ? filename.replace(re, '.' + ext) : filename,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function NotFoundError() {
|
function NotFoundError() {
|
||||||
Error.call(this);
|
Error.call(this);
|
||||||
Error.captureStackTrace(this, this.constructor);
|
Error.captureStackTrace(this, this.constructor);
|
||||||
|
|
|
@ -111,7 +111,7 @@ describe('Packager', function() {
|
||||||
var imgModule = {
|
var imgModule = {
|
||||||
isStatic: true,
|
isStatic: true,
|
||||||
path: '/root/img/new_image.png',
|
path: '/root/img/new_image.png',
|
||||||
uri: 'img/new_image.png',
|
uri: 'assets/img/new_image.png',
|
||||||
width: 25,
|
width: 25,
|
||||||
height: 50,
|
height: 50,
|
||||||
};
|
};
|
||||||
|
|
|
@ -194,7 +194,7 @@ function generateAssetModule(module, relPath) {
|
||||||
var img = {
|
var img = {
|
||||||
isStatic: true,
|
isStatic: true,
|
||||||
path: module.path, //TODO(amasad): this should be path inside tar file.
|
path: module.path, //TODO(amasad): this should be path inside tar file.
|
||||||
uri: relPath,
|
uri: path.join('assets', relPath),
|
||||||
width: dimensions.width / module.resolution,
|
width: dimensions.width / module.resolution,
|
||||||
height: dimensions.height / module.resolution,
|
height: dimensions.height / module.resolution,
|
||||||
};
|
};
|
||||||
|
|
|
@ -229,4 +229,32 @@ describe('processRequest', function() {
|
||||||
expect(res.end).not.toBeCalled();
|
expect(res.end).not.toBeCalled();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe.only('/assets endpoint', function() {
|
||||||
|
var AssetServer;
|
||||||
|
beforeEach(function() {
|
||||||
|
AssetServer = require('../../AssetServer');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should serve simple case', function() {
|
||||||
|
var req = {
|
||||||
|
url: '/assets/imgs/a.png',
|
||||||
|
};
|
||||||
|
var res = {
|
||||||
|
end: jest.genMockFn(),
|
||||||
|
};
|
||||||
|
|
||||||
|
AssetServer.prototype.get.mockImpl(function() {
|
||||||
|
return Promise.resolve('i am image');
|
||||||
|
});
|
||||||
|
|
||||||
|
server.processRequest(req, res);
|
||||||
|
jest.runAllTimers();
|
||||||
|
expect(res.end).toBeCalledWith('i am image');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 404', function() {
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -14,6 +14,7 @@ var declareOpts = require('../lib/declareOpts');
|
||||||
var FileWatcher = require('../FileWatcher');
|
var FileWatcher = require('../FileWatcher');
|
||||||
var Packager = require('../Packager');
|
var Packager = require('../Packager');
|
||||||
var Activity = require('../Activity');
|
var Activity = require('../Activity');
|
||||||
|
var AssetServer = require('../AssetServer');
|
||||||
var Promise = require('bluebird');
|
var Promise = require('bluebird');
|
||||||
var _ = require('underscore');
|
var _ = require('underscore');
|
||||||
|
|
||||||
|
@ -99,6 +100,11 @@ function Server(options) {
|
||||||
packagerOpts.fileWatcher = this._fileWatcher;
|
packagerOpts.fileWatcher = this._fileWatcher;
|
||||||
this._packager = new Packager(packagerOpts);
|
this._packager = new Packager(packagerOpts);
|
||||||
|
|
||||||
|
this._assetServer = new AssetServer({
|
||||||
|
projectRoots: opts.projectRoots,
|
||||||
|
assetExts: opts.assetExts,
|
||||||
|
});
|
||||||
|
|
||||||
var onFileChange = this._onFileChange.bind(this);
|
var onFileChange = this._onFileChange.bind(this);
|
||||||
this._fileWatcher.on('all', onFileChange);
|
this._fileWatcher.on('all', onFileChange);
|
||||||
|
|
||||||
|
@ -230,6 +236,22 @@ Server.prototype._processOnChangeRequest = function(req, res) {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Server.prototype._processAssetsRequest = function(req, res) {
|
||||||
|
var urlObj = url.parse(req.url, true);
|
||||||
|
var assetPath = urlObj.pathname.match(/^\/assets\/(.+)$/);
|
||||||
|
this._assetServer.get(assetPath[1])
|
||||||
|
.then(
|
||||||
|
function(data) {
|
||||||
|
res.end(data);
|
||||||
|
},
|
||||||
|
function(error) {
|
||||||
|
console.error(error.stack);
|
||||||
|
res.writeHead('404');
|
||||||
|
res.end('Asset not found');
|
||||||
|
}
|
||||||
|
).done();
|
||||||
|
};
|
||||||
|
|
||||||
Server.prototype.processRequest = function(req, res, next) {
|
Server.prototype.processRequest = function(req, res, next) {
|
||||||
var urlObj = url.parse(req.url, true);
|
var urlObj = url.parse(req.url, true);
|
||||||
var pathname = urlObj.pathname;
|
var pathname = urlObj.pathname;
|
||||||
|
@ -245,6 +267,9 @@ Server.prototype.processRequest = function(req, res, next) {
|
||||||
} else if (pathname.match(/^\/onchange\/?$/)) {
|
} else if (pathname.match(/^\/onchange\/?$/)) {
|
||||||
this._processOnChangeRequest(req, res);
|
this._processOnChangeRequest(req, res);
|
||||||
return;
|
return;
|
||||||
|
} else if (pathname.match(/^\/assets\//)) {
|
||||||
|
this._processAssetsRequest(req, res);
|
||||||
|
return;
|
||||||
} else {
|
} else {
|
||||||
next();
|
next();
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -42,6 +42,11 @@ fs.readdir.mockImpl(function(filepath, callback) {
|
||||||
});
|
});
|
||||||
|
|
||||||
fs.readFile.mockImpl(function(filepath, encoding, callback) {
|
fs.readFile.mockImpl(function(filepath, encoding, callback) {
|
||||||
|
if (arguments.length === 2) {
|
||||||
|
callback = encoding;
|
||||||
|
encoding = null;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
var node = getToNode(filepath);
|
var node = getToNode(filepath);
|
||||||
// dir check
|
// dir check
|
|
@ -0,0 +1,42 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
jest.autoMockOff();
|
||||||
|
var extractAssetResolution = require('../extractAssetResolution');
|
||||||
|
|
||||||
|
describe('extractAssetResolution', function() {
|
||||||
|
it('should extract resolution simple case', function() {
|
||||||
|
var data = extractAssetResolution('test@2x.png');
|
||||||
|
expect(data).toEqual({
|
||||||
|
assetName: 'test.png',
|
||||||
|
resolution: 2,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should default resolution to 1', function() {
|
||||||
|
var data = extractAssetResolution('test.png');
|
||||||
|
expect(data).toEqual({
|
||||||
|
assetName: 'test.png',
|
||||||
|
resolution: 1,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should support float', function() {
|
||||||
|
var data = extractAssetResolution('test@1.1x.png');
|
||||||
|
expect(data).toEqual({
|
||||||
|
assetName: 'test.png',
|
||||||
|
resolution: 1.1,
|
||||||
|
});
|
||||||
|
|
||||||
|
data = extractAssetResolution('test@.1x.png');
|
||||||
|
expect(data).toEqual({
|
||||||
|
assetName: 'test.png',
|
||||||
|
resolution: 0.1,
|
||||||
|
});
|
||||||
|
|
||||||
|
data = extractAssetResolution('test@0.2x.png');
|
||||||
|
expect(data).toEqual({
|
||||||
|
assetName: 'test.png',
|
||||||
|
resolution: 0.2,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,28 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
var path = require('path');
|
||||||
|
|
||||||
|
function extractAssetResolution(filename) {
|
||||||
|
var ext = path.extname(filename);
|
||||||
|
|
||||||
|
var re = new RegExp('@([\\d\\.]+)x\\' + ext + '$');
|
||||||
|
|
||||||
|
var match = filename.match(re);
|
||||||
|
var resolution;
|
||||||
|
|
||||||
|
if (!(match && match[1])) {
|
||||||
|
resolution = 1;
|
||||||
|
} else {
|
||||||
|
resolution = parseFloat(match[1], 10);
|
||||||
|
if (isNaN(resolution)) {
|
||||||
|
resolution = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
resolution: resolution,
|
||||||
|
assetName: match ? filename.replace(re, ext) : filename,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = extractAssetResolution;
|
Loading…
Reference in New Issue