blob: ee0469d84e238ee4e92af9572abbeaacb913b0d2 [file] [log] [blame] [edit]
/*
* Copyright (C) 2013 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
WebInspector.SourceMapResource = function(url, sourceMap)
{
WebInspector.Resource.call(this, url, null);
console.assert(url);
console.assert(sourceMap);
this._sourceMap = sourceMap;
};
WebInspector.SourceMapResource.prototype = {
constructor: WebInspector.SourceMapResource,
// Public
get sourceMap()
{
return this._sourceMap;
},
get sourceMapDisplaySubpath()
{
var sourceMappingBasePathURLComponents = this._sourceMap.sourceMappingBasePathURLComponents;
var resourceURLComponents = this.urlComponents;
// Different schemes / hosts. Return the host + path of this resource.
if (resourceURLComponents.scheme !== sourceMappingBasePathURLComponents.scheme || resourceURLComponents.host !== sourceMappingBasePathURLComponents.host)
return resourceURLComponents.host + (resourceURLComponents.port ? (":" + resourceURLComponents.port) : "") + resourceURLComponents.path;
// Same host, but not a subpath of the base. This implies a ".." in the relative path.
if (!resourceURLComponents.path.startsWith(sourceMappingBasePathURLComponents.path))
return relativePath(resourceURLComponents.path, sourceMappingBasePathURLComponents.path);
// Same host. Just a subpath of the base.
return resourceURLComponents.path.substring(sourceMappingBasePathURLComponents.path.length, resourceURLComponents.length);
},
canRequestContentFromBackend: function()
{
// COMPATIBILITY (iOS 6 & 7): NetworkAgent.loadResource does not exist.
// We shouldn't get here, but just to be safe.
return !this.finished || !NetworkAgent.loadResource;
},
requestContentFromBackend: function(callback)
{
// COMPATIBILITY (iOS 6 & 7): NetworkAgent.loadResource does not exist.
// We shouldn't get here, but just to be safe.
console.assert(NetworkAgent.loadResource);
if (!NetworkAgent.loadResource)
return false;
var inlineContent = this._sourceMap.sourceContent(this.url);
if (inlineContent) {
// Force inline content to be asynchronous to match the expected load pattern.
setTimeout(function() {
// FIXME: We don't know the MIME-type for inline content. Guess by analyzing the content?
// Guess by using the type of the original resource?
sourceMapResourceLoaded.call(this, null, inlineContent, "text/javascript");
}.bind(this));
return true;
}
function sourceMapResourceLoaded(error, body, mimeType, statusCode)
{
const base64encoded = false;
if (error || statusCode >= 400) {
this.markAsFailed();
callback(error, body, base64encoded);
return;
}
var fileExtension = WebInspector.fileExtensionForURL(this.url);
var fileExtensionMIMEType = WebInspector.mimeTypeForFileExtension(fileExtension, true) || "text/javascript";
// FIXME: Add support for picking the best MIME-type. Right now the file extension is the best bet.
mimeType = fileExtensionMIMEType;
var oldType = this._type;
var oldMIMEType = this._mimeType;
// FIXME: This is a layering violation. It should use a helper function on the Resource base-class.
this._mimeType = mimeType;
this._type = WebInspector.Resource.Type.fromMIMEType(this._mimeType);
if (oldMIMEType !== mimeType)
this.dispatchEventToListeners(WebInspector.Resource.Event.MIMETypeDidChange, {oldMIMEType: oldMIMEType});
if (oldType !== this._type)
this.dispatchEventToListeners(WebInspector.Resource.Event.TypeDidChange, {oldType: oldType});
this.markAsFinished();
callback(null, body, base64encoded);
}
var frameIdentifier = null;
if (this._sourceMap.originalSourceCode instanceof WebInspector.Resource && this._sourceMap.originalSourceCode.parentFrame)
frameIdentifier = this._sourceMap.originalSourceCode.parentFrame.id;
if (!frameIdentifier)
frameIdentifier = WebInspector.frameResourceManager.mainFrame.id;
NetworkAgent.loadResource(frameIdentifier, this.url, sourceMapResourceLoaded.bind(this));
return true;
},
createSourceCodeLocation: function(lineNumber, columnNumber)
{
// SourceCodeLocations are always constructed with raw resources and raw locations. Lookup the raw location.
var entry = this._sourceMap.findEntryReversed(this.url, lineNumber);
var rawLineNumber = entry[0];
var rawColumnNumber = entry[1];
// If the raw location is an inline script we need to include that offset.
var originalSourceCode = this._sourceMap.originalSourceCode;
if (originalSourceCode instanceof WebInspector.Script) {
if (rawLineNumber === 0)
rawColumnNumber += originalSourceCode.range.startColumn;
rawLineNumber += originalSourceCode.range.startLine;
}
// Create the SourceCodeLocation and since we already know the the mapped location set it directly.
var location = originalSourceCode.createSourceCodeLocation(rawLineNumber, rawColumnNumber);
location._setMappedLocation(this, lineNumber, columnNumber);
return location;
},
createSourceCodeTextRange: function(textRange)
{
// SourceCodeTextRanges are always constructed with raw resources and raw locations.
// However, we can provide the most accurate mapped locations in construction.
var startSourceCodeLocation = this.createSourceCodeLocation(textRange.startLine, textRange.startColumn);
var endSourceCodeLocation = this.createSourceCodeLocation(textRange.endLine, textRange.endColumn);
return new WebInspector.SourceCodeTextRange(this._sourceMap.originalSourceCode, startSourceCodeLocation, endSourceCodeLocation);
}
};
WebInspector.SourceMapResource.prototype.__proto__ = WebInspector.Resource.prototype;