mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-22 08:00:25 +08:00
b00e5439e5
Summary: Related to #320, #494, #515, #553 Ignore newlines and returns in HTML, they can be inside tags Allow all attributes so that paste from excel looks nice Never let someone paste a `contenteditable` attribute Update specs Test Plan: Run new specs Reviewers: juan, evan Reviewed By: evan Differential Revision: https://phab.nylas.com/D2309
957 lines
40 KiB
CoffeeScript
957 lines
40 KiB
CoffeeScript
File = require '../../src/flux/models/file'
|
|
Thread = require '../../src/flux/models/thread'
|
|
Message = require '../../src/flux/models/message'
|
|
Contact = require '../../src/flux/models/contact'
|
|
ModelQuery = require '../../src/flux/models/query'
|
|
AccountStore = require '../../src/flux/stores/account-store'
|
|
DatabaseStore = require '../../src/flux/stores/database-store'
|
|
DraftStore = require '../../src/flux/stores/draft-store'
|
|
ComposerExtension = require '../../src/flux/extensions/composer-extension'
|
|
SendDraftTask = require '../../src/flux/tasks/send-draft'
|
|
DestroyDraftTask = require '../../src/flux/tasks/destroy-draft'
|
|
SoundRegistry = require '../../src/sound-registry'
|
|
Actions = require '../../src/flux/actions'
|
|
Utils = require '../../src/flux/models/utils'
|
|
|
|
InlineStyleTransformer = require '../../src/services/inline-style-transformer'
|
|
SanitizeTransformer = require '../../src/services/sanitize-transformer'
|
|
|
|
{ipcRenderer} = require 'electron'
|
|
_ = require 'underscore'
|
|
|
|
fakeThread = null
|
|
fakeMessage1 = null
|
|
fakeMessage2 = null
|
|
msgFromMe = null
|
|
msgWithReplyTo = null
|
|
msgWithReplyToDuplicates = null
|
|
messageWithStyleTags = null
|
|
fakeMessages = null
|
|
fakeMessageWithFiles = null
|
|
|
|
class TestExtension extends ComposerExtension
|
|
@prepareNewDraft: (draft) ->
|
|
draft.body = "Edited by TestExtension!" + draft.body
|
|
|
|
describe "DraftStore", ->
|
|
beforeEach ->
|
|
spyOn(NylasEnv, 'newWindow').andCallFake ->
|
|
for id, session of DraftStore._draftSessions
|
|
if session.teardown
|
|
DraftStore._doneWithSession(session)
|
|
DraftStore._draftSessions = {}
|
|
|
|
describe "creating drafts", ->
|
|
beforeEach ->
|
|
spyOn(DraftStore, "_prepareBodyForQuoting").andCallFake (body) ->
|
|
Promise.resolve(body)
|
|
spyOn(ipcRenderer, "send").andCallFake (message, body) ->
|
|
if message is "inline-style-parse"
|
|
# There needs to be a defer block in here so the promise
|
|
# responsible for handling the `inline-style-parse` can be
|
|
# properly set. If the whole path is synchronous instead of
|
|
# asynchrounous, the promise is not cleared properly. Doing this
|
|
# requires us to add `advanceClock` blocks.
|
|
_.defer -> DraftStore._onInlineStylesResult({}, body)
|
|
|
|
fakeThread = new Thread
|
|
id: 'fake-thread-id'
|
|
subject: 'Fake Subject'
|
|
|
|
fakeMessage1 = new Message
|
|
id: 'fake-message-1'
|
|
to: [new Contact(email: 'ben@nylas.com'), new Contact(email: 'evan@nylas.com')]
|
|
cc: [new Contact(email: 'mg@nylas.com'), new Contact(email: AccountStore.current().me().email)]
|
|
bcc: [new Contact(email: 'recruiting@nylas.com')]
|
|
from: [new Contact(email: 'customer@example.com', name: 'Customer')]
|
|
threadId: 'fake-thread-id'
|
|
body: 'Fake Message 1'
|
|
subject: 'Fake Subject'
|
|
date: new Date(1415814587)
|
|
|
|
fakeMessage2 = new Message
|
|
id: 'fake-message-2'
|
|
to: [new Contact(email: 'customer@example.com')]
|
|
from: [new Contact(email: 'ben@nylas.com')]
|
|
threadId: 'fake-thread-id'
|
|
body: 'Fake Message 2'
|
|
subject: 'Re: Fake Subject'
|
|
date: new Date(1415814587)
|
|
|
|
fakeMessageWithFiles = new Message
|
|
id: 'fake-message-with-files'
|
|
to: [new Contact(email: 'ben@nylas.com'), new Contact(email: 'evan@nylas.com')]
|
|
cc: [new Contact(email: 'mg@nylas.com'), new Contact(email: AccountStore.current().me().email)]
|
|
bcc: [new Contact(email: 'recruiting@nylas.com')]
|
|
from: [new Contact(email: 'customer@example.com', name: 'Customer')]
|
|
files: [new File(filename: "test.jpg"), new File(filename: "test.pdj")]
|
|
threadId: 'fake-thread-id'
|
|
body: 'Fake Message 1'
|
|
subject: 'Fake Subject'
|
|
date: new Date(1415814587)
|
|
|
|
msgFromMe = new Message
|
|
id: 'fake-message-3'
|
|
to: [new Contact(email: '1@1.com'), new Contact(email: '2@2.com')]
|
|
cc: [new Contact(email: '3@3.com'), new Contact(email: '4@4.com')]
|
|
bcc: [new Contact(email: '5@5.com'), new Contact(email: '6@6.com')]
|
|
from: [new Contact(email: AccountStore.current().me().email)]
|
|
threadId: 'fake-thread-id'
|
|
body: 'Fake Message 2'
|
|
subject: 'Re: Fake Subject'
|
|
date: new Date(1415814587)
|
|
|
|
msgWithReplyTo = new Message
|
|
id: 'fake-message-reply-to'
|
|
to: [new Contact(email: '1@1.com'), new Contact(email: '2@2.com')]
|
|
cc: [new Contact(email: '3@3.com'), new Contact(email: '4@4.com')]
|
|
bcc: [new Contact(email: '5@5.com'), new Contact(email: '6@6.com')]
|
|
replyTo: [new Contact(email: 'reply-to@5.com'), new Contact(email: 'reply-to@6.com')]
|
|
from: [new Contact(email: 'from@5.com')]
|
|
threadId: 'fake-thread-id'
|
|
body: 'Fake Message 2'
|
|
subject: 'Re: Fake Subject'
|
|
date: new Date(1415814587)
|
|
|
|
msgWithReplyToDuplicates = new Message
|
|
id: 'fake-message-reply-to-duplicates'
|
|
to: [new Contact(email: '1@1.com'), new Contact(email: '2@2.com')]
|
|
cc: [new Contact(email: '1@1.com'), new Contact(email: '4@4.com')]
|
|
from: [new Contact(email: 'reply-to@5.com')]
|
|
replyTo: [new Contact(email: 'reply-to@5.com')]
|
|
threadId: 'fake-thread-id'
|
|
body: 'Fake Message Duplicates'
|
|
subject: 'Re: Fake Subject'
|
|
date: new Date(1415814587)
|
|
|
|
messageWithStyleTags = new Message
|
|
id: 'message-with-style-tags'
|
|
to: [new Contact(email: 'ben@nylas.com'), new Contact(email: 'evan@nylas.com')]
|
|
cc: [new Contact(email: 'mg@nylas.com'), new Contact(email: AccountStore.current().me().email)]
|
|
bcc: [new Contact(email: 'recruiting@nylas.com')]
|
|
from: [new Contact(email: 'customer@example.com', name: 'Customer')]
|
|
threadId: 'fake-thread-id'
|
|
body: '<style>div {color: red;}</style><div>Fake Message 1</div>'
|
|
subject: 'Fake Subject'
|
|
date: new Date(1415814587)
|
|
|
|
fakeMessages =
|
|
'fake-message-1': fakeMessage1
|
|
'fake-message-3': msgFromMe
|
|
'fake-message-2': fakeMessage2
|
|
'fake-message-reply-to': msgWithReplyTo
|
|
'fake-message-with-files': fakeMessageWithFiles
|
|
'fake-message-reply-to-duplicates': msgWithReplyToDuplicates
|
|
'message-with-style-tags': messageWithStyleTags
|
|
|
|
spyOn(DatabaseStore, 'find').andCallFake (klass, id) ->
|
|
query = new ModelQuery(klass, {id})
|
|
spyOn(query, 'then').andCallFake (fn) ->
|
|
return fn(fakeThread) if klass is Thread
|
|
return fn(fakeMessages[id]) if klass is Message
|
|
return fn(new Error('Not Stubbed'))
|
|
query
|
|
|
|
spyOn(DatabaseStore, 'run').andCallFake (query) ->
|
|
return Promise.resolve(fakeMessage2) if query._klass is Message
|
|
return Promise.reject(new Error('Not Stubbed'))
|
|
spyOn(DatabaseStore, 'persistModel').andCallFake -> Promise.resolve()
|
|
|
|
afterEach ->
|
|
# Have to cleanup the DraftStoreProxy objects or we'll get a memory
|
|
# leak error
|
|
for id, session of DraftStore._draftSessions
|
|
DraftStore._doneWithSession(session)
|
|
|
|
describe "onComposeReply", ->
|
|
beforeEach ->
|
|
runs ->
|
|
DraftStore._onComposeReply({threadId: fakeThread.id, messageId: fakeMessage1.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
|
|
it "should include quoted text", ->
|
|
expect(@model.body.indexOf('blockquote') > 0).toBe(true)
|
|
expect(@model.body.indexOf(fakeMessage1.body) > 0).toBe(true)
|
|
|
|
it "should address the message to the previous message's sender", ->
|
|
expect(@model.to).toEqual(fakeMessage1.from)
|
|
|
|
it "should set the replyToMessageId to the previous message's ids", ->
|
|
expect(@model.replyToMessageId).toEqual(fakeMessage1.id)
|
|
|
|
it "should sanitize the HTML", ->
|
|
expect(DraftStore._prepareBodyForQuoting).toHaveBeenCalled()
|
|
|
|
describe "onComposeReply", ->
|
|
describe "when the message provided as context has one or more 'ReplyTo' recipients", ->
|
|
it "addresses the draft to all of the message's 'ReplyTo' recipients", ->
|
|
runs ->
|
|
DraftStore._onComposeReply({threadId: fakeThread.id, messageId: msgWithReplyTo.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
expect(@model.to).toEqual(msgWithReplyTo.replyTo)
|
|
expect(@model.cc.length).toBe 0
|
|
expect(@model.bcc.length).toBe 0
|
|
|
|
describe "onComposeReply", ->
|
|
describe "when the message provided as context was sent by the current user", ->
|
|
it "addresses the draft to all of the last messages's 'To' recipients", ->
|
|
runs ->
|
|
DraftStore._onComposeReply({threadId: fakeThread.id, messageId: msgFromMe.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
expect(@model.to).toEqual(msgFromMe.to)
|
|
expect(@model.cc.length).toBe 0
|
|
expect(@model.bcc.length).toBe 0
|
|
|
|
describe "onComposeReplyAll", ->
|
|
beforeEach ->
|
|
runs ->
|
|
DraftStore._onComposeReplyAll({threadId: fakeThread.id, messageId: fakeMessage1.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
|
|
it "should include quoted text", ->
|
|
expect(@model.body.indexOf('blockquote') > 0).toBe(true)
|
|
expect(@model.body.indexOf(fakeMessage1.body) > 0).toBe(true)
|
|
|
|
it "should address the message to the previous message's sender", ->
|
|
expect(@model.to).toEqual(fakeMessage1.from)
|
|
|
|
it "should cc everyone who was on the previous message in to or cc", ->
|
|
ccEmails = @model.cc.map (cc) -> cc.email
|
|
expect(ccEmails.sort()).toEqual([ 'ben@nylas.com', 'evan@nylas.com', 'mg@nylas.com'])
|
|
|
|
it "should not include people who were bcc'd on the previous message", ->
|
|
expect(@model.bcc).toEqual([])
|
|
expect(@model.cc.indexOf(fakeMessage1.bcc[0])).toEqual(-1)
|
|
|
|
it "should not include you when you were cc'd on the previous message", ->
|
|
ccEmails = @model.cc.map (cc) -> cc.email
|
|
expect(ccEmails.indexOf(AccountStore.current().me().email)).toEqual(-1)
|
|
|
|
it "should set the replyToMessageId to the previous message's ids", ->
|
|
expect(@model.replyToMessageId).toEqual(fakeMessage1.id)
|
|
|
|
it "should sanitize the HTML", ->
|
|
expect(DraftStore._prepareBodyForQuoting).toHaveBeenCalled()
|
|
|
|
describe "onComposeReplyAll", ->
|
|
describe "when the message provided as context has one or more 'ReplyTo' recipients", ->
|
|
beforeEach ->
|
|
runs ->
|
|
DraftStore._onComposeReply({threadId: fakeThread.id, messageId: msgWithReplyTo.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
|
|
it "addresses the draft to all of the message's 'ReplyTo' recipients", ->
|
|
expect(@model.to).toEqual(msgWithReplyTo.replyTo)
|
|
|
|
it "should not include the message's 'From' recipient in any field", ->
|
|
all = [].concat(@model.to, @model.cc, @model.bcc)
|
|
match = _.find all, (c) -> c.email is msgWithReplyTo.from[0].email
|
|
expect(match).toEqual(undefined)
|
|
|
|
describe "onComposeReplyAll", ->
|
|
describe "when the message provided has one or more 'ReplyTo' recipients and duplicates in the To/Cc fields", ->
|
|
it "should unique the to/cc fields", ->
|
|
runs ->
|
|
DraftStore._onComposeReplyAll({threadId: fakeThread.id, messageId: msgWithReplyToDuplicates.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
ccEmails = model.cc.map (cc) -> cc.email
|
|
expect(ccEmails.sort()).toEqual(['1@1.com', '2@2.com', '4@4.com'])
|
|
toEmails = model.to.map (to) -> to.email
|
|
expect(toEmails.sort()).toEqual(['reply-to@5.com'])
|
|
|
|
describe "onComposeReplyAll", ->
|
|
describe "when the message provided as context was sent by the current user", ->
|
|
it "addresses the draft to all of the last messages's recipients", ->
|
|
runs ->
|
|
DraftStore._onComposeReplyAll({threadId: fakeThread.id, messageId: msgFromMe.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
expect(@model.to).toEqual(msgFromMe.to)
|
|
expect(@model.cc).toEqual(msgFromMe.cc)
|
|
expect(@model.bcc.length).toBe 0
|
|
|
|
describe "forwarding with attachments", ->
|
|
it "should include the attached files", ->
|
|
runs ->
|
|
DraftStore._onComposeForward({threadId: fakeThread.id, messageId: fakeMessageWithFiles.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
expect(@model.files.length).toBe 2
|
|
expect(@model.files[0].filename).toBe "test.jpg"
|
|
|
|
describe "onComposeForward", ->
|
|
beforeEach ->
|
|
runs ->
|
|
DraftStore._onComposeForward({threadId: fakeThread.id, messageId: fakeMessage1.id})
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
|
|
it "should include quoted text", ->
|
|
expect(@model.body.indexOf('blockquote') > 0).toBe(true)
|
|
expect(@model.body.indexOf(fakeMessage1.body) > 0).toBe(true)
|
|
|
|
it "should not address the message to anyone", ->
|
|
expect(@model.to).toEqual([])
|
|
expect(@model.cc).toEqual([])
|
|
expect(@model.bcc).toEqual([])
|
|
|
|
it "should not set the replyToMessageId", ->
|
|
expect(@model.replyToMessageId).toEqual(undefined)
|
|
|
|
it "should sanitize the HTML", ->
|
|
expect(DraftStore._prepareBodyForQuoting).toHaveBeenCalled()
|
|
|
|
describe "popout drafts", ->
|
|
beforeEach ->
|
|
spyOn(Actions, "composePopoutDraft")
|
|
|
|
it "can popout a reply", ->
|
|
runs ->
|
|
DraftStore._onComposeReply({threadId: fakeThread.id, messageId: fakeMessage1.id, popout: true}).catch (error) -> throw new Error (error)
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
expect(Actions.composePopoutDraft).toHaveBeenCalledWith(@model.clientId)
|
|
|
|
it "can popout a forward", ->
|
|
runs ->
|
|
DraftStore._onComposeForward({threadId: fakeThread.id, messageId: fakeMessage1.id, popout: true}).catch (error) -> throw new Error (error)
|
|
waitsFor ->
|
|
DatabaseStore.persistModel.callCount > 0
|
|
runs ->
|
|
@model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
expect(Actions.composePopoutDraft).toHaveBeenCalledWith(@model.clientId)
|
|
|
|
describe "_newMessageWithContext", ->
|
|
beforeEach ->
|
|
# A helper method that makes it easy to test _newMessageWithContext, which
|
|
# is asynchronous and whose output is a model persisted to the database.
|
|
@_callNewMessageWithContext = (context, attributesCallback, modelCallback) ->
|
|
waitsForPromise ->
|
|
DraftStore._newMessageWithContext(context, attributesCallback).then ->
|
|
model = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
modelCallback(model) if modelCallback
|
|
|
|
it "should create a new message", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
{}
|
|
, (model) ->
|
|
expect(model.constructor).toBe(Message)
|
|
|
|
it "should setup a draft session for the draftClientId, so that a subsequent request for the session's draft resolves immediately.", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
{}
|
|
, (model) ->
|
|
session = DraftStore.sessionForClientId(model.id).value()
|
|
expect(session.draft()).toBe(model)
|
|
|
|
it "should set the subject of the new message automatically", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
{}
|
|
, (model) ->
|
|
expect(model.subject).toEqual("Re: Fake Subject")
|
|
|
|
it "should apply attributes provided by the attributesCallback", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
subject: "Fwd: Fake subject"
|
|
to: [new Contact(email: 'weird@example.com')]
|
|
, (model) ->
|
|
expect(model.subject).toEqual("Fwd: Fake subject")
|
|
|
|
describe "extensions", ->
|
|
beforeEach ->
|
|
DraftStore.registerExtension(TestExtension)
|
|
afterEach ->
|
|
DraftStore.unregisterExtension(TestExtension)
|
|
|
|
it "should give extensions a chance to customize the draft via ext.prepareNewDraft", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
{}
|
|
, (model) ->
|
|
expect(model.body.indexOf("Edited by TestExtension!")).toBe(0)
|
|
|
|
describe "context", ->
|
|
it "should accept `thread` or look up a thread when given `threadId`", ->
|
|
@_callNewMessageWithContext {thread: fakeThread}
|
|
, (thread, message) ->
|
|
expect(thread).toBe(fakeThread)
|
|
expect(DatabaseStore.find).not.toHaveBeenCalled()
|
|
{}
|
|
, (model) ->
|
|
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
expect(thread).toBe(fakeThread)
|
|
expect(DatabaseStore.find).toHaveBeenCalled()
|
|
{}
|
|
, (model) ->
|
|
|
|
it "should accept `message` or look up a message when given `messageId`", ->
|
|
@_callNewMessageWithContext {thread: fakeThread, message: fakeMessage1}
|
|
, (thread, message) ->
|
|
expect(message).toBe(fakeMessage1)
|
|
expect(DatabaseStore.find).not.toHaveBeenCalled()
|
|
{}
|
|
, (model) ->
|
|
|
|
@_callNewMessageWithContext {thread: fakeThread, messageId: fakeMessage1.id}
|
|
, (thread, message) ->
|
|
expect(message).toBe(fakeMessage1)
|
|
expect(DatabaseStore.find).toHaveBeenCalled()
|
|
{}
|
|
, (model) ->
|
|
|
|
|
|
describe "when a reply-to message is provided by the attributesCallback", ->
|
|
it "should include quoted text in the new message", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
replyToMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.body.indexOf('gmail_quote') > 0).toBe(true)
|
|
expect(model.body.indexOf('Fake Message 1') > 0).toBe(true)
|
|
|
|
it "should include the `On ... wrote:` line", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
replyToMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.body.search(/On .+, at .+, Customer <customer@example.com> wrote/) > 0).toBe(true)
|
|
|
|
it "should make the subject the subject of the message, not the thread", ->
|
|
fakeMessage1.subject = "OLD SUBJECT"
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
replyToMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.subject).toEqual("Re: OLD SUBJECT")
|
|
|
|
it "should change the subject from Fwd: back to Re: if necessary", ->
|
|
fakeMessage1.subject = "Fwd: This is my DRAFT"
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
replyToMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.subject).toEqual("Re: This is my DRAFT")
|
|
|
|
it "should only include the sender's name if it was available", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
replyToMessage: fakeMessage2
|
|
, (model) ->
|
|
expect(model.body.search(/On .+, at .+, ben@nylas.com wrote:/) > 0).toBe(true)
|
|
|
|
describe "when a forward message is provided by the attributesCallback", ->
|
|
it "should include quoted text in the new message", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
forwardMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.body.indexOf('gmail_quote') > 0).toBe(true)
|
|
expect(model.body.indexOf('Fake Message 1') > 0).toBe(true)
|
|
|
|
it "should include the `Begin forwarded message:` line", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
forwardMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.body.indexOf('Begin forwarded message') > 0).toBe(true)
|
|
|
|
it "should make the subject the subject of the message, not the thread", ->
|
|
fakeMessage1.subject = "OLD SUBJECT"
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
forwardMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.subject).toEqual("Fwd: OLD SUBJECT")
|
|
|
|
it "should change the subject from Re: back to Fwd: if necessary", ->
|
|
fakeMessage1.subject = "Re: This is my DRAFT"
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
forwardMessage: fakeMessage1
|
|
, (model) ->
|
|
expect(model.subject).toEqual("Fwd: This is my DRAFT")
|
|
|
|
it "should print the headers of the original message", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
forwardMessage: fakeMessage2
|
|
, (model) ->
|
|
expect(model.body.indexOf('From: ben@nylas.com') > 0).toBe(true)
|
|
expect(model.body.indexOf('Subject: Re: Fake Subject') > 0).toBe(true)
|
|
expect(model.body.indexOf('To: customer@example.com') > 0).toBe(true)
|
|
|
|
describe "attributesCallback", ->
|
|
describe "when a threadId is provided", ->
|
|
it "should receive the thread", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
expect(thread).toEqual(fakeThread)
|
|
{}
|
|
|
|
it "should receive the last message in the fakeThread", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id}
|
|
, (thread, message) ->
|
|
expect(message).toEqual(fakeMessage2)
|
|
{}
|
|
|
|
describe "when a threadId and messageId are provided", ->
|
|
it "should receive the thread", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id, messageId: fakeMessage1.id}
|
|
, (thread, message) ->
|
|
expect(thread).toEqual(fakeThread)
|
|
{}
|
|
|
|
it "should receive the desired message in the thread", ->
|
|
@_callNewMessageWithContext {threadId: fakeThread.id, messageId: fakeMessage1.id}
|
|
, (thread, message) ->
|
|
expect(message).toEqual(fakeMessage1)
|
|
{}
|
|
|
|
describe "sanitizing draft bodies", ->
|
|
it "should transform inline styles and sanitize unsafe html", ->
|
|
spyOn(InlineStyleTransformer, 'run').andCallFake (input) => Promise.resolve(input)
|
|
spyOn(SanitizeTransformer, 'run').andCallFake (input) => Promise.resolve(input)
|
|
|
|
input = "test 123"
|
|
DraftStore._prepareBodyForQuoting(input)
|
|
expect(InlineStyleTransformer.run).toHaveBeenCalledWith(input)
|
|
advanceClock()
|
|
expect(SanitizeTransformer.run).toHaveBeenCalledWith(input, SanitizeTransformer.Preset.UnsafeOnly)
|
|
|
|
describe "onDestroyDraft", ->
|
|
beforeEach ->
|
|
@draftSessionTeardown = jasmine.createSpy('draft teardown')
|
|
@session =
|
|
draft: ->
|
|
pristine: false
|
|
changes:
|
|
commit: -> Promise.resolve()
|
|
teardown: ->
|
|
teardown: @draftSessionTeardown
|
|
DraftStore._draftSessions = {"abc": @session}
|
|
spyOn(Actions, 'queueTask')
|
|
|
|
it "should teardown the draft session, ensuring no more saves are made", ->
|
|
DraftStore._onDestroyDraft('abc')
|
|
expect(@draftSessionTeardown).toHaveBeenCalled()
|
|
|
|
it "should not throw if the draft session is not in the window", ->
|
|
expect ->
|
|
DraftStore._onDestroyDraft('other')
|
|
.not.toThrow()
|
|
|
|
it "should queue a destroy draft task", ->
|
|
DraftStore._onDestroyDraft('abc')
|
|
expect(Actions.queueTask).toHaveBeenCalled()
|
|
expect(Actions.queueTask.mostRecentCall.args[0] instanceof DestroyDraftTask).toBe(true)
|
|
|
|
it "should clean up the draft session", ->
|
|
spyOn(DraftStore, '_doneWithSession')
|
|
DraftStore._onDestroyDraft('abc')
|
|
expect(DraftStore._doneWithSession).toHaveBeenCalledWith(@session)
|
|
|
|
it "should close the window if it's a popout", ->
|
|
spyOn(NylasEnv, "close")
|
|
spyOn(DraftStore, "_isPopout").andReturn true
|
|
DraftStore._onDestroyDraft('abc')
|
|
expect(NylasEnv.close).toHaveBeenCalled()
|
|
|
|
it "should NOT close the window if isn't a popout", ->
|
|
spyOn(NylasEnv, "close")
|
|
spyOn(DraftStore, "_isPopout").andReturn false
|
|
DraftStore._onDestroyDraft('abc')
|
|
expect(NylasEnv.close).not.toHaveBeenCalled()
|
|
|
|
describe "before unloading", ->
|
|
it "should destroy pristine drafts", ->
|
|
DraftStore._draftSessions = {"abc": {
|
|
changes: {}
|
|
draft: ->
|
|
pristine: true
|
|
}}
|
|
|
|
spyOn(Actions, 'queueTask')
|
|
DraftStore._onBeforeUnload()
|
|
expect(Actions.queueTask).toHaveBeenCalled()
|
|
expect(Actions.queueTask.mostRecentCall.args[0] instanceof DestroyDraftTask).toBe(true)
|
|
|
|
describe "when drafts return unresolved commit promises", ->
|
|
beforeEach ->
|
|
@resolve = null
|
|
DraftStore._draftSessions = {"abc": {
|
|
changes:
|
|
commit: => new Promise (resolve, reject) => @resolve = resolve
|
|
draft: ->
|
|
pristine: false
|
|
}}
|
|
|
|
it "should return false and call window.close itself", ->
|
|
spyOn(NylasEnv, 'finishUnload')
|
|
expect(DraftStore._onBeforeUnload()).toBe(false)
|
|
expect(NylasEnv.finishUnload).not.toHaveBeenCalled()
|
|
@resolve()
|
|
advanceClock(1000)
|
|
expect(NylasEnv.finishUnload).toHaveBeenCalled()
|
|
|
|
describe "when drafts return immediately fulfilled commit promises", ->
|
|
beforeEach ->
|
|
DraftStore._draftSessions = {"abc": {
|
|
changes:
|
|
commit: => Promise.resolve()
|
|
draft: ->
|
|
pristine: false
|
|
}}
|
|
|
|
it "should still wait one tick before firing NylasEnv.close again", ->
|
|
spyOn(NylasEnv, 'finishUnload')
|
|
expect(DraftStore._onBeforeUnload()).toBe(false)
|
|
expect(NylasEnv.finishUnload).not.toHaveBeenCalled()
|
|
advanceClock()
|
|
expect(NylasEnv.finishUnload).toHaveBeenCalled()
|
|
|
|
describe "when there are no drafts", ->
|
|
beforeEach ->
|
|
DraftStore._draftSessions = {}
|
|
|
|
it "should return true and allow the window to close", ->
|
|
expect(DraftStore._onBeforeUnload()).toBe(true)
|
|
|
|
describe "sending a draft", ->
|
|
draftClientId = "local-123"
|
|
beforeEach ->
|
|
DraftStore._draftSessions = {}
|
|
DraftStore._draftsSending = {}
|
|
@forceCommit = false
|
|
proxy =
|
|
prepare: -> Promise.resolve(proxy)
|
|
teardown: ->
|
|
draft: -> {}
|
|
changes:
|
|
commit: ({force}={}) =>
|
|
@forceCommit = force
|
|
Promise.resolve()
|
|
DraftStore._draftSessions[draftClientId] = proxy
|
|
spyOn(DraftStore, "_doneWithSession").andCallThrough()
|
|
spyOn(DraftStore, "trigger")
|
|
spyOn(SoundRegistry, "playSound")
|
|
|
|
it "plays a sound immediately when sending draft", ->
|
|
spyOn(NylasEnv.config, "get").andReturn true
|
|
DraftStore._onSendDraft(draftClientId)
|
|
expect(NylasEnv.config.get).toHaveBeenCalledWith("core.sending.sounds")
|
|
expect(SoundRegistry.playSound).toHaveBeenCalledWith("hit-send")
|
|
|
|
it "doesn't plays a sound if the setting is off", ->
|
|
spyOn(NylasEnv.config, "get").andReturn false
|
|
DraftStore._onSendDraft(draftClientId)
|
|
expect(NylasEnv.config.get).toHaveBeenCalledWith("core.sending.sounds")
|
|
expect(SoundRegistry.playSound).not.toHaveBeenCalled()
|
|
|
|
it "sets the sending state when sending", ->
|
|
spyOn(NylasEnv, "isMainWindow").andReturn true
|
|
spyOn(Actions, "queueTask").andCallThrough()
|
|
runs ->
|
|
DraftStore._onSendDraft(draftClientId)
|
|
waitsFor ->
|
|
Actions.queueTask.calls.length > 0
|
|
runs ->
|
|
expect(DraftStore.isSendingDraft(draftClientId)).toBe true
|
|
|
|
# Since all changes haven't been applied yet, we want to ensure that
|
|
# no view of the draft renders the draft as if its sending, but with
|
|
# the wrong text.
|
|
it "does NOT trigger until the latest changes have been applied", ->
|
|
spyOn(NylasEnv, "isMainWindow").andReturn true
|
|
spyOn(Actions, "queueTask").andCallThrough()
|
|
runs ->
|
|
DraftStore._onSendDraft(draftClientId)
|
|
expect(DraftStore.trigger).not.toHaveBeenCalled()
|
|
waitsFor ->
|
|
Actions.queueTask.calls.length > 0
|
|
runs ->
|
|
# Normally, the session.changes.commit will persist to the
|
|
# Database. Since that's stubbed out, we need to manually invoke
|
|
# to database update event to get the trigger (which we want to
|
|
# test) to fire
|
|
DraftStore._onDataChanged
|
|
objectClass: "Message"
|
|
objects: [draft: true]
|
|
expect(DraftStore.isSendingDraft(draftClientId)).toBe true
|
|
expect(DraftStore.trigger).toHaveBeenCalled()
|
|
expect(DraftStore.trigger.calls.length).toBe 1
|
|
|
|
it "returns false if the draft hasn't been seen", ->
|
|
spyOn(NylasEnv, "isMainWindow").andReturn true
|
|
expect(DraftStore.isSendingDraft(draftClientId)).toBe false
|
|
|
|
it "closes the window if it's a popout", ->
|
|
spyOn(NylasEnv, "getWindowType").andReturn "composer"
|
|
spyOn(NylasEnv, "isMainWindow").andReturn false
|
|
spyOn(NylasEnv, "close")
|
|
runs ->
|
|
DraftStore._onSendDraft(draftClientId)
|
|
waitsFor "N1 to close", ->
|
|
NylasEnv.close.calls.length > 0
|
|
|
|
it "doesn't close the window if it's inline", ->
|
|
spyOn(NylasEnv, "getWindowType").andReturn "other"
|
|
spyOn(NylasEnv, "isMainWindow").andReturn false
|
|
spyOn(NylasEnv, "close")
|
|
spyOn(DraftStore, "_isPopout").andCallThrough()
|
|
runs ->
|
|
DraftStore._onSendDraft(draftClientId)
|
|
waitsFor ->
|
|
DraftStore._isPopout.calls.length > 0
|
|
runs ->
|
|
expect(NylasEnv.close).not.toHaveBeenCalled()
|
|
|
|
it "forces a commit to happen before sending", ->
|
|
spyOn(Actions, "queueTask")
|
|
runs ->
|
|
DraftStore._onSendDraft(draftClientId)
|
|
waitsFor ->
|
|
DraftStore._doneWithSession.calls.length > 0
|
|
runs ->
|
|
expect(@forceCommit).toBe true
|
|
|
|
it "queues a SendDraftTask", ->
|
|
spyOn(Actions, "queueTask")
|
|
runs ->
|
|
DraftStore._onSendDraft(draftClientId)
|
|
waitsFor ->
|
|
DraftStore._doneWithSession.calls.length > 0
|
|
runs ->
|
|
expect(Actions.queueTask).toHaveBeenCalled()
|
|
task = Actions.queueTask.calls[0].args[0]
|
|
expect(task instanceof SendDraftTask).toBe true
|
|
expect(task.draftClientId).toBe draftClientId
|
|
expect(task.fromPopout).toBe false
|
|
|
|
it "queues a SendDraftTask with popout info", ->
|
|
spyOn(NylasEnv, "getWindowType").andReturn "composer"
|
|
spyOn(NylasEnv, "isMainWindow").andReturn false
|
|
spyOn(NylasEnv, "close")
|
|
spyOn(Actions, "queueTask")
|
|
runs ->
|
|
DraftStore._onSendDraft(draftClientId)
|
|
waitsFor ->
|
|
DraftStore._doneWithSession.calls.length > 0
|
|
runs ->
|
|
expect(Actions.queueTask).toHaveBeenCalled()
|
|
task = Actions.queueTask.calls[0].args[0]
|
|
expect(task.fromPopout).toBe true
|
|
|
|
it "resets the sending state if there's an error", ->
|
|
spyOn(NylasEnv, "isMainWindow").andReturn false
|
|
DraftStore._draftsSending[draftClientId] = true
|
|
Actions.draftSendingFailed({errorMessage: "boohoo", draftClientId})
|
|
expect(DraftStore.isSendingDraft(draftClientId)).toBe false
|
|
expect(DraftStore.trigger).toHaveBeenCalledWith(draftClientId)
|
|
|
|
it "displays a popup in the main window if there's an error", ->
|
|
spyOn(NylasEnv, "isMainWindow").andReturn true
|
|
remote = require('remote')
|
|
dialog = remote.require('dialog')
|
|
spyOn(dialog, "showMessageBox")
|
|
DraftStore._draftsSending[draftClientId] = true
|
|
Actions.draftSendingFailed({errorMessage: "boohoo", draftClientId})
|
|
advanceClock(200)
|
|
expect(DraftStore.isSendingDraft(draftClientId)).toBe false
|
|
expect(DraftStore.trigger).toHaveBeenCalledWith(draftClientId)
|
|
expect(dialog.showMessageBox).toHaveBeenCalled()
|
|
dialogArgs = dialog.showMessageBox.mostRecentCall.args[1]
|
|
expect(dialogArgs.detail).toEqual("boohoo")
|
|
|
|
describe "session teardown", ->
|
|
beforeEach ->
|
|
spyOn(NylasEnv, 'isMainWindow').andReturn true
|
|
@draftTeardown = jasmine.createSpy('draft teardown')
|
|
@session =
|
|
draftClientId: "abc"
|
|
draft: ->
|
|
pristine: false
|
|
changes:
|
|
commit: -> Promise.resolve()
|
|
reset: ->
|
|
teardown: @draftTeardown
|
|
DraftStore._draftSessions = {"abc": @session}
|
|
DraftStore._doneWithSession(@session)
|
|
|
|
it "removes from the list of draftSessions", ->
|
|
expect(DraftStore._draftSessions["abc"]).toBeUndefined()
|
|
|
|
it "Calls teardown on the session", ->
|
|
expect(@draftTeardown).toHaveBeenCalled
|
|
|
|
describe "mailto handling", ->
|
|
beforeEach ->
|
|
spyOn(NylasEnv, 'isMainWindow').andReturn true
|
|
|
|
describe "extensions", ->
|
|
beforeEach ->
|
|
DraftStore.registerExtension(TestExtension)
|
|
afterEach ->
|
|
DraftStore.unregisterExtension(TestExtension)
|
|
|
|
it "should give extensions a chance to customize the draft via ext.prepareNewDraft", ->
|
|
received = null
|
|
spyOn(DatabaseStore, 'persistModel').andCallFake (draft) ->
|
|
received = draft
|
|
Promise.resolve()
|
|
waitsForPromise ->
|
|
DraftStore._onHandleMailtoLink({}, 'mailto:bengotow@gmail.com').then ->
|
|
expect(received.body.indexOf("Edited by TestExtension!")).toBe(0)
|
|
|
|
describe "when testing subject keys", ->
|
|
beforeEach ->
|
|
spyOn(DraftStore, '_finalizeAndPersistNewMessage').andCallFake (draft) ->
|
|
Promise.resolve({draftClientId: 123})
|
|
|
|
@expected = "EmailSubjectLOLOL"
|
|
|
|
it "works for lowercase", ->
|
|
waitsForPromise =>
|
|
DraftStore._onHandleMailtoLink({}, 'mailto:asdf@asdf.com?subject=' + @expected).then =>
|
|
received = DraftStore._finalizeAndPersistNewMessage.mostRecentCall.args[0]
|
|
expect(received.subject).toBe(@expected)
|
|
|
|
it "works for title case", ->
|
|
waitsForPromise =>
|
|
DraftStore._onHandleMailtoLink({}, 'mailto:asdf@asdf.com?Subject=' + @expected).then =>
|
|
received = DraftStore._finalizeAndPersistNewMessage.mostRecentCall.args[0]
|
|
expect(received.subject).toBe(@expected)
|
|
|
|
it "works for uppercase", ->
|
|
waitsForPromise =>
|
|
DraftStore._onHandleMailtoLink({}, 'mailto:asdf@asdf.com?SUBJECT=' + @expected).then =>
|
|
received = DraftStore._finalizeAndPersistNewMessage.mostRecentCall.args[0]
|
|
expect(received.subject).toBe(@expected)
|
|
|
|
describe "should correctly instantiate drafts for a wide range of mailto URLs", ->
|
|
beforeEach ->
|
|
spyOn(DatabaseStore, 'persistModel').andCallFake (draft) ->
|
|
Promise.resolve()
|
|
|
|
links = [
|
|
'mailto:'
|
|
'mailto://bengotow@gmail.com'
|
|
'mailto:bengotow@gmail.com'
|
|
'mailto:?subject=%1z2a', # fails uriDecode
|
|
'mailto:?subject=%52z2a', # passes uriDecode
|
|
'mailto:?subject=Martha Stewart',
|
|
'mailto:?subject=Martha Stewart&cc=cc@nylas.com',
|
|
'mailto:bengotow@gmail.com&subject=Martha Stewart&cc=cc@nylas.com',
|
|
'mailto:bengotow@gmail.com?subject=Martha%20Stewart&cc=cc@nylas.com&bcc=bcc@nylas.com',
|
|
'mailto:bengotow@gmail.com?subject=Martha%20Stewart&cc=cc@nylas.com&bcc=Ben <bcc@nylas.com>',
|
|
'mailto:Ben Gotow <bengotow@gmail.com>,Shawn <shawn@nylas.com>?subject=Yes this is really valid',
|
|
'mailto:Ben%20Gotow%20<bengotow@gmail.com>,Shawn%20<shawn@nylas.com>?subject=Yes%20this%20is%20really%20valid',
|
|
'mailto:Reply <d+AORGpRdj0KXKUPBE1LoI0a30F10Ahj3wu3olS-aDk5_7K5Wu6WqqqG8t1HxxhlZ4KEEw3WmrSdtobgUq57SkwsYAH6tG57IrNqcQR0K6XaqLM2nGNZ22D2k@docs.google.com>?subject=Nilas%20Message%20to%20Customers',
|
|
'mailto:email@address.com?&subject=test&body=type%20your%0Amessage%20here'
|
|
'mailto:?body=type%20your%0D%0Amessage%0D%0Ahere'
|
|
]
|
|
expected = [
|
|
new Message(),
|
|
new Message(
|
|
to: [new Contact(name: 'bengotow@gmail.com', email: 'bengotow@gmail.com')]
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'bengotow@gmail.com', email: 'bengotow@gmail.com')]
|
|
),
|
|
new Message(
|
|
subject: '%1z2a'
|
|
),
|
|
new Message(
|
|
subject: 'Rz2a'
|
|
),
|
|
new Message(
|
|
subject: 'Martha Stewart'
|
|
),
|
|
new Message(
|
|
cc: [new Contact(name: 'cc@nylas.com', email: 'cc@nylas.com')],
|
|
subject: 'Martha Stewart'
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'bengotow@gmail.com', email: 'bengotow@gmail.com')],
|
|
cc: [new Contact(name: 'cc@nylas.com', email: 'cc@nylas.com')],
|
|
subject: 'Martha Stewart'
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'bengotow@gmail.com', email: 'bengotow@gmail.com')],
|
|
cc: [new Contact(name: 'cc@nylas.com', email: 'cc@nylas.com')],
|
|
bcc: [new Contact(name: 'bcc@nylas.com', email: 'bcc@nylas.com')],
|
|
subject: 'Martha Stewart'
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'bengotow@gmail.com', email: 'bengotow@gmail.com')],
|
|
cc: [new Contact(name: 'cc@nylas.com', email: 'cc@nylas.com')],
|
|
bcc: [new Contact(name: 'Ben', email: 'bcc@nylas.com')],
|
|
subject: 'Martha Stewart'
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'Ben Gotow', email: 'bengotow@gmail.com'), new Contact(name: 'Shawn', email: 'shawn@nylas.com')],
|
|
subject: 'Yes this is really valid'
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'Ben Gotow', email: 'bengotow@gmail.com'), new Contact(name: 'Shawn', email: 'shawn@nylas.com')],
|
|
subject: 'Yes this is really valid'
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'Reply', email: 'd+AORGpRdj0KXKUPBE1LoI0a30F10Ahj3wu3olS-aDk5_7K5Wu6WqqqG8t1HxxhlZ4KEEw3WmrSdtobgUq57SkwsYAH6tG57IrNqcQR0K6XaqLM2nGNZ22D2k@docs.google.com')],
|
|
subject: 'Nilas Message to Customers'
|
|
),
|
|
new Message(
|
|
to: [new Contact(name: 'email@address.com', email: 'email@address.com')],
|
|
subject: 'test'
|
|
body: 'type your\nmessage here'
|
|
),
|
|
new Message(
|
|
to: [],
|
|
body: 'type your\r\nmessage\r\nhere'
|
|
)
|
|
]
|
|
|
|
links.forEach (link, idx) ->
|
|
it "works for #{link}", ->
|
|
waitsForPromise ->
|
|
DraftStore._onHandleMailtoLink({}, link).then ->
|
|
expectedDraft = expected[idx]
|
|
received = DatabaseStore.persistModel.mostRecentCall.args[0]
|
|
expect(received['subject']).toEqual(expectedDraft['subject'])
|
|
expect(received['body']).toEqual(expectedDraft['body']) if expectedDraft['body']
|
|
for attr in ['to', 'cc', 'bcc']
|
|
for contact, jdx in received[attr]
|
|
expectedContact = expectedDraft[attr][jdx]
|
|
expect(contact.email).toEqual(expectedContact.email)
|
|
expect(contact.name).toEqual(expectedContact.name)
|