使用libfaketime
更改进程的时间速度时,setTimout
设置的超时根据在Linux下运行时更改的时间到期,但根据在Mac下运行时的原始系统时间OS。
在Mac OS中:
DYLD_INSERT_LIBRARIES=src/libfaketime.1.dylib DYLD_FORCE_FLAT_NAMESPACE=y FAKETIME="@2020-12-24 00:00:00 x3600" node
> setTimeout(() => {console.log('hello');}, 3600 * 1000); // Takes an hour
在Linux中:
LD_PRELOAD=src/libfaketime.1.so FAKETIME="@2020-12-24 00:00:00 x3600" node
> setTimeout(() => {console.log('hello');}, 3600 * 1000); // Takes a second
在调查此问题时,我注意到libc
clock_gettime
函数在Linux下由node.js(libuv?)轮询,但在Mac OS下运行时不会调用此函数。 (我在printf
函数中添加了一些libfaketime
)
node.js(libuv?)的实现有什么不同,导致Mac OS和Linux之间的行为差异?为什么存在这种差异?
我做的另一个观察是,当使用libfaketime
冻结时间时,setImmediate
和setTimeout(cb, 0)
的行为在Linux下的不同之处在于使用setImmediate
时运行回调但在使用setTimeout(cb, 0)
时没有。
答案 0 :(得分:4)
这绝对是libuv的一个不同之处。达尔文没有support CLOCK_MONOTONIC*
因此mach_absolute_time()
must be called以获得当前时间。最终绕过libfaketime
,导致客户端代码在OS X上实时运行。