mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-01 13:14:16 +08:00
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:
parent
00839e37aa
commit
e094261470
2 changed files with 233 additions and 30 deletions
203
spec-nylas/stores/draft-store-proxy-spec.coffee
Normal file
203
spec-nylas/stores/draft-store-proxy-spec.coffee
Normal 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)
|
|
@ -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
|
||||
|
|
Loading…
Reference in a new issue