blob: 4a76e3d50cec88583ef1886decb6a2a42b60ac1b [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/renderer/loadtimes_extension_bindings.h"
#include "base/time/time.h"
#include "extensions/renderer/v8_helpers.h"
#include "net/http/http_connection_info.h"
#include "third_party/blink/public/platform/web_url_response.h"
#include "third_party/blink/public/web/web_local_frame.h"
#include "third_party/blink/public/web/web_navigation_type.h"
#include "third_party/blink/public/web/web_performance_metrics_for_reporting.h"
#include "v8/include/v8-extension.h"
#include "v8/include/v8-isolate.h"
#include "v8/include/v8-object.h"
#include "v8/include/v8-primitive.h"
#include "v8/include/v8-template.h"
using blink::WebDocumentLoader;
using blink::WebLocalFrame;
using blink::WebNavigationType;
// Values for CSI "tran" property
const int kTransitionLink = 0;
const int kTransitionForwardBack = 6;
const int kTransitionOther = 15;
const int kTransitionReload = 16;
namespace extensions_v8 {
static const char* const kLoadTimesExtensionName = "v8/LoadTimes";
class LoadTimesExtensionWrapper : public v8::Extension {
public:
// Creates an extension which adds a new function, chrome.loadTimes()
// This function returns an object containing the following members:
// requestTime: The time the request to load the page was received
// loadTime: The time the renderer started the load process
// finishDocumentLoadTime: The time the document itself was loaded
// (this is before the onload() method is fired)
// finishLoadTime: The time all loading is done, after the onload()
// method and all resources
// navigationType: A string describing what user action initiated the load
//
// Note that chrome.loadTimes() is deprecated in favor of performance.timing.
// Many of the timings reported via chrome.loadTimes() match timings available
// in performance.timing. Timing data will be removed from chrome.loadTimes()
// in a future release. No new timings or other information should be exposed
// via these APIs.
LoadTimesExtensionWrapper() :
v8::Extension(kLoadTimesExtensionName,
"var chrome;"
"if (!chrome)"
" chrome = {};"
"chrome.loadTimes = function() {"
" native function GetLoadTimes();"
" return GetLoadTimes();"
"};"
"chrome.csi = function() {"
" native function GetCSI();"
" return GetCSI();"
"}") {}
v8::Local<v8::FunctionTemplate> GetNativeFunctionTemplate(
v8::Isolate* isolate,
v8::Local<v8::String> name) override {
if (name->StringEquals(
v8::String::NewFromUtf8(isolate, "GetLoadTimes",
v8::NewStringType::kInternalized)
.ToLocalChecked())) {
return v8::FunctionTemplate::New(isolate, GetLoadTimes);
} else if (name->StringEquals(
v8::String::NewFromUtf8(isolate, "GetCSI",
v8::NewStringType::kInternalized)
.ToLocalChecked())) {
return v8::FunctionTemplate::New(isolate, GetCSI);
}
return v8::Local<v8::FunctionTemplate>();
}
static const char* GetNavigationType(WebNavigationType nav_type) {
switch (nav_type) {
case blink::kWebNavigationTypeLinkClicked:
return "LinkClicked";
case blink::kWebNavigationTypeFormSubmitted:
return "FormSubmitted";
case blink::kWebNavigationTypeBackForward:
case blink::kWebNavigationTypeRestore:
return "BackForward";
case blink::kWebNavigationTypeReload:
return "Reload";
case blink::kWebNavigationTypeFormResubmittedBackForward:
case blink::kWebNavigationTypeFormResubmittedReload:
return "Resubmitted";
case blink::kWebNavigationTypeOther:
return "Other";
}
return "";
}
static int GetCSITransitionType(WebNavigationType nav_type) {
switch (nav_type) {
case blink::kWebNavigationTypeLinkClicked:
case blink::kWebNavigationTypeFormSubmitted:
case blink::kWebNavigationTypeFormResubmittedBackForward:
case blink::kWebNavigationTypeFormResubmittedReload:
return kTransitionLink;
case blink::kWebNavigationTypeBackForward:
case blink::kWebNavigationTypeRestore:
return kTransitionForwardBack;
case blink::kWebNavigationTypeReload:
return kTransitionReload;
case blink::kWebNavigationTypeOther:
return kTransitionOther;
}
return kTransitionOther;
}
static void LoadtimesGetter(
v8::Local<v8::Name> name,
const v8::PropertyCallbackInfo<v8::Value>& info) {
if (WebLocalFrame* frame = WebLocalFrame::FrameForCurrentContext()) {
frame->UsageCountChromeLoadTimes(blink::WebString::FromUTF8(
*v8::String::Utf8Value(info.GetIsolate(), name)));
}
info.GetReturnValue().Set(info.Data());
}
static void GetLoadTimes(const v8::FunctionCallbackInfo<v8::Value>& args) {
args.GetReturnValue().SetNull();
WebLocalFrame* frame = WebLocalFrame::FrameForCurrentContext();
if (!frame) {
return;
}
WebDocumentLoader* document_loader = frame->GetDocumentLoader();
if (!document_loader) {
return;
}
const blink::WebURLResponse& response = document_loader->GetWebResponse();
blink::WebPerformanceMetricsForReporting web_performance =
frame->PerformanceMetricsForReporting();
// Though request time now tends to be used to describe the time that the
// request for the main resource was issued, when chrome.loadTimes() was
// added, it was used to describe 'The time the request to load the page was
// received', which is the time now known as navigation start. For backward
// compatibility, we continue to provide request_time, setting its value to
// navigation start.
double request_time = web_performance.NavigationStart();
// Developers often use start_load_time as the time the navigation was
// started, so we return navigationStart for this value as well. See
// https://gist.github.com/search?utf8=%E2%9C%93&q=startLoadTime.
// Note that, historically, start_load_time reported the time that a
// provisional load was first processed in the render process. For
// browser-initiated navigations, this is some time after navigation start,
// which means that developers who used this value as a way to track the
// start of a navigation were misusing this timestamp and getting the wrong
// value - they should be using navigationStart instead. Provisional loads
// will not be processed by the render process for browser-initiated
// navigations, so reporting the time a provisional load was processed in
// the render process will no longer make sense. Thus, we now report the
// time for navigationStart, which is a value more consistent with what
// developers currently use start_load_time for.
double start_load_time = web_performance.NavigationStart();
// TODO(bmcquade): Remove this. 'commit' time is a concept internal to
// chrome that shouldn't be exposed to the web platform.
double commit_load_time = web_performance.ResponseStart();
double finish_document_load_time =
web_performance.DomContentLoadedEventEnd();
double finish_load_time = web_performance.LoadEventEnd();
double first_paint_time = web_performance.FirstPaint();
// TODO(bmcquade): remove this. It's misleading to track the first paint
// after the load event, since many pages perform their meaningful paints
// long before the load event fires. We report a time of zero for the
// time being.
double first_paint_after_load_time = 0.0;
std::string navigation_type =
GetNavigationType(document_loader->GetNavigationType());
bool was_fetched_via_spdy = response.WasFetchedViaSPDY();
bool was_alpn_negotiated = response.WasAlpnNegotiated();
std::string alpn_negotiated_protocol =
response.AlpnNegotiatedProtocol().Utf8();
bool was_alternate_protocol_available =
response.WasAlternateProtocolAvailable();
std::string_view connection_info =
net::HttpConnectionInfoToString(response.ConnectionInfo());
// Important: |frame| and |document_loader| should not be
// referred to below this line, as JS setters below can invalidate these
// pointers.
v8::Isolate* isolate = args.GetIsolate();
v8::Local<v8::Context> ctx = isolate->GetCurrentContext();
v8::Local<v8::Object> load_times = v8::Object::New(isolate);
if (!load_times
->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "requestTime", v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Number::New(isolate, request_time))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "startLoadTime",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Number::New(isolate, start_load_time))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "commitLoadTime",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Number::New(isolate, commit_load_time))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "finishDocumentLoadTime",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Number::New(isolate, finish_document_load_time))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "finishLoadTime",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Number::New(isolate, finish_load_time))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "firstPaintTime",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Number::New(isolate, first_paint_time))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "firstPaintAfterLoadTime",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Number::New(isolate, first_paint_after_load_time))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "navigationType",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::String::NewFromUtf8(isolate, navigation_type.c_str())
.ToLocalChecked())
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "wasFetchedViaSpdy",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Boolean::New(isolate, was_fetched_via_spdy))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "wasNpnNegotiated",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Boolean::New(isolate, was_alpn_negotiated))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(ctx,
v8::String::NewFromUtf8Literal(
isolate, "npnNegotiatedProtocol",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::String::NewFromUtf8(
isolate, alpn_negotiated_protocol.c_str())
.ToLocalChecked())
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "wasAlternateProtocolAvailable",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::Boolean::New(isolate, was_alternate_protocol_available))
.FromMaybe(false)) {
return;
}
if (!load_times
->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "connectionInfo",
v8::NewStringType::kInternalized),
LoadtimesGetter, nullptr,
v8::String::NewFromUtf8(isolate, connection_info.data(),
v8::NewStringType::kNormal,
connection_info.length())
.ToLocalChecked())
.FromMaybe(false)) {
return;
}
args.GetReturnValue().Set(load_times);
}
static void CSIGetter(v8::Local<v8::Name> name,
const v8::PropertyCallbackInfo<v8::Value>& info) {
if (WebLocalFrame* frame = WebLocalFrame::FrameForCurrentContext()) {
frame->UsageCountChromeCSI(blink::WebString::FromUTF8(
*v8::String::Utf8Value(info.GetIsolate(), name)));
}
info.GetReturnValue().Set(info.Data());
}
static void GetCSI(const v8::FunctionCallbackInfo<v8::Value>& args) {
args.GetReturnValue().SetNull();
WebLocalFrame* frame = WebLocalFrame::FrameForCurrentContext();
if (!frame) {
return;
}
WebDocumentLoader* document_loader = frame->GetDocumentLoader();
if (!document_loader) {
return;
}
blink::WebPerformanceMetricsForReporting web_performance =
frame->PerformanceMetricsForReporting();
base::Time now = base::Time::Now();
base::Time start = base::Time::FromSecondsSinceUnixEpoch(
web_performance.NavigationStart());
base::Time dom_content_loaded_end = base::Time::FromSecondsSinceUnixEpoch(
web_performance.DomContentLoadedEventEnd());
base::TimeDelta page = now - start;
int navigation_type =
GetCSITransitionType(document_loader->GetNavigationType());
// Important: |frame| and |document_loader| should not be referred to below
// this line, as JS setters below can invalidate these pointers.
v8::Isolate* isolate = args.GetIsolate();
v8::Local<v8::Context> ctx = isolate->GetCurrentContext();
v8::Local<v8::Object> csi = v8::Object::New(isolate);
if (!csi->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "startE", v8::NewStringType::kInternalized),
CSIGetter, nullptr,
v8::Number::New(isolate, start.InMillisecondsSinceUnixEpoch()))
.FromMaybe(false)) {
return;
}
// NOTE: historically, the CSI onload field has reported the time the
// document finishes parsing, which is DOMContentLoaded. Thus, we continue
// to report that here, despite the fact that the field is named onloadT.
if (!csi->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "onloadT", v8::NewStringType::kInternalized),
CSIGetter, nullptr,
v8::Number::New(
isolate,
dom_content_loaded_end.InMillisecondsSinceUnixEpoch()))
.FromMaybe(false)) {
return;
}
if (!csi->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "pageT", v8::NewStringType::kInternalized),
CSIGetter, nullptr,
v8::Number::New(isolate, page.InMillisecondsF()))
.FromMaybe(false)) {
return;
}
if (!csi->SetAccessor(
ctx,
v8::String::NewFromUtf8Literal(
isolate, "tran", v8::NewStringType::kInternalized),
CSIGetter, nullptr, v8::Number::New(isolate, navigation_type))
.FromMaybe(false)) {
return;
}
args.GetReturnValue().Set(csi);
}
};
std::unique_ptr<v8::Extension> LoadTimesExtension::Get() {
return std::make_unique<LoadTimesExtensionWrapper>();
}
} // namespace extensions_v8