Skip to content

Commit

Permalink
Improve Observer tests
Browse files Browse the repository at this point in the history
Basically all tests are rewritten here. Implementation details should
not be checked because tests are becoming fragile when it comes to any
refactoring.

What instead should be tested is the actual **public behavior** of the
library when using its public interfaces.

While doing all these changes the following assumptions about this codebase
have been made:
 - class fields which are prefixed with underscore (like `observer._onNext`)
 are intended to be private and thus should only be accessed by members of
 the same class or its subclasses
 - `observer.stopped` is a public **readonly** field and should never be
 modified by the outside world, because its role is considered to be
 purely informational

If the above assumptions are not correct, this will need to be discussed
further. These improvements of the tests are necessary to introduce any
changes which include refactoring of the `Observer` in order to fix some
bugs.
  • Loading branch information
4O4 committed Oct 2, 2019
1 parent f9c29e1 commit f5d29f7
Showing 1 changed file with 131 additions and 54 deletions.
185 changes: 131 additions & 54 deletions tests/observer.lua
Original file line number Diff line number Diff line change
@@ -1,91 +1,168 @@
describe('Observer', function()
describe('create', function()
it('returns an Observer', function()
expect(Rx.Observer.create()).to.be.an(Rx.Observer)
end)

it('assigns onNext, onError, and onCompleted', function()
local function onNext() end
local function onError() end
local function onCompleted() end

local observer = Rx.Observer.create(onNext, onError, onCompleted)
describe('create', function()
local function expectObserverToBeInCleanStateAndValid(observer)
expect(observer).to.be.an(Rx.Observer)
expect(observer.stopped).to.equal(false)
end

expect(observer._onNext).to.equal(onNext)
expect(observer._onError).to.equal(onError)
expect(observer._onCompleted).to.equal(onCompleted)
it('works when no parameters is passed', function()
local observer = Rx.Observer.create()
expectObserverToBeInCleanStateAndValid(observer)
end)

it('initializes stopped to false', function()
expect(Rx.Observer.create().stopped).to.equal(false)

it('works when onNext callback parameter is passed', function()
local observer = Rx.Observer.create(function () end)
expectObserverToBeInCleanStateAndValid(observer)
end)

it('works when onError callback parameter is passed', function()
local observer = Rx.Observer.create(nil, function () end)
expectObserverToBeInCleanStateAndValid(observer)
end)

it('works when onComplete callback parameter is passed', function()
local observer = Rx.Observer.create(nil, nil, function () end)
expectObserverToBeInCleanStateAndValid(observer)
end)

it('works when all callback parameters are passed', function()
local observer = Rx.Observer.create(function () end, function () end, function () end)
expectObserverToBeInCleanStateAndValid(observer)
end)
end)

describe('onNext', function()
it('calls _onNext', function()
local observer = Rx.Observer.create()
local function run() observer:onNext() end
expect(#spy(observer, '_onNext', run)).to.equal(1)
it('calls custom onNext callback if it was provided', function()
local onNext = spy()
local observer = Rx.Observer.create(onNext, nil, nil)
observer:onNext()
expect(#onNext).to.equal(1)
end)

it('passes all arguments to _onNext', function()
local observer = Rx.Observer.create()
local function run() observer:onNext(1, '2', 3, nil, 5) end
expect(spy(observer, '_onNext', run)).to.equal({{1, '2', 3, nil, 5}})
it('passes all arguments to custom onNext callback if it was provided', function()
local onNext = spy()
local observer = Rx.Observer.create(onNext, nil, nil)
observer:onNext(1, '2', 3, nil, 5, { key = 6 })
expect(onNext).to.equal({{1, '2', 3, nil, 5, { key = 6 }}})
end)

it('does not call _onNext if stopped is true', function()
it('works and does not error when custom onNext callback was not provided', function()
local observer = Rx.Observer.create()
observer.stopped = true
local function run() observer:onNext() end
expect(#spy(observer, '_onNext', run)).to.equal(0)
local errors = {}

-- would gladly use something like to_not.fail() here but it's
-- not quite good with producing useful error messages
local success = tryCall(function () observer:onNext() end, errors)
tryCall(function () expect(success).to.equal(true) end, errors)
throwErrorsIfAny(errors)
end)

describe('does not call custom onError callback', function ()
it('if observer already received completion notification', function()
local onNext = spy()
local observer = Rx.Observer.create(onNext, nil, nil)
observer:onCompleted()
observer:onNext()
expect(#onNext).to.equal(0)
end)

it('if observer already received error notification', function()
local onNext = spy()
local observer = Rx.Observer.create(onNext, nil, nil)
observer:onCompleted()
observer:onNext()
expect(#onNext).to.equal(0)
end)
end)
end)

describe('onError', function()
it('calls _onError with the first argument it was passed', function()
local observer = Rx.Observer.create(_, function() end, _)
local function run() observer:onError('sheeit', 1) end
expect(spy(observer, '_onError', run)).to.equal({{'sheeit'}})
it('causes an error by default if custom onError callback was not provided', function()
local observer = Rx.Observer.create()
expect(function () observer:onError() end).to.fail()
end)

it('sets stopped to true', function()
local observer = Rx.Observer.create(_, function() end, _)
it('calls custom onError callback if it was provided', function()
local onError = spy()
local observer = Rx.Observer.create(nil, onError, nil)
observer:onError()
expect(observer.stopped).to.equal(true)
expect(#onError).to.equal(1)
end)

it('does not call _onError if stopped is already true', function()
local observer = Rx.Observer.create(_, function() end, _)
observer.stopped = true
local function run() observer:onError() end
expect(#spy(observer, '_onError', run)).to.equal(0)
it('passes first value from error notification to custom onError callback', function()
local onError = spy()
local observer = Rx.Observer.create(nil, onError, nil)
observer:onError("err msg", "excessive arg", 1)
expect(onError).to.equal({{"err msg"}})
end)

it('causes an error by default', function()
local observer = Rx.Observer.create()
expect(observer.onError).to.fail()
it('marks observer as stopped', function()
local observer = Rx.Observer.create(nil, function() end, nil)
observer:onError()
expect(observer.stopped).to.equal(true)
end)

describe('does not call custom onError callback', function ()
it('if observer already received completion notification', function()
local onError = spy()
local observer = Rx.Observer.create(nil, onError, nil)
observer:onCompleted()
observer:onError()
expect(#onError).to.equal(0)
end)

it('if observer already received error notification', function()
local spyEnabled = false
local onError = spy()
local observer = Rx.Observer.create(nil, function () if spyEnabled then onError() end end, nil)
observer:onError()
spyEnabled = true
observer:onError()
expect(#onError).to.equal(0)
end)
end)
end)

describe('onCompleted', function()
it('calls _onCompleted with no arguments', function()
local observer = Rx.Observer.create()
local function run() observer:onCompleted(1, 2, 3) end
expect(spy(observer, '_onCompleted', run)).to.equal({{}})
it('calls custom onCompleted callback if it was provided', function()
local onCompleted = spy()
local observer = Rx.Observer.create(nil, nil, onCompleted)
observer:onCompleted()
expect(#onCompleted).to.equal(1)
end)

it('sets stopped to true', function()
local observer = Rx.Observer.create()
it('calls custom onCompleted callback with no parameters', function()
local onCompleted = spy()
local observer = Rx.Observer.create(nil, nil, onCompleted)
observer:onCompleted("excessive arg", 1)
expect(onCompleted).to.equal({{}})
end)

it('marks observer as stopped', function()
local observer = Rx.Observer.create(nil, function() end, nil)
observer:onCompleted()
expect(observer.stopped).to.equal(true)
end)

it('does not call _onCompleted if stopped is already true', function()
local observer = Rx.Observer.create()
observer.stopped = true
local function run() observer:onCompleted() end
expect(#spy(observer, '_onCompleted', run)).to.equal(0)
describe('does not call custom onCompleted callback', function ()
it('if observer already received completion notification', function()
local spyEnabled = false
local onCompleted = spy()
local observer = Rx.Observer.create(nil, function () if spyEnabled then onCompleted() end end, nil)
observer:onCompleted()
spyEnabled = true
observer:onCompleted()
expect(#onCompleted).to.equal(0)
end)

it('if observer already received error notification', function()
local onCompleted = spy()
local observer = Rx.Observer.create(nil, function () end, onCompleted)
observer:onError()
observer:onCompleted()
expect(#onCompleted).to.equal(0)
end)
end)
end)
end)

0 comments on commit f5d29f7

Please sign in to comment.