More work on source-map support for node.js

Relates to #656
This commit is contained in:
Jürg Lehni 2016-01-26 12:38:58 +01:00
parent c479ec9272
commit 46f415ca81
2 changed files with 74 additions and 49 deletions

View file

@ -95,8 +95,6 @@ Base.exports.PaperScript = (function() {
return scope.acorn.parse(code, options); return scope.acorn.parse(code, options);
} }
var sourceMaps = {};
/** /**
* Compiles PaperScript code into JavaScript code. * Compiles PaperScript code into JavaScript code.
* *
@ -104,13 +102,14 @@ Base.exports.PaperScript = (function() {
* @function * @function
* *
* @option options.url {String} the url of the source, for source-map * @option options.url {String} the url of the source, for source-map
* debugging * generation
* @option options.source {String} the source to be used for the source- * @option options.source {String} the source to be used for the source-
* mapping, in case the code that's passed in has already been mingled. * mapping, in case the code that's passed in has already been mingled.
* *
* @param {String} code the PaperScript code * @param {String} code the PaperScript code
* @param {Object} [option] the compilation options * @param {Object} [option] the compilation options
* @return {String} the compiled PaperScript translated into JavaScript code * @return {Object} an object holding the compiled PaperScript translated
* into JavaScript code along with source-maps and other information.
*/ */
function compile(code, options) { function compile(code, options) {
if (!code) if (!code)
@ -285,21 +284,23 @@ Base.exports.PaperScript = (function() {
agent = paper.agent, agent = paper.agent,
version = agent.versionNumber, version = agent.versionNumber,
offsetCode = false, offsetCode = false,
sourceMap = null, sourceMaps = options.sourceMaps,
// Include the original code in the sourceMap if there is no linked // Include the original code in the sourceMap if there is no linked
// source file so the debugger can still display it correctly. // source file so the debugger can still display it correctly.
source = options.source || code, source = options.source || code,
lineBreaks = /\r\n|\n|\r/mg, lineBreaks = /\r\n|\n|\r/mg,
offset = 0; offset = options.offset || 0,
map;
// TODO: Verify these browser versions for source map support, and check // TODO: Verify these browser versions for source map support, and check
// other browsers. // other browsers.
if (agent.chrome && version >= 30 if (sourceMaps && (agent.chrome && version >= 30
|| agent.webkit && version >= 537.76 // >= Safari 7.0.4 || agent.webkit && version >= 537.76 // >= Safari 7.0.4
|| agent.firefox && version >= 23 || agent.firefox && version >= 23
|| agent.node) { || agent.node)) {
if (agent.node) { if (agent.node) {
code = 'require("source-map-support").install(paper.PaperScript.sourceMapSupport);\n' + code; // -2 required to remove function header:
offset = -3; // -2 for function body, - 1 for require("source-map-support") // https://code.google.com/p/chromium/issues/detail?id=331655
offset -= 2;
} else if (url && window.location.href.indexOf(url) === 0) { } else if (url && window.location.href.indexOf(url) === 0) {
// If the code stems from the actual html page, determine the // If the code stems from the actual html page, determine the
// offset of inlined code. // offset of inlined code.
@ -309,18 +310,21 @@ Base.exports.PaperScript = (function() {
offset = html.substr(0, html.indexOf(code) + 1).match( offset = html.substr(0, html.indexOf(code) + 1).match(
lineBreaks).length + 1; lineBreaks).length + 1;
} }
// A hack required by most versions of browsers except chrome 36+: // A hack required by older versions of browsers to align inlined
// Instead of starting the mappings at the given offset, we have to // code: Instead of starting the mappings at the given offset, we
// shift the actual code down to the place in the original file, as // have to shift the actual code down to the place in the original
// source-map support seems incomplete in these browsers. // file, as source-map support seems incomplete in these browsers.
// TODO: Report as bugs? offsetCode = offset > 0 && !(
offsetCode = offset > 0 && (!browser.chrome || version < 36); agent.chrome && version >= 36 ||
agent.safari && version >= 600 ||
agent.firefox && version >= 40 ||
agent.node);
var mappings = ['AA' + encodeVLQ(offsetCode ? 0 : offset) + 'A']; var mappings = ['AA' + encodeVLQ(offsetCode ? 0 : offset) + 'A'];
// Create empty entries by the amount of lines + 1, so join can be // Create empty entries by the amount of lines + 1, so join can be
// used below to produce the actual instructions that many times. // used below to produce the actual instructions that many times.
mappings.length = (code.match(lineBreaks) || []).length + 1 mappings.length = (code.match(lineBreaks) || []).length + 1
+ (offsetCode ? offset : 0); + (offsetCode ? offset : 0);
sourceMap = { map = {
version: 3, version: 3,
file: url, file: url,
names:[], names:[],
@ -328,6 +332,7 @@ Base.exports.PaperScript = (function() {
// the lines of the original code, all that is required is a // the lines of the original code, all that is required is a
// mappings string that increments by one between each line. // mappings string that increments by one between each line.
// AACA is the instruction to increment the line by one. // AACA is the instruction to increment the line by one.
// TODO: Add support for column offsets!
mappings: mappings.join(';AACA'), mappings: mappings.join(';AACA'),
sourceRoot: '', sourceRoot: '',
sources: [url], sources: [url],
@ -336,19 +341,25 @@ Base.exports.PaperScript = (function() {
} }
// Now do the parsing magic // Now do the parsing magic
walkAST(parse(code, { ranges: true })); walkAST(parse(code, { ranges: true }));
if (sourceMap) { if (map) {
if (offsetCode) {
// Adjust the line offset of the resulting code if required. // Adjust the line offset of the resulting code if required.
// This is part of a browser hack, see above. // This is part of a browser hack, see above.
if (offsetCode)
code = new Array(offset + 1).join('\n') + code; code = new Array(offset + 1).join('\n') + code;
}
if (/^(inline|both)$/.test(sourceMaps)) {
code += "\n//# sourceMappingURL=data:application/json;base64," code += "\n//# sourceMappingURL=data:application/json;base64,"
+ window.btoa(unescape(encodeURIComponent( + window.btoa(unescape(encodeURIComponent(
JSON.stringify(sourceMap)))) JSON.stringify(map))));
+ "\n//# sourceURL=" + (url || 'paperscript');
if (url)
sourceMaps[url] = sourceMap;
} }
return code; code += "\n//# sourceURL=" + (url || 'paperscript');
}
return {
url: url,
source: source,
code: code,
map: map
};
} }
/** /**
@ -362,14 +373,15 @@ Base.exports.PaperScript = (function() {
* @function * @function
* *
* @option options.url {String} the url of the source, for source-map * @option options.url {String} the url of the source, for source-map
* debugging * generation
* @option options.source {String} the source to be used for the source- * @option options.source {String} the source to be used for the source-
* mapping, in case the code that's passed in has already been mingled. * mapping, in case the code that's passed in has already been mingled.
* *
* @param {String} code the PaperScript code * @param {String} code the PaperScript code
* @param {PaperScope} scope the scope for which the code is executed * @param {PaperScope} scope the scope for which the code is executed
* @param {Object} [option] the compilation options * @param {Object} [option] the compilation options
* @return {String} the compiled PaperScript translated into JavaScript code * @return {Object} an object holding the compiled PaperScript translated
* into JavaScript code along with source-maps and other information.
*/ */
function execute(code, scope, options) { function execute(code, scope, options) {
// Set currently active scope. // Set currently active scope.
@ -393,8 +405,9 @@ Base.exports.PaperScript = (function() {
// function call. // function call.
params = [], params = [],
args = [], args = [],
func; func,
code = compile(code, options); compiled = typeof code === 'object' ? code : compile(code, options);
code = compiled.code;
function expose(scope, hidden) { function expose(scope, hidden) {
// Look through all enumerable properties on the scope and expose // Look through all enumerable properties on the scope and expose
// these too as pseudo-globals, but only if they seem to be in use. // these too as pseudo-globals, but only if they seem to be in use.
@ -428,12 +441,12 @@ Base.exports.PaperScript = (function() {
if (handlers) if (handlers)
code += '\nreturn { ' + handlers + ' };'; code += '\nreturn { ' + handlers + ' };';
var agent = paper.agent; var agent = paper.agent;
if (agent.chrome || agent.firefox) { if (agent.chrome || agent.firefox && agent.versionNumber < 40) {
// On Firefox, all error numbers inside dynamically compiled code // On older Firefox, all error numbers inside dynamically compiled
// are relative to the line where the eval / compilation happened. // code are relative to the line where the eval / compilation
// To fix this issue, we're temporarily inserting a new script // happened. To fix this issue, we're temporarily inserting a new
// tag. We also use this on Chrome to fix an issue with compiled // script tag.
// functions: // We also use this on Chrome to fix issues with compiled functions:
// https://code.google.com/p/chromium/issues/detail?id=331655 // https://code.google.com/p/chromium/issues/detail?id=331655
var script = document.createElement('script'), var script = document.createElement('script'),
head = document.head || document.getElementsByTagName('head')[0]; head = document.head || document.getElementsByTagName('head')[0];
@ -472,6 +485,7 @@ Base.exports.PaperScript = (function() {
// Automatically update view at the end. // Automatically update view at the end.
view.update(); view.update();
} }
return compiled;
} }
function loadScript(script) { function loadScript(script) {
@ -562,13 +576,7 @@ Base.exports.PaperScript = (function() {
compile: compile, compile: compile,
execute: execute, execute: execute,
load: load, load: load,
parse: parse, parse: parse
sourceMapSupport: {
retrieveSourceMap: function(source) {
var map = sourceMaps[source];
return map ? { url: source, map: map } : null;
}
}
}; };
// Pass on `this` as the binding object, so we can reference Acorn both in // Pass on `this` as the binding object, so we can reference Acorn both in
// development and in the built library. // development and in the built library.

View file

@ -74,23 +74,40 @@ DOMParser.prototype.parseFromString = function(string, contenType) {
return div.firstChild; return div.firstChild;
}; };
var sourceMaps = {};
var sourceMapSupport = {
retrieveSourceMap: function(source) {
var map = sourceMaps[source];
return map ? { url: source, map: map } : null;
}
};
// Register the .pjs extension for automatic compilation as PaperScript // Register the .pjs extension for automatic compilation as PaperScript
require.extensions['.pjs'] = function(module, filename) { require.extensions['.pjs'] = function(module, filename) {
// Requiring a PaperScript on Node.js returns an initialize method which // Requiring a PaperScript on Node.js returns an initialize method which
// needs to receive a Canvas object when called and returns the // needs to receive a Canvas object when called and returns the
// PaperScope. // PaperScope.
module.exports = function(canvas) { module.exports = function(canvas) {
var source = fs.readFileSync(filename, 'utf8'); // TODO: Fix this once we can require('paper') from node specific code.
paper.PaperScript.sourceMapSupport = sourceMapSupport;
var source = fs.readFileSync(filename, 'utf8'),
code = 'require("source-map-support").install(paper.PaperScript.sourceMapSupport);\n' + source,
compiled = paper.PaperScript.compile(code, {
url: filename,
source: source,
sourceMaps: true,
offset: -1 // remove require("source-map-support")...
}),
scope = new paper.PaperScope(); scope = new paper.PaperScope();
// Keep track of sourceMaps so retrieveSourceMap() can link them up
scope.setup(canvas); scope.setup(canvas);
scope.__filename = filename; scope.__filename = filename;
scope.__dirname = path.dirname(filename); scope.__dirname = path.dirname(filename);
// Expose core methods and values // Expose core methods and values
scope.require = require; scope.require = require;
scope.console = console; scope.console = console;
paper.PaperScript.execute(source, scope, { sourceMaps[filename] = compiled.map;
url: filename paper.PaperScript.execute(compiled, scope);
});
return scope; return scope;
}; };
}; };