2018-09-12 22:13:22 +00:00
|
|
|
'use strict'
|
|
|
|
|
2018-06-19 15:22:08 +00:00
|
|
|
const chai = require('chai')
|
|
|
|
const dirtyChai = require('dirty-chai')
|
2018-09-13 16:10:51 +00:00
|
|
|
const { deprecations, deprecate } = require('electron')
|
2016-02-16 23:09:35 +00:00
|
|
|
|
2018-09-13 16:10:51 +00:00
|
|
|
const { expect } = chai
|
2018-06-19 15:22:08 +00:00
|
|
|
chai.use(dirtyChai)
|
|
|
|
|
2018-09-12 06:00:41 +00:00
|
|
|
describe('deprecations', () => {
|
2017-10-27 00:35:33 +00:00
|
|
|
beforeEach(() => {
|
2016-03-25 20:03:49 +00:00
|
|
|
deprecations.setHandler(null)
|
|
|
|
process.throwDeprecation = true
|
|
|
|
})
|
2016-02-16 23:09:35 +00:00
|
|
|
|
2017-11-14 19:56:16 +00:00
|
|
|
it('allows a deprecation handler function to be specified', () => {
|
|
|
|
const messages = []
|
2017-11-14 01:21:57 +00:00
|
|
|
|
2018-06-19 15:22:08 +00:00
|
|
|
deprecations.setHandler(message => {
|
2017-11-14 19:56:16 +00:00
|
|
|
messages.push(message)
|
|
|
|
})
|
|
|
|
|
|
|
|
deprecate.log('this is deprecated')
|
2018-06-19 15:22:08 +00:00
|
|
|
expect(messages).to.deep.equal(['this is deprecated'])
|
2017-11-14 19:56:16 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
it('returns a deprecation handler after one is set', () => {
|
|
|
|
const messages = []
|
|
|
|
|
2018-06-19 15:22:08 +00:00
|
|
|
deprecations.setHandler(message => {
|
2017-11-14 19:56:16 +00:00
|
|
|
messages.push(message)
|
|
|
|
})
|
|
|
|
|
|
|
|
deprecate.log('this is deprecated')
|
2018-06-19 15:22:08 +00:00
|
|
|
expect(deprecations.getHandler()).to.be.a('function')
|
2017-11-14 19:56:16 +00:00
|
|
|
})
|
|
|
|
|
2018-05-29 13:40:48 +00:00
|
|
|
it('renames a property', () => {
|
|
|
|
let msg
|
2018-09-11 18:16:49 +00:00
|
|
|
deprecations.setHandler(m => { msg = m })
|
2018-05-29 13:40:48 +00:00
|
|
|
|
2018-09-11 18:16:49 +00:00
|
|
|
const oldProp = 'dingyOldName'
|
|
|
|
const newProp = 'shinyNewName'
|
2018-05-29 13:40:48 +00:00
|
|
|
|
|
|
|
let value = 0
|
2018-09-13 16:10:51 +00:00
|
|
|
const o = { [newProp]: value }
|
2018-09-11 18:16:49 +00:00
|
|
|
expect(o).to.not.have.a.property(oldProp)
|
|
|
|
expect(o).to.have.a.property(newProp).that.is.a('number')
|
2018-05-29 13:40:48 +00:00
|
|
|
|
2018-09-11 18:16:49 +00:00
|
|
|
deprecate.renameProperty(o, oldProp, newProp)
|
|
|
|
o[oldProp] = ++value
|
2018-05-29 13:40:48 +00:00
|
|
|
|
2018-06-19 15:22:08 +00:00
|
|
|
expect(msg).to.be.a('string')
|
2018-09-11 18:16:49 +00:00
|
|
|
expect(msg).to.include(oldProp)
|
|
|
|
expect(msg).to.include(newProp)
|
2018-05-29 13:40:48 +00:00
|
|
|
|
2018-09-11 18:16:49 +00:00
|
|
|
expect(o).to.have.a.property(newProp).that.is.equal(value)
|
|
|
|
expect(o).to.have.a.property(oldProp).that.is.equal(value)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('doesn\'t deprecate a property not on an object', () => {
|
|
|
|
const o = {}
|
|
|
|
|
|
|
|
expect(() => {
|
2018-09-12 22:13:22 +00:00
|
|
|
deprecate.removeProperty(o, 'iDoNotExist')
|
|
|
|
}).to.throw(/iDoNotExist/)
|
2018-05-29 13:40:48 +00:00
|
|
|
})
|
|
|
|
|
2018-06-27 06:47:01 +00:00
|
|
|
it('deprecates a property of an object', () => {
|
|
|
|
let msg
|
|
|
|
deprecations.setHandler(m => { msg = m })
|
|
|
|
|
2018-09-11 18:16:49 +00:00
|
|
|
const prop = 'itMustGo'
|
2018-10-02 01:56:31 +00:00
|
|
|
const o = { [prop]: 0 }
|
2018-09-11 18:16:49 +00:00
|
|
|
|
|
|
|
deprecate.removeProperty(o, prop)
|
2018-06-27 06:47:01 +00:00
|
|
|
|
2018-09-11 18:16:49 +00:00
|
|
|
const temp = o[prop]
|
2018-06-27 06:47:01 +00:00
|
|
|
|
2018-09-11 18:16:49 +00:00
|
|
|
expect(temp).to.equal(0)
|
2018-06-27 06:47:01 +00:00
|
|
|
expect(msg).to.be.a('string')
|
2018-09-11 18:16:49 +00:00
|
|
|
expect(msg).to.include(prop)
|
2018-06-27 06:47:01 +00:00
|
|
|
})
|
|
|
|
|
2019-02-08 17:50:11 +00:00
|
|
|
it('warns exactly once when a function is deprecated with no replacement', () => {
|
|
|
|
let msg
|
|
|
|
deprecations.setHandler(m => { msg = m })
|
|
|
|
|
|
|
|
function oldFn () { return 'hello' }
|
|
|
|
const deprecatedFn = deprecate.function(oldFn)
|
|
|
|
deprecatedFn()
|
|
|
|
|
|
|
|
expect(msg).to.be.a('string')
|
|
|
|
expect(msg).to.include('oldFn')
|
|
|
|
})
|
|
|
|
|
|
|
|
it('warns exactly once when a function is deprecated with a replacement', () => {
|
|
|
|
let msg
|
|
|
|
deprecations.setHandler(m => { msg = m })
|
|
|
|
|
|
|
|
function oldFn () { return 'hello' }
|
|
|
|
function newFn () { return 'goodbye' }
|
|
|
|
const deprecatedFn = deprecate.function(oldFn, newFn)
|
|
|
|
deprecatedFn()
|
|
|
|
|
|
|
|
expect(msg).to.be.a('string')
|
|
|
|
expect(msg).to.include('oldFn')
|
|
|
|
expect(msg).to.include('newFn')
|
|
|
|
})
|
|
|
|
|
2018-09-12 22:13:22 +00:00
|
|
|
it('warns only once per item', () => {
|
|
|
|
const messages = []
|
2019-02-08 17:50:11 +00:00
|
|
|
deprecations.setHandler(message => messages.push(message))
|
2018-09-12 22:13:22 +00:00
|
|
|
|
|
|
|
const key = 'foo'
|
|
|
|
const val = 'bar'
|
2018-10-02 01:56:31 +00:00
|
|
|
const o = { [key]: val }
|
2018-09-12 22:13:22 +00:00
|
|
|
deprecate.removeProperty(o, key)
|
|
|
|
|
|
|
|
for (let i = 0; i < 3; ++i) {
|
|
|
|
expect(o[key]).to.equal(val)
|
|
|
|
expect(messages).to.have.length(1)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2018-05-29 13:58:02 +00:00
|
|
|
it('warns if deprecated property is already set', () => {
|
|
|
|
let msg
|
2018-09-11 18:16:49 +00:00
|
|
|
deprecations.setHandler(m => { msg = m })
|
2018-05-29 13:58:02 +00:00
|
|
|
|
2018-09-11 18:16:49 +00:00
|
|
|
const oldProp = 'dingyOldName'
|
|
|
|
const newProp = 'shinyNewName'
|
2018-05-29 13:58:02 +00:00
|
|
|
|
2018-10-02 01:56:31 +00:00
|
|
|
const o = { [oldProp]: 0 }
|
2018-09-11 18:16:49 +00:00
|
|
|
deprecate.renameProperty(o, oldProp, newProp)
|
2018-05-29 13:58:02 +00:00
|
|
|
|
2018-06-19 15:22:08 +00:00
|
|
|
expect(msg).to.be.a('string')
|
2018-09-11 18:16:49 +00:00
|
|
|
expect(msg).to.include(oldProp)
|
|
|
|
expect(msg).to.include(newProp)
|
2018-05-29 13:58:02 +00:00
|
|
|
})
|
|
|
|
|
2017-11-14 19:56:16 +00:00
|
|
|
it('throws an exception if no deprecation handler is specified', () => {
|
2018-06-19 15:22:08 +00:00
|
|
|
expect(() => {
|
2017-11-14 19:56:16 +00:00
|
|
|
deprecate.log('this is deprecated')
|
2018-06-19 15:22:08 +00:00
|
|
|
}).to.throw(/this is deprecated/)
|
2017-11-14 19:56:16 +00:00
|
|
|
})
|
2018-12-10 16:13:09 +00:00
|
|
|
|
2019-03-14 22:19:19 +00:00
|
|
|
it('warns when a function is deprecated in favor of a property', () => {
|
|
|
|
const warnings = []
|
|
|
|
deprecations.setHandler(warning => warnings.push(warning))
|
|
|
|
|
|
|
|
function oldGetterFn () { return 'getter' }
|
|
|
|
function oldSetterFn () { return 'setter' }
|
|
|
|
|
|
|
|
const newProp = 'myRadProp'
|
|
|
|
|
|
|
|
const [
|
|
|
|
deprecatedGetter,
|
|
|
|
deprecatedSetter
|
|
|
|
] = deprecate.fnToProperty(newProp, oldGetterFn, oldSetterFn)
|
|
|
|
|
|
|
|
deprecatedGetter()
|
|
|
|
deprecatedSetter()
|
|
|
|
|
|
|
|
expect(warnings).to.have.lengthOf(2)
|
|
|
|
|
|
|
|
expect(warnings[0]).to.include('oldGetterFn')
|
|
|
|
expect(warnings[0]).to.include(newProp)
|
|
|
|
|
|
|
|
expect(warnings[1]).to.include('oldSetterFn')
|
|
|
|
expect(warnings[1]).to.include(newProp)
|
|
|
|
})
|
|
|
|
|
2018-12-10 16:13:09 +00:00
|
|
|
describe('promisify', () => {
|
|
|
|
const expected = 'Hello, world!'
|
|
|
|
let promiseFunc
|
|
|
|
let warnings
|
|
|
|
|
|
|
|
const enableCallbackWarnings = () => {
|
|
|
|
warnings = []
|
2019-02-08 17:50:11 +00:00
|
|
|
deprecations.setHandler(warning => warnings.push(warning))
|
2018-12-10 16:13:09 +00:00
|
|
|
process.enablePromiseAPIs = true
|
|
|
|
}
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
deprecations.setHandler(null)
|
|
|
|
process.throwDeprecation = true
|
|
|
|
|
2019-02-08 17:50:11 +00:00
|
|
|
promiseFunc = param => new Promise((resolve, reject) => resolve(param))
|
2018-12-10 16:13:09 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
it('acts as a pass-through for promise-based invocations', async () => {
|
|
|
|
enableCallbackWarnings()
|
2019-01-25 22:23:24 +00:00
|
|
|
promiseFunc = deprecate.promisify(promiseFunc)
|
2018-12-10 16:13:09 +00:00
|
|
|
|
|
|
|
const actual = await promiseFunc(expected)
|
|
|
|
expect(actual).to.equal(expected)
|
|
|
|
expect(warnings).to.have.lengthOf(0)
|
|
|
|
})
|
|
|
|
|
2019-02-07 18:25:20 +00:00
|
|
|
it('only calls back an error if the callback is called with (err, data)', (done) => {
|
|
|
|
enableCallbackWarnings()
|
|
|
|
let erringPromiseFunc = () => new Promise((resolve, reject) => {
|
|
|
|
reject(new Error('fail'))
|
|
|
|
})
|
|
|
|
erringPromiseFunc = deprecate.promisify(erringPromiseFunc)
|
|
|
|
|
|
|
|
erringPromiseFunc((err, data) => {
|
|
|
|
expect(data).to.be.an('undefined')
|
|
|
|
expect(err).to.be.an.instanceOf(Error).with.property('message', 'fail')
|
|
|
|
erringPromiseFunc(data => {
|
|
|
|
expect(data).to.not.be.an.instanceOf(Error)
|
|
|
|
expect(data).to.be.an('undefined')
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2018-12-10 16:13:09 +00:00
|
|
|
it('warns exactly once for callback-based invocations', (done) => {
|
|
|
|
enableCallbackWarnings()
|
2019-01-25 22:23:24 +00:00
|
|
|
promiseFunc = deprecate.promisify(promiseFunc)
|
2018-12-10 16:13:09 +00:00
|
|
|
|
|
|
|
let callbackCount = 0
|
|
|
|
const invocationCount = 3
|
2019-01-25 22:23:24 +00:00
|
|
|
const callback = (actual) => {
|
2018-12-10 16:13:09 +00:00
|
|
|
expect(actual).to.equal(expected)
|
|
|
|
expect(warnings).to.have.lengthOf(1)
|
|
|
|
expect(warnings[0]).to.include('promiseFunc')
|
|
|
|
callbackCount += 1
|
|
|
|
if (callbackCount === invocationCount) {
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 0; i < invocationCount; i += 1) {
|
|
|
|
promiseFunc(expected, callback)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2016-03-25 20:03:49 +00:00
|
|
|
})
|