moved to root
This commit is contained in:
@@ -0,0 +1,334 @@
|
||||
/*
|
||||
* Copyright (C) 2017-2019 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 NOEVENT 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.
|
||||
*/
|
||||
|
||||
// LocalResource is a complete resource constructed by the frontend. It
|
||||
// does not need to be tied to an object in the backend. The local resource
|
||||
// does not need to be complete at creation and can get its content later.
|
||||
//
|
||||
// Construction values try to mimic protocol inputs to WI.Resource:
|
||||
//
|
||||
// request: { url, method, headers, timestamp, walltime, finishedTimestamp, data }
|
||||
// response: { mimeType, headers, statusCode, statusText, failureReasonText, content, base64Encoded }
|
||||
// metrics: { responseSource, protocol, priority, remoteAddress, connectionIdentifier, sizes }
|
||||
// timing: { startTime, domainLookupStart, domainLookupEnd, connectStart, connectEnd, secureConnectionStart, requestStart, responseStart, responseEnd }
|
||||
|
||||
WI.LocalResource = class LocalResource extends WI.Resource
|
||||
{
|
||||
constructor({request, response, metrics, timing, mappedFilePath})
|
||||
{
|
||||
console.assert(request);
|
||||
console.assert(typeof request.url === "string");
|
||||
console.assert(response);
|
||||
|
||||
metrics = metrics || {};
|
||||
timing = timing || {};
|
||||
|
||||
super(request.url, {
|
||||
mimeType: response.mimeType || (response.headers || {}).valueForCaseInsensitiveKey("Content-Type") || null,
|
||||
requestMethod: request.method,
|
||||
requestHeaders: request.headers,
|
||||
requestData: request.data,
|
||||
requestSentTimestamp: request.timestamp,
|
||||
requestSentWalltime: request.walltime,
|
||||
});
|
||||
|
||||
// NOTE: This directly overwrites WI.Resource properties.
|
||||
this._finishedOrFailedTimestamp = request.finishedTimestamp || NaN;
|
||||
this._statusCode = response.statusCode || NaN;
|
||||
this._statusText = response.statusText || null;
|
||||
this._responseHeaders = response.headers || {};
|
||||
this._failureReasonText = response.failureReasonText || null;
|
||||
this._timingData = new WI.ResourceTimingData(this, timing);
|
||||
|
||||
this._responseSource = metrics.responseSource || WI.Resource.ResponseSource.Unknown;
|
||||
this._protocol = metrics.protocol || null;
|
||||
this._priority = metrics.priority || WI.Resource.NetworkPriority.Unknown;
|
||||
this._remoteAddress = metrics.remoteAddress || null;
|
||||
this._connectionIdentifier = metrics.connectionIdentifier || null;
|
||||
this._requestHeadersTransferSize = !isNaN(metrics.requestHeaderBytesSent) ? metrics.requestHeaderBytesSent : NaN;
|
||||
this._requestBodyTransferSize = !isNaN(metrics.requestBodyBytesSent) ? metrics.requestBodyBytesSent : NaN;
|
||||
this._responseHeadersTransferSize = !isNaN(metrics.responseHeaderBytesReceived) ? metrics.responseHeaderBytesReceived : NaN;
|
||||
this._responseBodyTransferSize = !isNaN(metrics.responseBodyBytesReceived) ? metrics.responseBodyBytesReceived : NaN;
|
||||
this._responseBodySize = !isNaN(metrics.responseBodyDecodedSize) ? metrics.responseBodyDecodedSize : NaN;
|
||||
this._isProxyConnection = !!metrics.isProxyConnection;
|
||||
|
||||
// Set by `WI.LocalResourceOverride`.
|
||||
this._localResourceOverride = null;
|
||||
|
||||
// Finalize WI.Resource.
|
||||
this._finished = true;
|
||||
this._failed = false; // FIXME: How should we denote a failure? Assume from status code / failure reason?
|
||||
this._cached = false; // FIXME: How should we denote cached? Assume from response source?
|
||||
|
||||
// Finalize WI.SourceCode.
|
||||
let content = response.content || "";
|
||||
let base64Encoded = response.base64Encoded || false;
|
||||
this._originalRevision = new WI.SourceCodeRevision(this, content, base64Encoded, this._mimeType);
|
||||
this._currentRevision = this._originalRevision;
|
||||
|
||||
this._mappedFilePath = mappedFilePath || null;
|
||||
}
|
||||
|
||||
// Static
|
||||
|
||||
static canMapToFile()
|
||||
{
|
||||
return InspectorFrontendHost.canLoad();
|
||||
}
|
||||
|
||||
static headersArrayToHeadersObject(headers)
|
||||
{
|
||||
let result = {};
|
||||
if (headers) {
|
||||
for (let {name, value} of headers)
|
||||
result[name] = value;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
static fromHAREntry(entry, archiveStartWalltime)
|
||||
{
|
||||
// FIXME: <https://webkit.org/b/195694> Web Inspector: HAR Extension for Redirect Timing Info
|
||||
|
||||
let {request, response, startedDateTime, timings} = entry;
|
||||
let requestSentWalltime = WI.HARBuilder.dateFromHARDate(startedDateTime) / 1000;
|
||||
let requestSentTimestamp = requestSentWalltime - archiveStartWalltime;
|
||||
let finishedTimestamp = NaN;
|
||||
|
||||
let timing = {
|
||||
startTime: NaN,
|
||||
domainLookupStart: NaN,
|
||||
domainLookupEnd: NaN,
|
||||
connectStart: NaN,
|
||||
connectEnd: NaN,
|
||||
secureConnectionStart: NaN,
|
||||
requestStart: NaN,
|
||||
responseStart: NaN,
|
||||
responseEnd: NaN,
|
||||
};
|
||||
|
||||
if (!isNaN(requestSentWalltime) && !isNaN(archiveStartWalltime)) {
|
||||
let hasBlocked = timings.blocked !== -1;
|
||||
let hasDNS = timings.dns !== -1;
|
||||
let hasConnect = timings.connect !== -1;
|
||||
let hasSecureConnect = timings.ssl !== -1;
|
||||
|
||||
// FIXME: <https://webkit.org/b/195692> Web Inspector: ResourceTimingData should allow a startTime of 0
|
||||
timing.startTime = requestSentTimestamp || Number.EPSILON;
|
||||
timing.fetchStart = timing.startTime;
|
||||
|
||||
let accumulation = timing.startTime;
|
||||
|
||||
if (hasBlocked)
|
||||
accumulation += (timings.blocked / 1000);
|
||||
|
||||
if (hasDNS) {
|
||||
timing.domainLookupStart = accumulation;
|
||||
accumulation += (timings.dns / 1000);
|
||||
timing.domainLookupEnd = accumulation;
|
||||
}
|
||||
|
||||
if (hasConnect) {
|
||||
timing.connectStart = accumulation;
|
||||
accumulation += (timings.connect / 1000);
|
||||
timing.connectEnd = accumulation;
|
||||
|
||||
if (hasSecureConnect)
|
||||
timing.secureConnectionStart = timing.connectEnd - (timings.ssl / 1000);
|
||||
}
|
||||
|
||||
accumulation += (timings.send / 1000);
|
||||
timing.requestStart = accumulation;
|
||||
|
||||
accumulation += (timings.wait / 1000);
|
||||
timing.responseStart = accumulation;
|
||||
|
||||
accumulation += (timings.receive / 1000);
|
||||
timing.responseEnd = accumulation;
|
||||
|
||||
finishedTimestamp = timing.responseEnd;
|
||||
}
|
||||
|
||||
let serverAddress = entry.serverIPAddress || null;
|
||||
if (serverAddress && typeof entry._serverPort === "number")
|
||||
serverAddress += ":" + entry._serverPort;
|
||||
|
||||
return new WI.LocalResource({
|
||||
request: {
|
||||
url: request.url,
|
||||
method: request.method,
|
||||
headers: LocalResource.headersArrayToHeadersObject(request.headers),
|
||||
timestamp: requestSentTimestamp,
|
||||
walltime: requestSentWalltime,
|
||||
finishedTimestamp: finishedTimestamp,
|
||||
data: request.postData ? request.postData.text : null,
|
||||
},
|
||||
response: {
|
||||
headers: LocalResource.headersArrayToHeadersObject(response.headers),
|
||||
mimeType: response.content.mimeType,
|
||||
statusCode: response.status,
|
||||
statusText: response.statusText,
|
||||
failureReasonText: response._error || null,
|
||||
content: response.content.text,
|
||||
base64Encoded: response.content.encoding === "base64",
|
||||
},
|
||||
metrics: {
|
||||
responseSource: WI.HARBuilder.responseSourceFromHARFetchType(entry._fetchType),
|
||||
protocol: WI.HARBuilder.protocolFromHARProtocol(response.httpVersion),
|
||||
priority: WI.HARBuilder.networkPriorityFromHARPriority(entry._priority),
|
||||
remoteAddress: serverAddress,
|
||||
connectionIdentifier: entry.connection ? parseInt(entry.connection) : null,
|
||||
requestHeaderBytesSent: request.headersSize >= 0 ? request.headersSize : NaN,
|
||||
requestBodyBytesSent: request.bodySize >= 0 ? request.bodySize : NaN,
|
||||
responseHeaderBytesReceived: response.headersSize >= 0 ? response.headersSize : NaN,
|
||||
responseBodyBytesReceived: response.bodySize >= 0 ? response.bodySize : NaN,
|
||||
responseBodyDecodedSize: response.content.size || NaN,
|
||||
},
|
||||
timing,
|
||||
});
|
||||
}
|
||||
|
||||
// Import / Export
|
||||
|
||||
static fromJSON(json)
|
||||
{
|
||||
return new WI.LocalResource(json);
|
||||
}
|
||||
|
||||
toJSON(key)
|
||||
{
|
||||
return {
|
||||
request: {
|
||||
url: this.url,
|
||||
method: this.requestMethod,
|
||||
headers: this.requestHeaders,
|
||||
data: this.requestData,
|
||||
},
|
||||
response: {
|
||||
headers: this.responseHeaders,
|
||||
mimeType: this.mimeType,
|
||||
statusCode: this.statusCode,
|
||||
statusText: this.statusText,
|
||||
content: this.currentRevision.content,
|
||||
base64Encoded: this.currentRevision.base64Encoded,
|
||||
},
|
||||
mappedFilePath: this._mappedFilePath,
|
||||
};
|
||||
}
|
||||
|
||||
// Public
|
||||
|
||||
get localResourceOverride() { return this._localResourceOverride; }
|
||||
|
||||
get mappedFilePath()
|
||||
{
|
||||
return this._mappedFilePath;
|
||||
}
|
||||
|
||||
set mappedFilePath(mappedFilePath)
|
||||
{
|
||||
console.assert(WI.LocalResource.canMapToFile());
|
||||
console.assert(mappedFilePath);
|
||||
|
||||
if (mappedFilePath === this._mappedFilePath)
|
||||
return;
|
||||
|
||||
this._mappedFilePath = mappedFilePath;
|
||||
|
||||
const forceUpdate = true;
|
||||
this._loadFromFileSystem(forceUpdate).then(() => {
|
||||
this.dispatchEventToListeners(WI.LocalResource.Event.MappedFilePathChanged);
|
||||
});
|
||||
}
|
||||
|
||||
// Protected
|
||||
|
||||
async requestContent()
|
||||
{
|
||||
await this._loadFromFileSystem();
|
||||
|
||||
return super.requestContent();
|
||||
}
|
||||
|
||||
requestContentFromBackend()
|
||||
{
|
||||
return Promise.resolve({
|
||||
content: this._originalRevision.content,
|
||||
base64Encoded: this._originalRevision.base64Encoded,
|
||||
});
|
||||
}
|
||||
|
||||
handleCurrentRevisionContentChange()
|
||||
{
|
||||
if (this._mimeType !== this.currentRevision.mimeType) {
|
||||
let oldMIMEType = this._mimeType;
|
||||
this._mimeType = this.currentRevision.mimeType;
|
||||
this.dispatchEventToListeners(WI.Resource.Event.MIMETypeDidChange, {oldMIMEType});
|
||||
}
|
||||
}
|
||||
|
||||
// Private
|
||||
|
||||
async _loadFromFileSystem(forceUpdate)
|
||||
{
|
||||
if (!this._mappedFilePath)
|
||||
return;
|
||||
|
||||
let content = "";
|
||||
try {
|
||||
content = await InspectorFrontendHost.load(this._mappedFilePath);
|
||||
} catch {
|
||||
if (this._didWarnAboutFailureToLoadFromFileSystem)
|
||||
return;
|
||||
|
||||
this._didWarnAboutFailureToLoadFromFileSystem = true;
|
||||
setTimeout(() => {
|
||||
this._didWarnAboutFailureToLoadFromFileSystem = false;
|
||||
});
|
||||
|
||||
let message = WI.UIString("Local Override: could not load \u201C%s\u201D").format(this._mappedFilePath);
|
||||
|
||||
if (window.InspectorTest) {
|
||||
console.warn(message);
|
||||
return;
|
||||
}
|
||||
|
||||
let consoleMessage = new WI.ConsoleMessage(WI.mainTarget, WI.ConsoleMessage.MessageSource.Other, WI.ConsoleMessage.MessageLevel.Warning, message);
|
||||
consoleMessage.shouldRevealConsole = true;
|
||||
|
||||
WI.consoleLogViewController.appendConsoleMessage(consoleMessage);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!forceUpdate && content === this.currentRevision.content)
|
||||
return;
|
||||
|
||||
this.editableRevision.updateRevisionContent(content);
|
||||
}
|
||||
};
|
||||
|
||||
WI.LocalResource.Event = {
|
||||
MappedFilePathChanged: "local-resource-mapped-file-path-changed",
|
||||
};
|
||||
Reference in New Issue
Block a user