deno_std icon indicating copy to clipboard operation
deno_std copied to clipboard

suggestion: mock `performance.now()` in `FakeTime`

Open jespertheend opened this issue 4 months ago • 8 comments

Is your feature request related to a problem? Please describe.

The FakeTime class only seems to mock things such as Date.now().

Describe the solution you'd like

To have it also mock performance.now().

Describe alternatives you've considered

I'm just mocking it myself right now but it's not pretty.

jespertheend avatar Apr 07 '24 17:04 jespertheend

Are you able to elaborate on your use case?

iuioiua avatar Apr 08 '24 03:04 iuioiua

My use case is that I'm using performance.now() in many places instead of Date.now(). I'm not really sure why to be honest, I think it's an old relic from when performance.now() used to have higher precision than Date.now().

I was trying to write tests for code that was using performance.now() and planned on using FakeTime, thinking it would mock it. But that's when I found out it wasn't added yet.

jespertheend avatar Apr 08 '24 16:04 jespertheend

Let's consider your use case. Is its purpose to measure durations between multiple points in time? Does it need the higher precision of performance.now()? If the answer to both of these questions is yes, using performance.now() makes sense. If the answer to both is now, Date.now(), which already has the mocking utility, it makes more sense to use. I just want to ensure that the use case for mocking performance.now() is valid enough to justify having the mocking utility for it.

iuioiua avatar Apr 09 '24 04:04 iuioiua

In this case specifically I'm using performance.now() to compare the deltatime between two "wheel" events in order to determine how long the user has been scrolling. But I have also used performance.now() a lot in the past when computing the deltatime between two frames, or animating an object over time for example.

Aside from the higher precision, performance.now() is also more beneficial in these cases because these systems could get messed up when the system time changes.

jespertheend avatar Apr 11 '24 09:04 jespertheend

~~Ok. That sounds reasonable to me. @kt3k, WDYT?~~ I misunderstood. I agree with Yoshiya's comment below.

iuioiua avatar Apr 11 '24 09:04 iuioiua

perfomance.now() doesn't return unix timestamp, but it returns elapsed time from some random time origin. If FakeTime works with perfomance.now() in a similar way as Deno.now, that feels confusing to me.

Also higher precision time is disabled by default, and its usage is generally discouraged because of the security issues. I think there's no particular reason to encourage the use of performance.now()

kt3k avatar Apr 11 '24 11:04 kt3k

I looked some more into this and it seems like the precision of performance.now() is 100 microseconds nowadays. Which is still one order of magnitude more precise than Date.now().

As for discouraging/encouraging the use of performance.now(), it really depends on what you plan on using it for. MDN puts it best:

Also, Date.now() may have been impacted by system and user clock adjustments, clock skew, etc. as it is relative to the Unix epoch (1970-01-01T00:00:00Z) and dependent on the system clock. The performance.now() method on the other hand is relative to the timeOrigin property which is a monotonic clock: its current time never decreases and isn't subject to adjustments.

FakeTime seems to suggest it affects all time related apis, including setTimeout, setInterval etc. So it was kind of surprising to me that performance.now() remained unaffected.

Although I agree including performance.now() might feel somewhat confusing. With most FakeTime methods I think it should be pretty clear. FakeTime.tick() for instance would move time forward for both Date.now() and performance.now(). The only two instances that seem confusing are getting/setting FakeTime.now and FakeTime.start, since these specifically seem related to system time.

jespertheend avatar Apr 11 '24 18:04 jespertheend

As for discouraging/encouraging the use of performance.now(), it really depends on what you plan on using it for. MDN puts it best:

Also, Date.now() may have been impacted by system and user clock adjustments, clock skew, etc. as it is relative to the Unix epoch (1970-01-01T00:00:00Z) and dependent on the system clock. The performance.now() method on the other hand is relative to the timeOrigin property which is a monotonic clock: its current time never decreases and isn't subject to adjustments.

If the point of performance.now() is being independent of system time, then I wonder if it's good idea to let FakeTime affect performance.now() as FakeTime simulates 'system time'

kt3k avatar Apr 15 '24 10:04 kt3k