add dummy PerformanceTiming

This commit is contained in:
Pierre Tachoire
2026-02-25 08:35:53 +01:00
parent b324be3b0b
commit 235aad32a6
2 changed files with 115 additions and 1 deletions

View File

@@ -274,6 +274,50 @@
} }
</script> </script>
<script id=performance_timing_exists>
{
// performance.timing must not be undefined (used by sites like Bing)
testing.expectEqual(true, performance.timing !== undefined);
testing.expectEqual(true, performance.timing !== null);
}
</script>
<script id=performance_timing_navigationStart>
{
// navigationStart must be a number (sites access performance.timing.navigationStart)
const timing = performance.timing;
testing.expectEqual('number', typeof timing.navigationStart);
testing.expectEqual(0, timing.navigationStart);
}
</script>
<script id=performance_timing_all_properties>
{
// All PerformanceTiming properties must be accessible and return numbers
const timing = performance.timing;
const props = [
'navigationStart', 'unloadEventStart', 'unloadEventEnd',
'redirectStart', 'redirectEnd', 'fetchStart',
'domainLookupStart', 'domainLookupEnd',
'connectStart', 'connectEnd', 'secureConnectionStart',
'requestStart', 'responseStart', 'responseEnd',
'domLoading', 'domInteractive',
'domContentLoadedEventStart', 'domContentLoadedEventEnd',
'domComplete', 'loadEventStart', 'loadEventEnd',
];
for (const prop of props) {
testing.expectEqual('number', typeof timing[prop]);
}
}
</script>
<script id=performance_timing_same_object>
{
// performance.timing should return the same object on each access
testing.expectEqual(true, performance.timing === performance.timing);
}
</script>
<script id=mixed_marks_and_measures> <script id=mixed_marks_and_measures>
{ {
performance.clearMarks(); performance.clearMarks();

View File

@@ -3,7 +3,7 @@ const Page = @import("../Page.zig");
const datetime = @import("../../datetime.zig"); const datetime = @import("../../datetime.zig");
pub fn registerTypes() []const type { pub fn registerTypes() []const type {
return &.{ Performance, Entry, Mark, Measure }; return &.{ Performance, Entry, Mark, Measure, PerformanceTiming };
} }
const std = @import("std"); const std = @import("std");
@@ -12,6 +12,7 @@ const Performance = @This();
_time_origin: u64, _time_origin: u64,
_entries: std.ArrayList(*Entry) = .{}, _entries: std.ArrayList(*Entry) = .{},
_timing: PerformanceTiming = .{},
/// Get high-resolution timestamp in microseconds, rounded to 5μs increments /// Get high-resolution timestamp in microseconds, rounded to 5μs increments
/// to match browser behavior (prevents fingerprinting) /// to match browser behavior (prevents fingerprinting)
@@ -27,9 +28,14 @@ pub fn init() Performance {
return .{ return .{
._time_origin = highResTimestamp(), ._time_origin = highResTimestamp(),
._entries = .{}, ._entries = .{},
._timing = .{},
}; };
} }
pub fn getTiming(self: *Performance) *PerformanceTiming {
return &self._timing;
}
pub fn now(self: *const Performance) f64 { pub fn now(self: *const Performance) f64 {
const current = highResTimestamp(); const current = highResTimestamp();
const elapsed = current - self._time_origin; const elapsed = current - self._time_origin;
@@ -263,6 +269,7 @@ pub const JsApi = struct {
pub const getEntriesByType = bridge.function(Performance.getEntriesByType, .{}); pub const getEntriesByType = bridge.function(Performance.getEntriesByType, .{});
pub const getEntriesByName = bridge.function(Performance.getEntriesByName, .{}); pub const getEntriesByName = bridge.function(Performance.getEntriesByName, .{});
pub const timeOrigin = bridge.accessor(Performance.getTimeOrigin, null, .{}); pub const timeOrigin = bridge.accessor(Performance.getTimeOrigin, null, .{});
pub const timing = bridge.accessor(Performance.getTiming, null, .{});
}; };
pub const Entry = struct { pub const Entry = struct {
@@ -449,6 +456,69 @@ pub const Measure = struct {
}; };
}; };
/// PerformanceTiming — Navigation Timing Level 1 (legacy, but widely used).
/// https://developer.mozilla.org/en-US/docs/Web/API/PerformanceTiming
/// All properties return 0 as stub values; the object must not be undefined
/// so that scripts accessing performance.timing.navigationStart don't crash.
pub const PerformanceTiming = struct {
// Padding to avoid zero-size struct, which causes identity_map pointer collisions.
_pad: bool = false,
pub fn getNavigationStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getUnloadEventStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getUnloadEventEnd(_: *const PerformanceTiming) f64 { return 0; }
pub fn getRedirectStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getRedirectEnd(_: *const PerformanceTiming) f64 { return 0; }
pub fn getFetchStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getDomainLookupStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getDomainLookupEnd(_: *const PerformanceTiming) f64 { return 0; }
pub fn getConnectStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getConnectEnd(_: *const PerformanceTiming) f64 { return 0; }
pub fn getSecureConnectionStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getRequestStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getResponseStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getResponseEnd(_: *const PerformanceTiming) f64 { return 0; }
pub fn getDomLoading(_: *const PerformanceTiming) f64 { return 0; }
pub fn getDomInteractive(_: *const PerformanceTiming) f64 { return 0; }
pub fn getDomContentLoadedEventStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getDomContentLoadedEventEnd(_: *const PerformanceTiming) f64 { return 0; }
pub fn getDomComplete(_: *const PerformanceTiming) f64 { return 0; }
pub fn getLoadEventStart(_: *const PerformanceTiming) f64 { return 0; }
pub fn getLoadEventEnd(_: *const PerformanceTiming) f64 { return 0; }
pub const JsApi = struct {
pub const bridge = js.Bridge(PerformanceTiming);
pub const Meta = struct {
pub const name = "PerformanceTiming";
pub const prototype_chain = bridge.prototypeChain();
pub var class_id: bridge.ClassId = undefined;
};
pub const navigationStart = bridge.accessor(PerformanceTiming.getNavigationStart, null, .{});
pub const unloadEventStart = bridge.accessor(PerformanceTiming.getUnloadEventStart, null, .{});
pub const unloadEventEnd = bridge.accessor(PerformanceTiming.getUnloadEventEnd, null, .{});
pub const redirectStart = bridge.accessor(PerformanceTiming.getRedirectStart, null, .{});
pub const redirectEnd = bridge.accessor(PerformanceTiming.getRedirectEnd, null, .{});
pub const fetchStart = bridge.accessor(PerformanceTiming.getFetchStart, null, .{});
pub const domainLookupStart = bridge.accessor(PerformanceTiming.getDomainLookupStart, null, .{});
pub const domainLookupEnd = bridge.accessor(PerformanceTiming.getDomainLookupEnd, null, .{});
pub const connectStart = bridge.accessor(PerformanceTiming.getConnectStart, null, .{});
pub const connectEnd = bridge.accessor(PerformanceTiming.getConnectEnd, null, .{});
pub const secureConnectionStart = bridge.accessor(PerformanceTiming.getSecureConnectionStart, null, .{});
pub const requestStart = bridge.accessor(PerformanceTiming.getRequestStart, null, .{});
pub const responseStart = bridge.accessor(PerformanceTiming.getResponseStart, null, .{});
pub const responseEnd = bridge.accessor(PerformanceTiming.getResponseEnd, null, .{});
pub const domLoading = bridge.accessor(PerformanceTiming.getDomLoading, null, .{});
pub const domInteractive = bridge.accessor(PerformanceTiming.getDomInteractive, null, .{});
pub const domContentLoadedEventStart = bridge.accessor(PerformanceTiming.getDomContentLoadedEventStart, null, .{});
pub const domContentLoadedEventEnd = bridge.accessor(PerformanceTiming.getDomContentLoadedEventEnd, null, .{});
pub const domComplete = bridge.accessor(PerformanceTiming.getDomComplete, null, .{});
pub const loadEventStart = bridge.accessor(PerformanceTiming.getLoadEventStart, null, .{});
pub const loadEventEnd = bridge.accessor(PerformanceTiming.getLoadEventEnd, null, .{});
};
};
const testing = @import("../../testing.zig"); const testing = @import("../../testing.zig");
test "WebApi: Performance" { test "WebApi: Performance" {
try testing.htmlRunner("performance.html", .{}); try testing.htmlRunner("performance.html", .{});