fix(drafts): If a draft disappears during editing, re-save instead of blowing up (Sentry 2844, more.)

Summary:
Previously, if a draft was deleted while a DraftChangeSet had uncommitted changes, it would blow up trying to find the draft.

This is bad, and can happen in normal scenarios. There are several changes in this diff:

- The DraftChangeSet no longer retrieves the draft from the database before saving changes. Instead, the save logic is in the DraftStoreProxy which already has a version of the draft kept fresh by a subscription to the DraftStore.

- The SyncbackDraftTask already had logic for POSTing when a PUT to a draft returns a 404, so no new logic was necessary there. THe new "commit" logic causes us to put back the draft that was lost, and then when we save it we detatch it from it's serverId and re-save.

In manual testing, this transparently handles the case where you delete a draft from Gmail while editing it.

Test Plan: I've added 22 tests for the DraftChangeSet and DraftStoreProxy

Reviewers: dillon, evan

Reviewed By: dillon, evan

Differential Revision: https://phab.nylas.com/D2012
This commit is contained in:
Ben Gotow 2015-09-14 11:30:16 -07:00
parent 00839e37aa
commit e094261470
2 changed files with 233 additions and 30 deletions

View file

@ -0,0 +1,203 @@
Message = require '../../src/flux/models/message'
DatabaseStore = require '../../src/flux/stores/database-store'
DraftStoreProxy = require '../../src/flux/stores/draft-store-proxy'
DraftChangeSet = DraftStoreProxy.DraftChangeSet
_ = require 'underscore'
describe "DraftChangeSet", ->
beforeEach ->
@triggerSpy = jasmine.createSpy('trigger')
@commitResolve = null
@commitResolves = []
@commitSpy = jasmine.createSpy('commit').andCallFake =>
new Promise (resolve, reject) =>
@commitResolves.push(resolve)
@commitResolve = resolve
@changeSet = new DraftChangeSet(@triggerSpy, @commitSpy)
@changeSet._pending =
subject: 'Change to subject line'
describe "teardown", ->
it "should remove all of the pending and saving changes", ->
@changeSet.teardown()
expect(@changeSet._saving).toEqual({})
expect(@changeSet._pending).toEqual({})
describe "add", ->
it "should mark that the draft is not pristine", ->
@changeSet.add(body: 'Hello World!')
expect(@changeSet._pending.pristine).toEqual(false)
it "should add the changes to the _pending set", ->
@changeSet.add(body: 'Hello World!')
expect(@changeSet._pending.body).toEqual('Hello World!')
describe "when the immediate option is passed", ->
it "should commit", ->
spyOn(@changeSet, 'commit')
@changeSet.add({body: 'Hello World!'}, {immediate: true})
expect(@changeSet.commit).toHaveBeenCalled()
describe "otherwise", ->
it "should commit after five seconds", ->
spyOn(@changeSet, 'commit')
@changeSet.add({body: 'Hello World!'})
expect(@changeSet.commit).not.toHaveBeenCalled()
advanceClock(6000)
expect(@changeSet.commit).toHaveBeenCalled()
describe "commit", ->
it "should resolve immediately if the pending set is empty", ->
@changeSet._pending = {}
waitsForPromise =>
@changeSet.commit().then =>
expect(@commitSpy).not.toHaveBeenCalled()
it "should move changes to the saving set", ->
pendingBefore = _.extend({}, @changeSet._pending)
expect(@changeSet._saving).toEqual({})
@changeSet.commit()
advanceClock()
expect(@changeSet._pending).toEqual({})
expect(@changeSet._saving).toEqual(pendingBefore)
it "should call the commit handler and then clear the saving set", ->
@changeSet.commit()
advanceClock()
expect(@changeSet._saving).not.toEqual({})
@commitResolve()
advanceClock()
expect(@changeSet._saving).toEqual({})
describe "concurrency", ->
it "the commit function should always run serially", ->
firstFulfilled = false
secondFulfilled = false
@changeSet._pending = {subject: 'A'}
@changeSet.commit().then =>
@changeSet._pending = {subject: 'B'}
firstFulfilled = true
@changeSet.commit().then =>
secondFulfilled = true
advanceClock()
expect(firstFulfilled).toBe(false)
expect(secondFulfilled).toBe(false)
@commitResolves[0]()
advanceClock()
expect(firstFulfilled).toBe(true)
expect(secondFulfilled).toBe(false)
@commitResolves[1]()
advanceClock()
expect(firstFulfilled).toBe(true)
expect(secondFulfilled).toBe(true)
describe "applyToModel", ->
it "should apply the saving and then the pending change sets, in that order", ->
@changeSet._saving = {subject: 'A', body: 'Basketb'}
@changeSet._pending = {body: 'Basketball'}
m = new Message()
@changeSet.applyToModel(m)
expect(m.subject).toEqual('A')
expect(m.body).toEqual('Basketball')
describe "DraftStoreProxy", ->
describe "constructor", ->
it "should make a query to fetch the draft", ->
spyOn(DatabaseStore, 'run').andCallFake =>
new Promise (resolve, reject) =>
proxy = new DraftStoreProxy('client-id')
expect(DatabaseStore.run).toHaveBeenCalled()
describe "when given a draft object", ->
beforeEach ->
spyOn(DatabaseStore, 'run')
@draft = new Message(draft: true, body: '123')
@proxy = new DraftStoreProxy('client-id', @draft)
it "should not make a query for the draft", ->
expect(DatabaseStore.run).not.toHaveBeenCalled()
it "should immediately make the draft available", ->
expect(@proxy.draft()).toEqual(@draft)
describe "teardown", ->
it "should mark the session as destroyed", ->
proxy = new DraftStoreProxy('client-id')
proxy.teardown()
expect(proxy._destroyed).toEqual(true)
describe "prepare", ->
beforeEach ->
@draft = new Message(draft: true, body: '123', clientId: 'client-id')
@proxy = new DraftStoreProxy('client-id')
spyOn(@proxy, '_setDraft')
spyOn(DatabaseStore, 'run').andCallFake =>
Promise.resolve(@draft)
@proxy._draftPromise = null
it "should call setDraft with the retrieved draft", ->
waitsForPromise =>
@proxy.prepare().then =>
expect(@proxy._setDraft).toHaveBeenCalledWith(@draft)
it "should resolve with the DraftStoreProxy", ->
waitsForPromise =>
@proxy.prepare().then (val) =>
expect(val).toBe(@proxy)
describe "error handling", ->
it "should reject if the draft session has already been destroyed", ->
@proxy._destroyed = true
waitsForPromise =>
@proxy.prepare().then =>
expect(false).toBe(true)
.catch (val) =>
expect(val instanceof Error).toBe(true)
it "should reject if the draft cannot be found", ->
@draft = null
waitsForPromise =>
@proxy.prepare().then =>
expect(false).toBe(true)
.catch (val) =>
expect(val instanceof Error).toBe(true)
describe "when a draft changes", ->
beforeEach ->
@draft = new Message(draft: true, clientId: 'client-id', body: 'A', subject: 'initial')
@proxy = new DraftStoreProxy('client-id', @draft)
it "should ignore the update unless it applies to the current draft", ->
spyOn(@proxy, 'trigger')
@proxy._onDraftChanged(objectClass: 'message', objects: [new Message()])
expect(@proxy.trigger).not.toHaveBeenCalled()
@proxy._onDraftChanged(objectClass: 'message', objects: [@draft])
expect(@proxy.trigger).toHaveBeenCalled()
it "should apply the update to the current draft", ->
updatedDraft = @draft.clone()
updatedDraft.subject = 'This is the new subject'
@proxy._onDraftChanged(objectClass: 'message', objects: [updatedDraft])
expect(@proxy.draft().subject).toEqual(updatedDraft.subject)
describe "draft pristine body", ->
describe "when the draft given to the session is pristine", ->
it "should return the initial body", ->
pristineDraft = new Message(draft: true, body: 'Hiya', pristine: true, clientId: 'client-id')
updatedDraft = pristineDraft.clone()
updatedDraft.body = '123444'
updatedDraft.pristine = false
@proxy = new DraftStoreProxy('client-id', pristineDraft)
@proxy._onDraftChanged(objectClass: 'message', objects: [updatedDraft])
expect(@proxy.draftPristineBody()).toBe('Hiya')
describe "when the draft given to the session is not pristine", ->
it "should return null", ->
dirtyDraft = new Message(draft: true, body: 'Hiya', pristine: false)
@proxy = new DraftStoreProxy('client-id', dirtyDraft)
expect(@proxy.draftPristineBody()).toBe(null)

View file

@ -1,5 +1,6 @@
Message = require '../models/message'
Actions = require '../actions'
DatabaseStore = require './database-store'
{Listener, Publisher} = require '../modules/reflux-coffee'
SyncbackDraftTask = require '../tasks/syncback-draft'
@ -22,7 +23,7 @@ DraftChangeSet associated with the store proxy. The DraftChangeSet does two thin
Section: Drafts
###
class DraftChangeSet
constructor: (@clientId, @_onChange) ->
constructor: (@_onTrigger, @_onCommit) ->
@_commitChain = Promise.resolve()
@_pending = {}
@_saving = {}
@ -31,14 +32,14 @@ class DraftChangeSet
teardown: ->
@_pending = {}
@_saving = {}
@_destroyed = true
clearTimeout(@_timer) if @_timer
@_timer = null
if @_timer
clearTimeout(@_timer)
@_timer = null
add: (changes, {immediate, silent}={}) =>
@_pending = _.extend(@_pending, changes)
@_pending['pristine'] = false
@_onChange() unless silent
@_onTrigger() unless silent
if immediate
@commit()
else
@ -47,25 +48,13 @@ class DraftChangeSet
commit: =>
@_commitChain = @_commitChain.finally =>
if Object.keys(@_pending).length is 0 or @_destroyed
if Object.keys(@_pending).length is 0
return Promise.resolve(true)
DatabaseStore = require './database-store'
DatabaseStore.findBy(Message, clientId: @clientId).include(Message.attributes.body).then (draft) =>
if @_destroyed
return Promise.resolve(true)
if not draft
throw new Error("DraftChangeSet.commit: Assertion failure. Draft #{@clientId} is not in the database.")
@_saving = @_pending
@_pending = {}
draft = @applyToModel(draft)
return DatabaseStore.persistModel(draft).then =>
syncback = new SyncbackDraftTask(@clientId)
Actions.queueTask(syncback)
@_saving = {}
@_saving = @_pending
@_pending = {}
return @_onCommit().then =>
@_saving = {}
return @_commitChain
@ -96,18 +85,13 @@ class DraftStoreProxy
constructor: (@draftClientId, draft = null) ->
DraftStore = require './draft-store'
@listenTo DraftStore, @_onDraftChanged
@_draft = false
@_draftPristineBody = null
@_destroyed = false
@changes = new DraftChangeSet @draftClientId, =>
return if @_destroyed
if !@_draft
throw new Error("DraftChangeSet was modified before the draft was prepared.")
@trigger()
@changes = new DraftChangeSet(@_changeSetTrigger, @_changeSetCommit)
if draft
@_setDraft(draft)
@ -129,7 +113,6 @@ class DraftStoreProxy
@_draftPristineBody
prepare: ->
DatabaseStore = require './database-store'
@_draftPromise ?= DatabaseStore.findBy(Message, clientId: @draftClientId).include(Message.attributes.body).then (draft) =>
return Promise.reject(new Error("Draft has been destroyed.")) if @_destroyed
return Promise.reject(new Error("Assertion Failure: Draft #{@draftClientId} not found.")) if not draft
@ -161,9 +144,26 @@ class DraftStoreProxy
# Is this change an update to our draft?
myDrafts = _.filter(change.objects, (obj) => obj.clientId is @_draft.clientId)
if myDrafts.length > 0
@_draft = _.extend @_draft, _.last(myDrafts)
@trigger()
_changeSetTrigger: =>
return if @_destroyed
if !@_draft
throw new Error("DraftChangeSet was modified before the draft was prepared.")
@trigger()
_changeSetCommit: =>
if @_destroyed or not @_draft
return Promise.resolve(true)
updated = @changes.applyToModel(@_draft)
return DatabaseStore.persistModel(updated).then =>
Actions.queueTask(new SyncbackDraftTask(@draftClientId))
DraftStoreProxy.DraftChangeSet = DraftChangeSet
module.exports = DraftStoreProxy