Implement Web Timing when using NETWORK_SESSION
[WebKit-https.git] / Source / WebCore / platform / network / cocoa / ResourceLoadTiming.mm
1 /*
2  * Copyright (C) 2015 Apple, Inc.  All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #import "config.h"
27 #import "ResourceLoadTiming.h"
28
29 #import <WebCore/NSURLConnectionSPI.h>
30
31 namespace WebCore {
32
33 static double timingValue(NSDictionary *timingData, NSString *key)
34 {
35     if (id object = [timingData objectForKey:key])
36         return [object doubleValue];
37     return 0.0;
38 }
39     
40 void copyTimingData(NSDictionary *timingData, ResourceLoadTiming& timing)
41 {
42     if (!timingData)
43         return;
44     
45     // This is not the navigationStart time in monotonic time, but the other times are relative to this time
46     // and only the differences between times are stored.
47     double referenceStart = timingValue(timingData, @"_kCFNTimingDataFetchStart");
48     
49     double domainLookupStart = timingValue(timingData, @"_kCFNTimingDataDomainLookupStart");
50     double domainLookupEnd = timingValue(timingData, @"_kCFNTimingDataDomainLookupEnd");
51     double connectStart = timingValue(timingData, @"_kCFNTimingDataConnectStart");
52     double secureConnectionStart = timingValue(timingData, @"_kCFNTimingDataSecureConnectionStart");
53     double connectEnd = timingValue(timingData, @"_kCFNTimingDataConnectEnd");
54     double requestStart = timingValue(timingData, @"_kCFNTimingDataRequestStart");
55     double responseStart = timingValue(timingData, @"_kCFNTimingDataResponseStart");
56     
57     timing.domainLookupStart = domainLookupStart <= 0 ? -1 : (domainLookupStart - referenceStart) * 1000;
58     timing.domainLookupEnd = domainLookupEnd <= 0 ? -1 : (domainLookupEnd - referenceStart) * 1000;
59     timing.connectStart = connectStart <= 0 ? -1 : (connectStart - referenceStart) * 1000;
60     timing.secureConnectionStart = secureConnectionStart <= 0 ? -1 : (secureConnectionStart - referenceStart) * 1000;
61     timing.connectEnd = connectEnd <= 0 ? -1 : (connectEnd - referenceStart) * 1000;
62     timing.requestStart = requestStart <= 0 ? 0 : (requestStart - referenceStart) * 1000;
63     timing.responseStart = responseStart <= 0 ? 0 : (responseStart - referenceStart) * 1000;
64 }
65
66 #if !HAVE(TIMINGDATAOPTIONS)
67 void setCollectsTimingData()
68 {
69     static dispatch_once_t onceToken;
70     dispatch_once(&onceToken, ^{
71         [NSURLConnection _setCollectsTimingData:YES];
72         [NSURLConnection _collectTimingDataWithOptions:TimingDataCollectionNStatsOff | TimingDataCollectionConnectionDataOff];
73     });
74 }
75 #endif
76     
77 }