2015-05-20 07:06:59 +08:00
|
|
|
_ = require 'underscore'
|
2015-05-16 01:53:00 +08:00
|
|
|
NylasLongConnection = require '../src/flux/nylas-long-connection'
|
|
|
|
NylasSyncWorker = require '../src/flux/nylas-sync-worker'
|
2015-07-17 08:47:02 +08:00
|
|
|
Namespace = require '../src/flux/models/namespace'
|
2015-04-07 02:46:20 +08:00
|
|
|
Thread = require '../src/flux/models/thread'
|
|
|
|
|
2015-05-16 01:53:00 +08:00
|
|
|
describe "NylasSyncWorker", ->
|
2015-04-07 02:46:20 +08:00
|
|
|
beforeEach ->
|
|
|
|
@apiRequests = []
|
|
|
|
@api =
|
2015-05-20 06:59:37 +08:00
|
|
|
makeRequest: (requestOptions) =>
|
|
|
|
@apiRequests.push({requestOptions})
|
2015-04-07 02:46:20 +08:00
|
|
|
getCollection: (namespace, model, params, requestOptions) =>
|
|
|
|
@apiRequests.push({namespace, model, params, requestOptions})
|
|
|
|
getThreads: (namespace, params, requestOptions) =>
|
|
|
|
@apiRequests.push({namespace, model:'threads', params, requestOptions})
|
|
|
|
|
|
|
|
spyOn(atom.config, 'get').andCallFake (key) =>
|
2015-07-17 08:47:02 +08:00
|
|
|
expected = "nylas.sync-state.namespace-id"
|
2015-05-20 06:59:37 +08:00
|
|
|
return throw new Error("Not stubbed! #{key}") unless key is expected
|
|
|
|
return _.extend {}, {
|
|
|
|
"contacts":
|
|
|
|
busy: true
|
|
|
|
complete: false
|
|
|
|
"calendars":
|
|
|
|
busy:false
|
|
|
|
complete: true
|
|
|
|
}
|
2015-04-07 02:46:20 +08:00
|
|
|
|
|
|
|
spyOn(atom.config, 'set').andCallFake (key, val) =>
|
2015-05-20 06:59:37 +08:00
|
|
|
return
|
2015-04-07 02:46:20 +08:00
|
|
|
|
2015-07-17 08:47:02 +08:00
|
|
|
@namespace = new Namespace(id: 'namespace-id', organizationUnit: 'label')
|
|
|
|
@worker = new NylasSyncWorker(@api, @namespace)
|
2015-04-07 02:46:20 +08:00
|
|
|
@connection = @worker.connection()
|
|
|
|
|
2015-05-20 06:59:37 +08:00
|
|
|
it "should reset `busy` to false when reading state from disk", ->
|
|
|
|
state = @worker.state()
|
|
|
|
expect(state.contacts.busy).toEqual(false)
|
|
|
|
|
2015-04-07 02:46:20 +08:00
|
|
|
describe "start", ->
|
|
|
|
it "should open the long polling connection", ->
|
|
|
|
spyOn(@connection, 'start')
|
|
|
|
@worker.start()
|
|
|
|
expect(@connection.start).toHaveBeenCalled()
|
|
|
|
|
2015-05-20 06:59:37 +08:00
|
|
|
it "should start querying for model collections and counts that haven't been fully cached", ->
|
2015-04-07 02:46:20 +08:00
|
|
|
@worker.start()
|
2015-07-17 08:47:02 +08:00
|
|
|
expect(@apiRequests.length).toBe(8)
|
2015-05-20 06:59:37 +08:00
|
|
|
modelsRequested = _.compact _.map @apiRequests, ({model}) -> model
|
2015-07-17 08:47:02 +08:00
|
|
|
expect(modelsRequested).toEqual(['threads', 'contacts', 'files', 'labels'])
|
2015-05-20 06:59:37 +08:00
|
|
|
|
|
|
|
countsRequested = _.compact _.map @apiRequests, ({requestOptions}) ->
|
|
|
|
if requestOptions.qs?.view is 'count'
|
|
|
|
return requestOptions.path
|
|
|
|
|
2015-07-17 08:47:02 +08:00
|
|
|
expect(modelsRequested).toEqual(['threads', 'contacts', 'files', 'labels'])
|
|
|
|
expect(countsRequested).toEqual(['/n/namespace-id/threads', '/n/namespace-id/contacts', '/n/namespace-id/files', '/n/namespace-id/labels'])
|
2015-04-07 02:46:20 +08:00
|
|
|
|
|
|
|
it "should mark incomplete collections as `busy`", ->
|
|
|
|
@worker.start()
|
2015-05-20 06:59:37 +08:00
|
|
|
nextState = @worker.state()
|
2015-04-07 02:46:20 +08:00
|
|
|
|
2015-07-17 08:47:02 +08:00
|
|
|
for collection in ['contacts','threads','files', 'labels']
|
2015-05-20 06:59:37 +08:00
|
|
|
expect(nextState[collection].busy).toEqual(true)
|
|
|
|
|
|
|
|
it "should initialize count and fetched to 0", ->
|
|
|
|
@worker.start()
|
|
|
|
nextState = @worker.state()
|
|
|
|
|
2015-07-17 08:47:02 +08:00
|
|
|
for collection in ['contacts','threads','files', 'labels']
|
2015-05-20 06:59:37 +08:00
|
|
|
expect(nextState[collection].fetched).toEqual(0)
|
|
|
|
expect(nextState[collection].count).toEqual(0)
|
|
|
|
|
|
|
|
it "should periodically try to restart failed collection syncs", ->
|
|
|
|
spyOn(@worker, 'resumeFetches').andCallThrough()
|
|
|
|
@worker.start()
|
|
|
|
advanceClock(50000)
|
|
|
|
expect(@worker.resumeFetches.callCount).toBe(2)
|
|
|
|
|
|
|
|
describe "when a count request completes", ->
|
2015-04-07 02:46:20 +08:00
|
|
|
beforeEach ->
|
|
|
|
@worker.start()
|
|
|
|
@request = @apiRequests[0]
|
|
|
|
@apiRequests = []
|
|
|
|
|
2015-05-20 06:59:37 +08:00
|
|
|
it "should update the count on the collection", ->
|
|
|
|
@request.requestOptions.success({count: 1001})
|
|
|
|
nextState = @worker.state()
|
|
|
|
expect(nextState.threads.count).toEqual(1001)
|
|
|
|
|
|
|
|
describe "resumeFetches", ->
|
|
|
|
it "should fetch collections", ->
|
|
|
|
spyOn(@worker, 'fetchCollection')
|
|
|
|
@worker.resumeFetches()
|
2015-07-17 08:47:02 +08:00
|
|
|
expect(@worker.fetchCollection.calls.map (call) -> call.args[0]).toEqual(['threads', 'calendars', 'contacts', 'files', 'labels'])
|
2015-05-20 06:59:37 +08:00
|
|
|
|
|
|
|
describe "fetchCollection", ->
|
|
|
|
beforeEach ->
|
|
|
|
@apiRequests = []
|
|
|
|
|
|
|
|
it "should not start if the collection sync is already in progress", ->
|
|
|
|
@worker._state.threads = {
|
|
|
|
'busy': true
|
|
|
|
'complete': false
|
|
|
|
}
|
|
|
|
@worker.fetchCollection('threads')
|
|
|
|
expect(@apiRequests.length).toBe(0)
|
|
|
|
|
|
|
|
it "should not start if the collection sync is already complete", ->
|
|
|
|
@worker._state.threads = {
|
|
|
|
'busy': false
|
|
|
|
'complete': true
|
|
|
|
}
|
|
|
|
@worker.fetchCollection('threads')
|
|
|
|
expect(@apiRequests.length).toBe(0)
|
|
|
|
|
|
|
|
it "should start the request for the model count", ->
|
|
|
|
@worker._state.threads = {
|
|
|
|
'busy': false
|
|
|
|
'complete': false
|
|
|
|
}
|
|
|
|
@worker.fetchCollection('threads')
|
|
|
|
expect(@apiRequests[0].requestOptions.path).toBe('/n/namespace-id/threads')
|
|
|
|
expect(@apiRequests[0].requestOptions.qs.view).toBe('count')
|
|
|
|
|
|
|
|
it "should start the first request for models", ->
|
|
|
|
@worker._state.threads = {
|
|
|
|
'busy': false
|
|
|
|
'complete': false
|
|
|
|
}
|
|
|
|
@worker.fetchCollection('threads')
|
|
|
|
expect(@apiRequests[1].model).toBe('threads')
|
|
|
|
expect(@apiRequests[1].params.offset).toBe(0)
|
|
|
|
|
|
|
|
describe "when an API request completes", ->
|
|
|
|
beforeEach ->
|
|
|
|
@worker.start()
|
|
|
|
@request = @apiRequests[1]
|
|
|
|
@apiRequests = []
|
|
|
|
|
2015-04-07 02:46:20 +08:00
|
|
|
describe "successfully, with models", ->
|
|
|
|
it "should request the next page", ->
|
2015-05-20 06:59:37 +08:00
|
|
|
pageSize = @request.params.limit
|
2015-04-07 02:46:20 +08:00
|
|
|
models = []
|
2015-05-20 06:59:37 +08:00
|
|
|
models.push(new Thread) for i in [0..(pageSize-1)]
|
2015-04-07 02:46:20 +08:00
|
|
|
@request.requestOptions.success(models)
|
|
|
|
expect(@apiRequests.length).toBe(1)
|
2015-05-20 06:59:37 +08:00
|
|
|
expect(@apiRequests[0].params).toEqual
|
|
|
|
limit: pageSize,
|
|
|
|
offset: @request.params.offset + pageSize
|
|
|
|
|
|
|
|
it "should update the fetched count on the collection", ->
|
|
|
|
expect(@worker.state().threads.fetched).toEqual(0)
|
|
|
|
pageSize = @request.params.limit
|
|
|
|
models = []
|
|
|
|
models.push(new Thread) for i in [0..(pageSize-1)]
|
|
|
|
@request.requestOptions.success(models)
|
|
|
|
expect(@worker.state().threads.fetched).toEqual(pageSize)
|
2015-04-07 02:46:20 +08:00
|
|
|
|
|
|
|
describe "successfully, with fewer models than requested", ->
|
|
|
|
beforeEach ->
|
|
|
|
models = []
|
|
|
|
models.push(new Thread) for i in [0..100]
|
|
|
|
@request.requestOptions.success(models)
|
|
|
|
|
|
|
|
it "should not request another page", ->
|
|
|
|
expect(@apiRequests.length).toBe(0)
|
|
|
|
|
|
|
|
it "should update the state to complete", ->
|
2015-05-20 06:59:37 +08:00
|
|
|
expect(@worker.state().threads.busy).toEqual(false)
|
|
|
|
expect(@worker.state().threads.complete).toEqual(true)
|
|
|
|
|
|
|
|
it "should update the fetched count on the collection", ->
|
|
|
|
expect(@worker.state().threads.fetched).toEqual(101)
|
2015-04-07 02:46:20 +08:00
|
|
|
|
|
|
|
describe "successfully, with no models", ->
|
|
|
|
it "should not request another page", ->
|
|
|
|
@request.requestOptions.success([])
|
|
|
|
expect(@apiRequests.length).toBe(0)
|
|
|
|
|
|
|
|
it "should update the state to complete", ->
|
|
|
|
@request.requestOptions.success([])
|
2015-05-20 06:59:37 +08:00
|
|
|
expect(@worker.state().threads.busy).toEqual(false)
|
|
|
|
expect(@worker.state().threads.complete).toEqual(true)
|
2015-04-07 02:46:20 +08:00
|
|
|
|
|
|
|
describe "with an error", ->
|
|
|
|
it "should log the error to the state", ->
|
|
|
|
err = new Error("Oh no a network error")
|
|
|
|
@request.requestOptions.error(err)
|
2015-05-20 06:59:37 +08:00
|
|
|
expect(@worker.state().threads.busy).toEqual(false)
|
|
|
|
expect(@worker.state().threads.complete).toEqual(false)
|
|
|
|
expect(@worker.state().threads.error).toEqual(err.toString())
|
2015-04-07 02:46:20 +08:00
|
|
|
|
|
|
|
it "should not request another page", ->
|
|
|
|
@request.requestOptions.error(new Error("Oh no a network error"))
|
|
|
|
expect(@apiRequests.length).toBe(0)
|
|
|
|
|
|
|
|
describe "cleanup", ->
|
|
|
|
it "should termiate the long polling connection", ->
|
|
|
|
spyOn(@connection, 'end')
|
|
|
|
@worker.cleanup()
|
|
|
|
expect(@connection.end).toHaveBeenCalled()
|
2015-05-20 06:59:37 +08:00
|
|
|
|
|
|
|
it "should stop trying to restart failed collection syncs", ->
|
2015-06-16 09:29:59 +08:00
|
|
|
spyOn(console, 'log')
|
2015-05-20 06:59:37 +08:00
|
|
|
spyOn(@worker, 'resumeFetches').andCallThrough()
|
|
|
|
@worker.cleanup()
|
|
|
|
advanceClock(50000)
|
|
|
|
expect(@worker.resumeFetches.callCount).toBe(0)
|