繁体   English   中英

如何同时多次模拟 Date.now

[英]How can I mock Date.now multiple times concurrently

我正在尝试实现我自己的模拟日期函数,我希望它能够同时工作。

我认为最佳用法是:

mockDate(myFirstDate, async () => {
   // Here and only here Date.now() === myFirstDate
})

mockDate(mySecondDate, async () => {
   // Here and only here Date.now() === mySecondDate
})

// Outside Date.now is working as expected

这是我试图实现的最小失败代码示例:

async function mockDate(timestamp, callback) {
    const original_now = Date.now

    Date.now = function() {
        if (/* How can I check Date.now have been called inside callback */ true) return timestamp
        else return original_now()
    }

    await callback()

    Date.now = original_now
}

const wait = ms => new Promise(res => setTimeout(res, ms))

mockDate(1600000000000, async () => {
    await wait(100)
    const result = Date.now()
    if (result !== 1600000000000) throw new Error('Fail: 1600000000000')
}).catch(console.error)

mockDate(1650000000000, async () => {
    const result = Date.now()
    if (result !== 1650000000000) throw new Error('Fail: 1650000000000')
    await wait(200)
}).catch(console.error)

我的想法是检查if内部是否可以检索this.caller callback 不幸的是,访问this.caller给了我一个错误。

我也知道测试框架通常有串行模式,但我试图避免它。

禁用严格模式或使用串行模式。

这是一个有效的解决方案:

async function mockDate(timestamp, callback) {
    const original_now = Date.now

    Date.now = function() {
        try {
            let caller = arguments.callee.caller
            while (caller !== callback && caller.caller) {
                caller = caller.caller
            }
            return caller === callback
                ? timestamp
                : original_now()
        } catch {
            return original_now()
        }
    }

    await callback()

    Date.now = original_now
}

这个解决方案的一个主要缺点是你不能使用箭头函数

mockDate(date, async () => {
    const date1 = (function() { return Date.now() })() // Date is mocked

    const date2 = (() => Date.now())() // Date wont be mocked
})

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM