-- This file is part of purebred -- Copyright (C) 2017-2019 Róman Joost -- -- purebred is free software: you can redistribute it and/or modify -- it under the terms of the GNU Affero General Public License as published by -- the Free Software Foundation, either version 3 of the License, or -- (at your option) any later version. -- -- This program is distributed in the hope that it will be useful, -- but WITHOUT ANY WARRANTY; without even the implied warranty of -- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -- GNU Affero General Public License for more details. -- -- You should have received a copy of the GNU Affero General Public License -- along with this program. If not, see . {-# OPTIONS_GHC -fno-warn-unused-do-bind -fno-warn-missing-signatures #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE NoMonomorphismRestriction #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE RankNTypes #-} module Main where import Data.Char (chr) import System.IO.Temp ( createTempDirectory, getCanonicalTemporaryDirectory , emptySystemTempFile) import Data.Either (isRight) import Data.Functor (($>)) import Data.Foldable (for_) import Control.Concurrent (threadDelay) import System.IO (hPutStr, stderr) import System.Environment (lookupEnv, getEnvironment) import qualified System.Environment as Env import System.FilePath.Posix ( () , getSearchPath, isAbsolute, searchPathSeparator ) import Control.Monad (filterM, void, when) import Data.Maybe (fromMaybe, isJust) import Data.List (intercalate, isInfixOf, sort, sortBy) import qualified Data.ByteString.Char8 as B import Data.ByteString.Builder (toLazyByteString) import qualified Data.Text as T import qualified Data.Text.Encoding as T import Control.Monad.IO.Class (liftIO) import Control.Monad.Reader (MonadIO, MonadReader, runReaderT) import Control.Monad.State (MonadState) import System.Exit (die) import Control.Lens (Lens', _init, _last, at, lens, preview, set, to, view) import System.Directory ( copyFile, getCurrentDirectory, listDirectory, removeDirectoryRecursive , removeFile, doesPathExist, findExecutable ) import System.Posix.Files (getFileStatus, isRegularFile) import System.Process.Typed (byteStringInput, proc, readProcess_, runProcess_, setEnv, setStdin) import Test.Tasty (defaultMain) import Test.Tasty.HUnit (assertBool, assertEqual, assertFailure) import Test.Tasty.Tmux import Data.MIME (MIMEMessage, createTextPlainMessage, message, mime, parse, headers, buildMessage) {-# ANN module ("HLint: ignore Reduce duplication" :: String) #-} type PurebredTestCase = TestCase GlobalEnv main :: IO () main = do for_ ["purebred", "elinks"] $ \prog -> findExecutable prog >>= maybe (die $ "missing program: " <> prog) (\_ -> pure ()) Env.setEnv tastyNumThreadsEnv . fromMaybe "20" =<< lookupEnv tastyNumThreadsEnv defaultMain $ testTmux pre post tests where tastyNumThreadsEnv = "TASTY_NUM_THREADS" pre = do dir <- mkTempDir setUpPurebredConfig dir precompileConfig dir -- all tests can use same precompiled binary pure (GlobalEnv dir) post (GlobalEnv dir) = removeDirectoryRecursive dir tests = [ testUserViewsMailSuccessfully , testUserCanManipulateNMQuery , testUserCanSwitchBackToIndex , testUserCanAbortMailComposition , testSendMail , testSendFailureHandling , testCanToggleHeaders , testSetsMailToRead , testShowsAndClearsError , testHelp , testManageTagsOnMails , testManageTagsOnThreads , testConfig , testUpdatesReadState , testCanJumpToFirstListItem , testAddAttachments , testFileBrowserInvalidPath , testFromAddressIsProperlyReset , testRepliesToMailSuccessfully , testUserCanMoveBetweenThreads , testShowsMailEntities , testOpenCommandDoesNotKillPurebred , testOpenEntitiesSuccessfully , testPipeEntitiesSuccessfully , testEditingMailHeaders , testShowsInvalidCompositionInput , testShowsInvalidTaggingInput , testKeepDraftMail , testDiscardsMail , testShowsNewMail , testConfirmDialogResets , testCursorPositionedEndOnReply , testSubstringSearchInMailBody , testSubstringMatchesAreCleared , testAutoview , testSavesEntitySuccessfully , testForwardsMailSuccessfully , testBulkActionsOnThreadsByKeybinding , testBulkActionsOnThreadsByInput , testBulkActionsOnMailsByInput , testAbortedEditsResetState , testReloadsThreadListAfterReply , testAbortsCompositionIfEditorExits , testSearchRelated , testReplyRendersNonASCIIHeadersCorrectly , testGroupReply ] testSearchRelated :: PurebredTestCase testSearchRelated = purebredTmuxSession "searches related" $ \step -> do startApplication capture >>= put assertSubstringS "Item 1 of 4" assertRegexS (buildAnsiRegex ["37"] ["43"] [] <> "[[:space:]]Aug'17 frase@host.exa") step "search related" sendKeys "+" (Substring "Item 1 of 2") >>= put assertRegexS ("Query:[[:space:]]" <> buildAnsiRegex ["34"] [] [] <> "frase@host.example") -- https://github.com/purebred-mua/purebred/issues/336 testAbortsCompositionIfEditorExits :: PurebredTestCase testAbortsCompositionIfEditorExits = purebredTmuxSession "aborts composition if editor exits abnormally" $ \step -> do setEnvVarInSession "EDITOR" "doesnotexistFoo" startApplication step "start composition" sendKeys "m" (Substring "From") step "accept default" sendKeys "Enter" (Substring "To") step "enter to: email" sendLine "user@to.test" (Substring "Subject") step "enter subject" sendLine "Draft mail subject" (Substring "Editor exited abnormally") -- check reply step "Navigate second mail" sendKeys "Down" (Substring "Item 2 of 4") step "View mail" sendKeys "Enter" (Substring "HOLY PUREBRED") step "reply to mail" sendKeys "r" (Substring "Editor exited abnormally") -- mail composition from mail view step "start composition from mail view" sendKeys "f" (Substring "To:") step "accept default" sendKeys "Enter" (Substring "Editor exited abnormally") step "back to thread list" sendKeys "Escape" (Substring "Item 2 of 4") -- https://github.com/purebred-mua/purebred/issues/395 testReloadsThreadListAfterReply :: PurebredTestCase testReloadsThreadListAfterReply = purebredTmuxSession "reloads list of threads" $ \step -> do startApplication step "focus query editor" sendKeys ":" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "tag:inbox")) sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+")) step "query for receiver" sendLine "to:frase" (Substring "Item 1 of 2") step "view mail" sendKeys "Enter" (Substring "This is a test mail for purebred") step "compose as new" sendKeys "e" (Substring "Testmail with whitespace in the subject") step "navigate to latest attachment" sendKeys "Down" (Substring "Item 2 of 2") >>= put assertSubstringS "text/html" step "Remove HTML part" sendKeys "D" (Not (Substring "text/html")) >>= put assertSubstringS "Item 1 of 1" step "send mail" sendKeys "y" (Substring "Item 1 of 2") step "check current first mail" sendKeys "Enter" (Substring "This is a test mail for purebred") >>= put assertSubstringS "Date: Thu, 17 Aug" step "update list of threads" sendKeys "Escape" (Substring "Item 1 of 2") sendKeys ":" Unconditional sendKeys "Enter" (Substring "Item 1 of 3") step "open first mail" sendKeys "Enter" (Substring "This is a test mail for purebred") testAbortedEditsResetState :: PurebredTestCase testAbortedEditsResetState = purebredTmuxSession "aborted edits reset editor back to initial state" $ \step -> do startApplication step "edit search query" assertEditorResetsToInitialValue step ":" ("Query: " <> buildAnsiRegex [] ["37"] [] <> "tag:inbox") ("Query: " <> buildAnsiRegex [] ["34"] [] <> "tag:inbox") composeNewMail step step "edit Subject: field" assertEditorResetsToInitialValue step "s" ("Subject: " <> buildAnsiRegex [] ["37"] [] <> "Draft mail subject") "Subject: Draft mail subject" step "edit BCC: field" assertEditorResetsToInitialValue step "b" ("Bcc: " <> buildAnsiRegex [] ["37"] []) "Bcc:" step "edit CC: field" assertEditorResetsToInitialValue step "c" ("Cc: " <> buildAnsiRegex [] ["37"] []) "Cc:" step "edit From: field" assertEditorResetsToInitialValue step "f" ("From: " <> buildAnsiRegex [] ["37"] [] <> "\"Joe Bloggs\" ") "From: \"Joe Bloggs\" " step "edit To: field" assertEditorResetsToInitialValue step "t" ("To: " <> buildAnsiRegex [] ["37"] [] <> "user@to.test") "To: user@to.test" testBulkActionsOnMailsByInput :: PurebredTestCase testBulkActionsOnMailsByInput = purebredTmuxSession "perform bulk labeling on mails by editor" $ \step -> do startApplication step "navigate to thread with two mails" sendKeys "Down" (Substring "Item 2 of 4") sendKeys "Down" (Substring "Item 3 of 4") sendKeys "Enter" (Substring "Lorem ipsum dolor sit amet") step "toggle first mail" sendKeys "*" (Regex $ selectedListItem <> "[[:space:]]Feb'17.*WIP Refactor") step "toggle second mail" -- toggled *and* currently selected item sendKeys "*" (Regex $ buildAnsiRegex [] [] ["43"] <> "[[:space:]]Feb'17.*Re: WIP Refactor") step "open mail tag editor" sendKeys "`" (Regex ("Labels:." <> buildAnsiRegex [] ["37"] [])) >>= put -- guard against a case in which mails are already tagged with our test tag assertConditionS (Not (Substring "testTag")) step "add new tag" sendLine "+testTag" ( Regex ( "testTag" -- first list item starting with the tag <> buildAnsiRegex [] ["34"] [] <> "[[:space:]]+WIP Refactor[[:space:]]+\n" -- next mail/list item <> buildAnsiRegex [] ["37"] ["43"] <> "[[:space:]]+Feb'17.*testTag" <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+Re: WIP Refactor" ) ) >>= put -- Editor is not displayed any more assertConditionS (Not (Substring "Labels:")) -- Every toggled list item is now untoggled assertConditionS (Not (Substring "Marked")) testBulkActionsOnThreadsByInput :: PurebredTestCase testBulkActionsOnThreadsByInput = purebredTmuxSession "perform bulk labeling on threads by editor" $ \step -> do startApplication step "Toggle two thread items" sendKeys "*" (Regex $ toggledListItem <> "[[:space:]]Aug.*Testmail with whitespace in the subject") -- The previous line has the same colour, so start colour matching from the first line sendKeys "*" (Regex $ toggledListItem <> "[[:space:]]Aug.*in the subject.*[[:space:]][[:space:]]Aug.*This is Purebred") step "open thread tag editor" sendKeys "`" (Regex ("Labels:." <> buildAnsiRegex [] ["37"] [])) >>= put -- guard against a case in which mails are already tagged with our test tag assertConditionS (Not (Substring "testTag")) step "add tag" sendLine "+testTag" ( Regex ( "testTag" <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+Testmail with whitespace in the subject[[:space:]]*\n" <> buildAnsiRegex [] ["34"] [] <> "[[:space:]]+Aug'17.*testTag" <> buildAnsiRegex [] ["34"] [] <> "[[:space:]]+This is Purebred[[:space:]]+\n" ) ) testBulkActionsOnThreadsByKeybinding :: PurebredTestCase testBulkActionsOnThreadsByKeybinding = purebredTmuxSession "perform bulk labeling on threads by keybinding" $ \step -> do startApplication step "Toggle thread and list cursor moves to next list item" sendKeys "*" (Regex $ toggledListItem <> "[[:space:]]Aug'17.*whitespace in the subject[[:space:]]+\n") >>= assertRegex ( -- current selection buildAnsiRegex [] ["30"] ["43"] <> "[[:space:]]Aug'17 rjoost@url.use.*This is Purebred[[:space:]]+\n" -- unselected rest <> newListItem <> "[[:space:]]Feb'17.*WIP Refactor" ) step "Toggle thread and list cursor moves to next list item" sendKeys "*" (Regex $ selectedListItem <> "[[:space:]]Feb'17.*WIP Refactor") step "Tag toggled list items using key binding" sendKeys "a" (Substring "New: 3 ]") -- untoggled >>= assertRegex (buildAnsiRegex [] ["37"] [] <> "[[:space:]]Aug'17.*whitespace in the subject[[:space:]]+\n") testForwardsMailSuccessfully :: PurebredTestCase testForwardsMailSuccessfully = purebredTmuxSession "forwards mail successfully" $ \step -> do startApplication let subject = "[frase@host.example: Testmail with whitespace in the subject]" step "view mail" sendKeys "Enter" (Substring "This is a test mail") step "Start forwarding composition" sendKeys "f" (Regex $ "To: " <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+$") step "enter receipient address" sendLine "to_user@foo.test" (Substring "~") step "enter mail body" sendKeys "iFind attached a forwarded mail" (Substring "mail") step "exit insert mode in vim" sendKeys "Escape" (Substring "mail") step "exit vim" sendKeys ": x\r" (Substring "Attachments") step "start editing cc" sendKeys "c" (Substring "Cc") step "add cc email" sendKeys "cc_user@foo.test\r" (Substring "Cc: cc_user@foo.test") step "start editing bcc" sendKeys "b" (Regex $ "Bcc: " <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+$") step "add bcc email" sendKeys "bcc_user@foo.test\r" (Substring "Bcc: bcc_user@foo.test") >>= put assertRegexS "From: \"Joe Bloggs\" [[:space:]]+$" assertSubstringS "To: to_user@foo.test" assertSubstringS "Cc: cc_user@foo.test" assertSubstringS "Bcc: bcc_user@foo.test" assertSubstringS ("Subject: " <> subject) assertSubstringS "text/plain" assertSubstringS "message/rfc822" step "send mail" sendKeys "y" (Substring "Query") testdir <- view envConfigDir let fpath = testdir "sentMail" assertMailSuccessfullyParsed fpath contents <- liftIO $ B.readFile fpath let decoded = chr . fromEnum <$> B.unpack contents assertSubstr subject decoded assertSubstr "This is a test mail" decoded assertSubstr "Find attached a forwarded mail" decoded testSavesEntitySuccessfully :: PurebredTestCase testSavesEntitySuccessfully = purebredTmuxSession "saves entity to disk successfully" $ \step -> do startApplication let mailbody = "This is a test mail for purebred" bogusSavePath = "/tmp/this/path/should/not/exist" -- Better check and abort the test if our made up path really does -- exist however unlikely we think it is. liftIO $ assertBool "expected bogus path to not exist" <$> doesPathExist bogusSavePath tmpfile <- liftIO $ emptySystemTempFile "purebred_saves_entity_to_disk_successfully" step "show current mail body" sendKeys "Enter" (Substring mailbody) step "list attachments" sendKeys "v" (Substring "text/plain; charset=utf-8") step "show save to disk editor" sendKeys "s" (Substring "Save to file") step "enter (wrong) path" sendLine bogusSavePath (Regex "(open|with)BinaryFile: does not exist") step "show save to disk editor (again)" sendKeys "s" (Regex $ "Save to file:[[:space:]]+" <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+") step "enter (correct) path" sendLine tmpfile (Substring "Attachment saved") snapshot assertConditionS (Not (Substring "Save to file")) contents <- liftIO $ B.readFile tmpfile let decoded = chr . fromEnum <$> B.unpack contents assertSubstr mailbody decoded testAutoview :: PurebredTestCase testAutoview = purebredTmuxSession "automatically copies output for display" $ \step -> do startApplication step "search for HTML mail" findMail step "subject:\"HTML mail\"" step "open HTML mail" sendKeys "Enter" (Substring "This is a HTML mail for purebred in which the HTML part contains") step "use as reply" sendKeys "r" (Regex ">[[:space:]]+This is a HTML mail for purebred") testSubstringMatchesAreCleared :: PurebredTestCase testSubstringMatchesAreCleared = purebredTmuxSession "substring match indicator only shown on mail" $ \step -> do startApplication step "No match indicator is shown" snapshot assertRegexS "New:[[:space:]][0-9][[:space:]]+\\][[:space:]]+Threads" step "search for Lorem mail" sendKeys ":" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "tag:inbox")) sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+")) step "enter free text search" sendLine "Lorem ipsum" (Substring "Item 1 of 1") step "show mail contents" sendKeys "Enter" (Substring "Lorem ipsum dolor sit amet, consectetur") step "show substring search editor" sendKeys "/" (Substring "Search for") step "enter needle and show results" sendKeys "et\r" (Substring "1 of 20 matches") step "begin substring search" sendKeys "/" (Substring "Search for") step "enter empty search string (reset search)" sendKeys "\r" (Not (Substring "matches ]")) step "go back to threads" sendKeys "Escape" (Regex "New:[[:space:]][0-9][[:space:]]+\\][[:space:]]+Threads") testSubstringSearchInMailBody :: PurebredTestCase testSubstringSearchInMailBody = purebredTmuxSession "search for substrings in mailbody" $ \step -> do startApplication step "search for Lorem mail" sendKeys ":" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "tag:inbox")) sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+")) step "enter free text search" sendLine "Lorem ipsum" (Substring "Item 1 of 1") step "show mail contents" sendKeys "Enter" (Substring "Lorem ipsum dolor sit amet, consectetur") step "show substring search editor" sendKeys "/" (Substring "Search for") step "enter needle and show results" sendKeys "et\r" (Regex ("am" <> buildAnsiRegex [] ["32"] ["47"] <> "et" <> buildAnsiRegex [] ["39"] ["49"] <> ", consect" <> buildAnsiRegex [] ["37"] ["42"] <> "et" <> buildAnsiRegex [] ["39"] ["49"] <> "ur")) step "highlight next search result" sendKeys "n" (Regex ("am" <> buildAnsiRegex [] ["37"] ["42"] <> "et" <> buildAnsiRegex [] ["39"] ["49"] <> ", consect" <> buildAnsiRegex [] ["32"] ["47"] <> "et" <> buildAnsiRegex [] ["39"] ["49"] <> "ur")) step "focus search input editor again" sendKeys "/" (Regex (buildAnsiRegex [] ["33"] [] <> "Search for:[[:space:]]" <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+$")) step "search for different needle" sendKeys "Lorem\r" (Regex ("\"" <> buildAnsiRegex [] ["32"] ["47"] <> "Lorem" <> buildAnsiRegex [] ["39"] ["49"] <> " ipsum")) step "clear all highlights" sendKeys "Enter" (Substring "Lorem ipsum dolor sit amet, consectetur") testCursorPositionedEndOnReply :: PurebredTestCase testCursorPositionedEndOnReply = purebredTmuxSession "cursor positioned on EOL when replying" $ \step -> do startApplication step "pick first mail" sendKeys "Enter" (Substring "This is a test mail for purebred") step "start replying" sendKeys "r" (Substring "> This is a test mail for purebred") step "exit vim" sendKeys ": x\r" (Substring "Attachments") step "focus from field" sendKeys "f" (Regex $ "From: " <> buildAnsiRegex [] ["37"] [] <> "\"Joe Bloggs\" ") sendKeys ", fromuser@foo.test\r" (Substring $ "From: " <> "\"Joe Bloggs\" , fromuser@foo.test") step "user can change to header" sendKeys "t" (Regex $ "To: " <> buildAnsiRegex [] ["37"] [] <> "frase@host.example") step "append an additional from email" sendKeys ", touser@foo.test\r" (Substring "To: frase@host.example, touser@foo.test") step "change subject" sendKeys "s" (Regex $ "Subject: " <> buildAnsiRegex [] ["37"] [] <> ".*subject[[:space:]]+$") step "enter subject" sendKeys " appended\r" (Substring "Subject: Re: Testmail with whitespace in the subject appended") testConfirmDialogResets :: PurebredTestCase testConfirmDialogResets = purebredTmuxSession "confirm dialog resets state" $ \step -> do startApplication composeNewMail step step "abort composition" sendKeys "q" (Substring "Keep draft?") step "choose Discard" sendKeys "Tab" (Substring "Discard") step "confirm discard" sendKeys "Enter" (Substring "Testmail") composeNewMail step step "abort composition" sendKeys "q" (Regex (buildAnsiRegex [] ["30"] ["42"] <> "[[:space:]]+Keep" )) -- Note: The most time in this test is spend on waiting. The default -- time for the indicator to refresh is 5 seconds. testShowsNewMail :: PurebredTestCase testShowsNewMail = purebredTmuxSession "shows newly delivered mail" $ \step -> do startApplication step "shows new mails" sendKeys "Down" (Substring "New: 4") notmuchcfg <- view envNotmuchConfig let m = set (headers . at "subject") (Just "new mail notification") $ createTextPlainMessage "Hello there" rendered = toLazyByteString (buildMessage m) config = setStdin (byteStringInput rendered) $ proc "notmuch" [ "--config=" <> notmuchcfg , "insert" , "--folder" , "tmp" , "--create-folder" ] void $ readProcess_ config step "shows new delivered mail" sendKeys "Up" (Substring "New: 5") -- reload mails to see the new e-mail step "focus query widget" sendKeys ":" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "tag:inbox")) step "view mail" sendKeys "Enter" (Substring "new mail notification") testShowsInvalidTaggingInput :: PurebredTestCase testShowsInvalidTaggingInput = purebredTmuxSession "shows errors when tagging" $ \step -> do startApplication step "start tagging" sendKeys "`" (Regex ("Labels: " <> buildAnsiRegex [] ["37"] [])) step "enter invalid tag input" sendKeys "=," (Substring "Failed reading: unexpected ',' at offset 1") step "clear" sendKeys "BSpace" (Regex ("Labels: " <> buildAnsiRegex [] ["37"] [] <> "=")) step "exit editor" sendKeys "C-g" (Substring "Query") step "open thread" sendKeys "Enter" (Substring "Testmail with whitespace") step "start tagging" sendKeys "`" (Regex ("Labels: " <> buildAnsiRegex [] ["37"] [])) step "enter invalid tag input" sendKeys "=," (Substring "Failed reading: unexpected ',' at offset 1") step "clear" sendKeys "BSpace" (Regex ("Labels: " <> buildAnsiRegex [] ["37"] [] <> "=")) testShowsInvalidCompositionInput :: PurebredTestCase testShowsInvalidCompositionInput = purebredTmuxSession "shows errors when composing" $ \step -> do startApplication step "start composition" sendKeys "m" (Substring "From") step "trigger error" sendKeys "<" (Substring "Failed reading") step "continue" sendKeys "BSpace" (Substring "Purebred: (0,27)") sendKeys "Enter" (Substring "To:") step "trigger error" sendKeys "," (Substring "Failed reading") step "continue" sendKeys "BSpace" (Substring "Purebred: (0,0)") sendKeys "Enter" (Substring "Subject:") step "leave empty subject" sendKeys "Enter" (Substring "~") step "enter mail body" sendKeys "iThis is a test body" (Substring "body") step "exit insert mode in vim" sendKeys "Escape" (Substring "body") step "exit vim" sendKeys ": x\r" (Substring "text/plain") step "focus from field" sendKeys "f" (Regex $ "From:[[:space:]]" <> buildAnsiRegex [] ["37"] [] <> "\"Joe Bloggs\" ") step "trigger error" sendKeys "," (Substring "Failed reading: unexpected ',' at offset 27") step "abort editing" sendKeys "C-g" (Substring "ComposeView-Attachments") step "focus to field" sendKeys "t" (Regex $ "To:[[:space:]]" <> buildAnsiRegex [] ["37"] []) step "trigger error" sendKeys "," (Substring "Failed reading") step "abort editing" sendKeys "C-g" (Substring "ComposeView-Attachments") testDiscardsMail :: PurebredTestCase testDiscardsMail = purebredTmuxSession "discards draft mail" $ \step -> do startApplication composeNewMail step step "abort composition" sendKeys "Escape" (Substring "Keep draft?") step "choose Discard" sendKeys "Tab" (Substring "Discard") step "confirm discard" sendKeys "Enter" (Substring "Testmail") step "no draft mail exists in Maildir" maildir <- view envMaildir assertFileAmountInMaildir (maildir "Drafts" "new") 0 testKeepDraftMail :: PurebredTestCase testKeepDraftMail = purebredTmuxSession "compose mail from draft" $ \step -> do startApplication composeNewMail step step "abort composition" sendKeys "q" (Substring "Keep draft?") step "confirm Keep" sendKeys "Enter" (Substring "Draft saved") step "assert draft exists" maildir <- view envMaildir assertFileAmountInMaildir (maildir "Drafts" "new") 1 step "search for draft" sendKeys ":" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "tag:inbox")) sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+")) step "enter new tag" sendLine "tag:draft" (Substring "Item 1 of 1") step "view mail" sendKeys "Enter" (Substring "Draft mail subject") step "edit as new" sendKeys "e" (Regex "From: \"Joe Bloggs\" ") >>= put assertSubstringS "To: user@to.test" assertSubstringS "Subject: Draft mail subject" step "assert draft has been removed" mdir <- view envMaildir assertFileAmountInMaildir (mdir "Drafts" "new") 0 step "send mail" sendKeys "y" (Substring "Query") testdir <- view envConfigDir let fpath = testdir "sentMail" contents <- liftIO $ B.readFile fpath let decoded = chr . fromEnum <$> B.unpack contents assertSubstr "This is a test body" decoded testEditingMailHeaders :: PurebredTestCase testEditingMailHeaders = purebredTmuxSession "user can edit mail headers" $ \step -> do startApplication step "start composition" sendKeys "m" (Substring "From") step "accept default" sendKeys "Enter" (Substring "To") step "enter to: email" sendLine "user@to.test" (Substring "Subject") step "leave default" sendKeys "Enter" (Substring "~") step "enter mail body" sendKeys "iThis is a test body" (Substring "body") step "exit insert mode in vim" sendKeys "Escape" (Substring "body") step "exit vim" sendLine ": x" (Substring "text/plain") >>= assertSubstring "From: \"Joe Bloggs\" " step "user can change from header" sendKeys "f" (Regex $ "From: " <> buildAnsiRegex [] ["37"] [] <> "\"Joe Bloggs\" ") let lastLineIsStatusLine = Regex "Purebred:.*ComposeView-Attachments[[:space:]]+$^$" step "append an email" sendLine ", testuser@foo.test" lastLineIsStatusLine >>= assertSubstring "From: \"Joe Bloggs\" , testuser@foo.test" step "user can change to header" sendKeys "t" (Regex $ "To: " <> buildAnsiRegex [] ["37"] [] <> "user@to.test") step "append an additional from email" sendLine ", testuser@foo.test" lastLineIsStatusLine >>= assertSubstring "To: user@to.test, testuser@foo.test" step "user can add cc header" sendKeys "c" (Substring "Cc") step "enter cc: email" sendLine "user@cc.test" lastLineIsStatusLine >>= assertSubstring "Cc: user@cc.test" step "user can change cc header" sendKeys "c" (Regex $ "Cc: " <> buildAnsiRegex [] ["37"] [] <> "user@cc.test") step "append an additional from email" sendLine ", testuser@foo.test" lastLineIsStatusLine >>= assertSubstring "Cc: user@cc.test, testuser@foo.test" step "user can add bcc header" sendKeys "b" (Substring "Bcc") step "enter bcc: email" sendLine "user@bcc.test" lastLineIsStatusLine >>= assertSubstring "Bcc: user@bcc.test" step "user can change bcc header" sendKeys "b" (Regex $ "Bcc: " <> buildAnsiRegex [] ["37"] [] <> "user@bcc.test") step "append an additional from email" sendLine ", testuser@foo.test" lastLineIsStatusLine >>= assertSubstring "Bcc: user@bcc.test, testuser@foo.test" step "change subject" sendKeys "s" (Regex $ "Subject: " <> buildAnsiRegex [] ["37"] [] <> "") step "enter subject" sendLine "foo subject" lastLineIsStatusLine >>= assertSubstring "Subject: foo subject" testPipeEntitiesSuccessfully :: PurebredTestCase testPipeEntitiesSuccessfully = purebredTmuxSession "pipe entities successfully" $ \step -> do setEnvVarInSession "LESS" "" startApplication step "open thread" sendKeys "Enter" (Substring "This is a test mail for purebred") step "show entities" sendKeys "v" (Substring "text/plain") step "pipe to" sendKeys "|" (Substring "Pipe to") step "use less" sendLine "less" (Regex "This is a test mail for purebred(.|[[:space:]])*\\(END\\)") testOpenEntitiesSuccessfully :: PurebredTestCase testOpenEntitiesSuccessfully = purebredTmuxSession "open entities successfully" $ \step -> do setEnvVarInSession "LESS" "" startApplication step "open thread" sendKeys "Enter" (Substring "This is a test mail for purebred") step "show entities" sendKeys "v" (Substring "text/plain") step "open one entity" sendKeys "o" (Substring "Open With") sendLine "less" (Regex "This is a test mail for purebred(.|[[:space:]])*purebred[[:alnum:]-]+ \\(END\\)") testOpenCommandDoesNotKillPurebred :: PurebredTestCase testOpenCommandDoesNotKillPurebred = purebredTmuxSession "open attachment does not kill purebred" $ \step -> do startApplication step "open thread" sendKeys "Enter" (Substring "This is a test mail for purebred") step "show entities" sendKeys "v" (Substring "text/plain") step "open with" sendKeys "o" (Substring "Open With") step "Open with bogus command" sendLine "asdfasdfasdf" (Substring "ProcessError") testShowsMailEntities :: PurebredTestCase testShowsMailEntities = purebredTmuxSession "shows mail entities successfully" $ \step -> do startApplication step "open thread" sendKeys "Enter" (Substring "This is a test mail for purebred") step "show entities" sendKeys "v" (Substring "text/plain") step "select the second entity" sendKeys "j" (Substring "text/html") step "close the list of entities" out <- sendKeys "q" (Substring "This is a test mail for purebred") -- poor mans (?!text) assertRegex "[^t][^e][^x][^t]" out testUserCanMoveBetweenThreads :: PurebredTestCase testUserCanMoveBetweenThreads = purebredTmuxSession "user can navigate between threads" $ \step -> do startApplication -- assert that the first mail is really the one we're later navigating back -- to snapshot assertRegexS (buildAnsiRegex [] ["37"] ["43"] <> "[[:space:]]Aug'17.*Testmail with whitespace") step "View Mail" sendKeys "Enter" (Substring "This is a test mail for purebred") step "Navigate down the threads list" sendKeys "J" (Substring "HOLY PUREBRED") step "Navigate up the threads list" sendKeys "K" (Substring "This is a test mail for purebred") testRepliesToMailSuccessfully :: PurebredTestCase testRepliesToMailSuccessfully = purebredTmuxSession "replies to mail successfully" $ \step -> do let subject = "Testmail with whitespace in the subject" testdir <- view envConfigDir startApplication step "pick first mail" sendKeys "Enter" (Substring "This is a test mail for purebred") >>= put assertSubstringS "From: " assertSubstringS "To: " assertSubstringS ("Subject: " <> subject) step "start replying" sendKeys "r" (Substring "> This is a test mail for purebred") step "exit vim" sendLine ": x" (Substring "Attachments") >>= put assertRegexS "From: \"Joe Bloggs\" [[:space:]]+$" assertSubstringS "To: frase@host.example" assertRegexS "Cc:[[:space:]]*$" -- Cc should be empty assertSubstringS ("Subject: Re: " <> subject) -- https://github.com/purebred-mua/purebred/issues/379 step "edit the mail one more time" sendKeys "e" (Substring "> This is a test mail for purebred") step "insert body" sendKeys "oThis is more information" (Substring "This is more information") sendKeys "Escape" Unconditional step "exit vim" sendLine ": x" (Substring "Item 1 of 1") step "send mail" sendKeys "y" (Substring "Query") let fpath = testdir "sentMail" contents <- liftIO $ B.readFile fpath let decoded = chr . fromEnum <$> B.unpack contents assertSubstr ("Subject: Re: " <> subject) decoded assertSubstr "From: \"Joe Bloggs\" " decoded assertSubstr "To: frase@host.example" decoded assertSubstr "> This is a test mail for purebred" decoded testFromAddressIsProperlyReset :: PurebredTestCase testFromAddressIsProperlyReset = purebredTmuxSession "from address is reset to configured identity" $ \step -> do startApplication step "Start composing" sendKeys "m" (Substring "Joe Bloggs") step "abort editing" sendKeys "Escape" (Substring "tag:inbox") step "Start composing again" sendKeys "m" (Substring "Joe Bloggs") testCanJumpToFirstListItem :: PurebredTestCase testCanJumpToFirstListItem = purebredTmuxSession "can jump to first and last mail" $ \step -> do startApplication step "Jump to last mail" sendKeys "G" (Substring "4 of 4") step "Jump to first mail" sendKeys "1" (Substring "1 of 4") testUpdatesReadState :: PurebredTestCase testUpdatesReadState = purebredTmuxSession "updates read state for mail and thread" $ \step -> do startApplication findMail step "subject:WIP Refactor" step "view unread mail in thread" sendKeys "Enter" (Substring "WIP Refactor") step "view next unread in thread" sendKeys "Down" (Substring "2 of 2") step "go back to thread list which is now read" sendKeys "q" (Regex (buildAnsiRegex [] ["30"] ["43"] <> T.encodeUtf8 " Feb'17[[:space:]]Róman[[:space:]]Joost[[:space:]]+\\(2\\)")) step "set one mail to unread" sendKeys "Enter" (Substring "Beginning of large text") sendKeys "t" (Regex (buildAnsiRegex [] ["37"] [] <> "[[:space:]]Re: WIP Refactor[[:space:]]+" <> buildAnsiRegex [] ["34"] ["49"])) step "returning to thread list shows entire thread as unread" sendKeys "q" (Regex (buildAnsiRegex [] ["37"] [] <> "[[:space:]]WIP Refactor[[:space:]]")) testConfig :: PurebredTestCase testConfig = purebredTmuxSession "test custom config" $ \step -> do -- Set a short command prompt, to a value otherwise unlikely to -- appear, so that we can easily check for program termination. let unlikelyString = "unlikely" sendKeys ("PS1=" <> unlikelyString <> "$ \r") (Substring unlikelyString) startApplication step "archive thread" sendKeys "a" (Substring "archive") step "quit" sendKeys "q" Unconditional -- Wait a bit so that purebred, which may not yet have -- terminated, does not eat the upcoming keystroke(s) liftIO $ threadDelay 200000 -- 0.2 seconds -- Press Enter again to deal with case where cursor is not at -- column 0, which could cause target string to be split. sendKeys "Enter" (Substring unlikelyString) -- https://github.com/purebred-mua/purebred/issues/391 testFileBrowserInvalidPath :: PurebredTestCase testFileBrowserInvalidPath = purebredTmuxSession "file browser handles invalid path input" $ \step -> do startApplication composeNewMail step step "start file browser" cwd <- B.pack <$> liftIO getCurrentDirectory sendKeys "a" (Regex $ "Path:[[:space:]]" <> buildAnsiRegex [] ["34"] [] <> cwd) step "focus search path editor" sendKeys ":" (Regex $ "Path:[[:space:]]" <> buildAnsiRegex [] ["37"] [] <> cwd) step "clear input and enter invalid directory" sendKeys "C-u" Unconditional sendLine "asdfasdf" (Substring "asdfasdf does not exist") testAddAttachments :: PurebredTestCase testAddAttachments = purebredTmuxSession "use file browser to add attachments" $ \step -> do testdir <- view envConfigDir -- To be resilient against differences in list contents between -- git and sdist, list the directory ourselves to work out what -- the final entry should be. Note that dirs come first and the -- files are sorted case insensitively in the filebrowser widget. let caseInsensitive a b = compare (T.toLower a) (T.toLower b) files <- sortBy caseInsensitive . fmap T.pack <$> liftIO ( getSourceDirectory >>= listDirectory >>= filterM (fmap isRegularFile . getFileStatus) ) let lastFile = fromMaybe "MISSING" $ preview (_last . to T.encodeUtf8) files secondLastFile = fromMaybe "MISSING" $ preview (_init . _last . to T.encodeUtf8) files startApplication composeNewMail step step "start file browser" cwd <- B.pack <$> liftIO getCurrentDirectory sendKeys "a" (Regex $ "Path: " <> buildAnsiRegex [] ["34"] [] <> cwd) step "jump to the end of the list" sendKeys "G" (Regex $ buildAnsiRegex [] [] ["43"] <> lastFile) step "add first selected file" sendKeys "Enter" (Substring lastFile) step "up to select mail body" sendKeys "Up" (Substring "Item 1 of 2") -- edit the mail body a few times to check if the code not mistakenly adds -- the same mail body as an attachment step "edit mail body text" sendKeys "e" (Substring "test body") step "append to mail body" sendKeys "i. foo" (Substring "foo") step "exit insert mode in vim" sendKeys "Escape" (Substring "foo") step "exit vim" sendKeys ": x\r" (Substring "Attachments") step "edit mail body text" sendKeys "e" (Substring "test body") step "append to mail body" sendKeys "i. foo" (Substring "foo") step "exit insert mode in vim" sendKeys "Escape" (Substring "foo") step "exit vim" sendKeys ": x\r" (Substring "Item 1 of 2") -- try removing attachments step "select the attachment" sendKeys "Down" (Substring "Item 2 of 2") >>= put assertRegexS (buildAnsiRegex [] ["43"] [] <> "[[:space:]]A[[:space:]]" <> lastFile) step "remove the attachment" sendKeys "D" (Not (Substring lastFile)) >>= put assertSubstringS "Item 1 of 1" step "try to remove the last attachment" sendKeys "D" (Substring "You may not remove the only attachment") -- add the attachment again and send it step "start file browser" sendKeys "a" (Regex $ "Path: " <> buildAnsiRegex [] ["34"] [] <> cwd) step "jump to the end of the list" sendKeys "G" (Regex $ buildAnsiRegex [] [] ["43"] <> lastFile) step "select the file" sendKeys "Space" (Regex $ buildAnsiRegex [] [] ["43"] <> lastFile <> "*") step "move one item up" sendKeys "Up" (Regex $ buildAnsiRegex [] [] ["43"] <> secondLastFile) step "add selected files" out <- sendKeys "Enter" (Substring "Item 3 of 3") assertSubstring secondLastFile out step "send mail" sendKeys "y" (Substring "Query") let fpath = testdir "sentMail" contents <- liftIO $ B.readFile fpath let decoded = chr . fromEnum <$> B.unpack contents assertSubstr "attachment; filename" decoded assertSubstr (B.unpack secondLastFile) decoded assertSubstr (B.unpack lastFile) decoded assertSubstr "This is a test body" decoded testManageTagsOnMails :: PurebredTestCase testManageTagsOnMails = purebredTmuxSession "manage tags on mails" $ \step -> do startApplication step "view mail in thread" sendKeys "Enter" (Substring "Testmail") step "focus command to show mail tags" sendKeys "`" (Regex (buildAnsiRegex [] ["37"] [])) step "enter new tag" sendLine "+inbox +foo +bar" (Regex ("foo" <> buildAnsiRegex [] ["30"] [] <> "[[:space:]]" <> buildAnsiRegex [] ["36"] [] <> "bar")) >>= assertSubstring "This is a test mail" step "go back to list of threads" sendKeys "Escape" (Substring "List of Threads") -- find newly tagged mail step "focus tag search" sendKeys ":" (Regex (buildAnsiRegex [] ["37"] [] <> "tag")) sendKeys "C-u" (Regex (buildAnsiRegex [] ["37"] [])) step "enter tag to search `foo and bar`" sendLine "tag:foo and tag:bar" (Substring "tag:foo and tag:bar") step "view mail in thread" sendKeys "Enter" (Substring "Testmail") step "attempt to add a new tag" sendKeys "`" (Regex (buildAnsiRegex [] ["37"] [])) step "cancel tagging and expect old UI" -- instead of asserting the absence of the tagging editor, we assert the -- last visible "item" in the UI followed by whitespace. sendKeys "Escape" (Regex "This is a test mail for purebred[[:space:]]+$") testManageTagsOnThreads :: PurebredTestCase testManageTagsOnThreads = purebredTmuxSession "manage tags on threads" $ \step -> do startApplication -- setup: tag the mails in the thread with two different tags and then -- tag the thread as a whole with a new tag. All mails should keep their -- distinct tags, while having received a new tag. step "navigate to thread" sendKeys "Down" (Substring "Item 2 of 4") sendKeys "Down" (Substring "Item 3 of 4") step "show thread mails" sendKeys "Enter" (Substring "ViewMail") step "open mail tag editor" sendKeys "`" (Regex ("Labels:." <> buildAnsiRegex [] ["37"] [])) step "add new tag" sendLine "+archive" (Substring "archive") step "move to second mail" sendKeys "Down" (Substring "Item 2 of 2") step "open mail tag editor" sendKeys "`" (Regex ("Labels:." <> buildAnsiRegex [] ["37"] [])) step "add new tag" sendLine "+replied -inbox" (Substring "replied") step "thread tags shows new tags" sendKeys "Escape" (Regex ("archive" <> buildAnsiRegex [] ["30"] [] <> "[[:space:]]" <> buildAnsiRegex [] ["36"] [] <> "replied")) step "open thread tag editor" sendKeys "`" (Regex ("Labels:." <> buildAnsiRegex [] ["37"] [])) step "remove tag" -- "cheating" here a bit, since just invoking tmux with sending literally -- "-only" will fail due to tmux parsing it as an argument, but the mail is -- already tagged with "thread" so the additional adding won't do anything sendLine "+thread" (Regex ("archive" <> buildAnsiRegex [] ["30"] [] <> "[[:space:]]" <> buildAnsiRegex [] ["36"] [] <> "replied" <> buildAnsiRegex [] ["30"] [] <> "[[:space:]]" <> buildAnsiRegex [] ["36"] [] <> "thread")) step "show thread mails" sendKeys "Enter" (Substring "ViewMail") step "second mail shows old tag" sendKeys "Escape" (Regex ("replied" <> buildAnsiRegex [] ["30"] [] <> "[[:space:]]" <> buildAnsiRegex [] ["36"] [] <> "thread" <> buildAnsiRegex [] ["30"] [] <> "[[:space:]]WIP Refactor")) step "open thread tag editor" sendKeys "`" (Regex ("Labels:." <> buildAnsiRegex [] ["37"] [])) step "abort editing" sendKeys "Escape" (Substring "Query") testHelp :: PurebredTestCase testHelp = purebredTmuxSession "help view" $ \step -> do startApplication step "shows Keybindings" sendKeys "?" (Regex "Escape>[[:space:]]+cancel") sendKeys "Escape" (Substring "Purebred") testShowsAndClearsError :: PurebredTestCase testShowsAndClearsError = purebredTmuxSession "shows and clears error" $ \step -> do startApplication testmdir <- view envMaildir liftIO $ removeFile (testmdir <> "/new/1502941827.R15455991756849358775.url") step "open thread" sendKeys "Enter" (Substring "Testmail") step "shows error message" sendKeys "Enter" (Substring "FileReadError") >>= assertRegex "(open|with)(Binary)?File:.*does not exist" step "error is cleared with next registered keybinding" sendKeys "Up" (Substring "Purebred: Item 1 of 4") testSetsMailToRead :: PurebredTestCase testSetsMailToRead = purebredTmuxSession "user can toggle read tag" $ \step -> do startApplication step "open thread" sendKeys "Enter" (Substring "This is a test mail for purebred") step "first unread mail is opened" sendKeys "Escape" (Substring "List of Threads") >>= assertRegex (buildAnsiRegex [] ["30"] [] <> ".*Testmail") step "show mail" sendKeys "Enter" (Substring "This is a test mail for purebred") step "toggle single mail back to unread (bold again)" sendKeys "t" (Regex (buildAnsiRegex [] ["37"] [] <> ".*Testmail")) testCanToggleHeaders :: PurebredTestCase testCanToggleHeaders = purebredTmuxSession "user can toggle Headers" $ \step -> do startApplication step "open thread" sendKeys "Enter" (Substring "Testmail") step "view mail" sendKeys "Enter" (Substring "This is a test mail") step "toggle to show all headers" sendKeys "h" (Regex "[Rr]eturn-[Pp]ath") step "toggle filtered headers" out <- sendKeys "h" (Substring "This is a test mail") assertRegex "Purebred.*\n.*[Ff]rom" out testUserViewsMailSuccessfully :: PurebredTestCase testUserViewsMailSuccessfully = purebredTmuxSession "user can view mail" $ \step -> do startApplication step "shows tag" snapshot assertSubstringS "inbox" assertSubstringS "Testmail with whitespace in the subject" step "open thread" sendKeys "Enter" (Substring "Testmail with whitespace in the subject") step "view mail" sendKeys "Enter" (Substring "This is a test mail") step "go back to thread list" sendKeys "q" (Substring "WIP Refactor") step "Move down to threaded mails" sendKeys "Down" (Substring "Purebred: Item 2 of 4") sendKeys "Down" (Substring "Purebred: Item 3 of 4") sendKeys "Enter" (Substring "Re: WIP Refactor") step "Scroll down" sendKeys "Enter" (Substring "Beginning of large text") sendKeys "Space" (Substring "Sed ut perspiciatis") step "go to next unread mail" sendKeys "j" (Substring "Re: WIP Refactor") step "Scroll down (again)" sendKeys "Space" (Substring "Sed ut perspiciatis") step "go to previous mail with reset scroll state" sendKeys "k" (Regex "Subject:[[:space:]].*WIP Refactor") testUserCanManipulateNMQuery :: PurebredTestCase testUserCanManipulateNMQuery = purebredTmuxSession "manipulating notmuch search query results in empty index" $ \step -> do startApplication step "focus command" sendKeys ":" (Regex (buildAnsiRegex [] ["37"] [] <> "tag")) step "delete all input" sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [])) step "search for non existing tags yielding no results" sendLine "does not match anything" (Substring "No items") step "search for mail correctly tagged" sendKeys ":" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "does")) sendKeys "C-u" (Regex (buildAnsiRegex [] ["37"] [])) step "enter new tag" sendLine "tag:replied" (Substring "Item 1 of 1") step "open thread" sendKeys "Enter" (Substring "This is Purebred") step "view currently selected mail" sendKeys "Enter" (Substring "HOLY PUREBRED") testUserCanSwitchBackToIndex :: PurebredTestCase testUserCanSwitchBackToIndex = purebredTmuxSession "user can switch back to mail index during composition" $ \step -> do startApplication step "start composition" sendKeys "m" (Substring "From") step "enter from email" sendKeys "C-a" Unconditional sendKeys "C-k" Unconditional sendKeys "testuser@foo.test\r" (Substring "To") step "enter to: email" sendKeys "user@to.test\r" (Substring "Subject") step "enter subject" sendKeys "test subject\r" (Substring "~") step "enter mail body" sendKeys "iThis is a test body" (Substring "body") step "exit insert mode in vim" sendKeys "Escape" (Substring "body") step "exit vim" sendKeys ": x\r" (Regex "From: testuser@foo.test") step "switch back to index" sendKeys "Tab" (Substring "Testmail") step "switch back to the compose editor" sendKeys "Tab" (Substring "test subject") testUserCanAbortMailComposition :: PurebredTestCase testUserCanAbortMailComposition = purebredTmuxSession "user can abort composing mail" $ \step -> do startApplication composeNewMail step step "abort mail" sendKeys "q" (Substring "Keep draft?") step "choose discard" -- TODO: buildAnsiRegex will cause the generated Regex not -- to match. Maybe not \\s+ even though raw it looks like -- there is white space? -- see https://github.com/purebred-mua/tasty-tmux/issues/8 sendKeys "Tab" (Substring "Discard") step "confirm discard" sendKeys "Enter" (Substring "Testmail") step "start composition again" sendKeys "m" (Substring "From") sendKeys "Enter" (Regex ("To:[[:space:]]" <> buildAnsiRegex [] ["37"] [])) step "enter to: email" sendKeys "new@second.test\r" (Regex ("Subject:[[:space:]]" <> buildAnsiRegex [] ["37"] [])) step "enter subject" sendKeys "test subject\r" (Regex "~[[:space:]]+") step "enter mail body" sendKeys "iThis is my second mail" Unconditional step "exit insert mode in vim" sendKeys "Escape" Unconditional step "exit vim" sendKeys ": x\r" (Substring "text/plain") >>= put assertSubstringS "To: new@second.test" assertSubstringS "Subject: test subject" step "edit body" sendKeys "e" (Regex "This is my second mail[[:space:]]+") testSendMail :: PurebredTestCase testSendMail = purebredTmuxSession "sending mail successfully" $ \step -> do testdir <- view envConfigDir mdir <- view envMaildir startApplication composeNewMail step step "user can re-edit body" sendKeys "e" (Substring "This is a test body") step "Writes more text" sendKeys "i. More text" (Substring "text") step "exit insert mode in vim" sendKeys "Escape" (Substring "body") step "exit vim" sendKeys ": x\r" (Regex ("text/plain; charset=us-ascii[[:space:]]" <> buildAnsiRegex [] ["34"] ["49"] <> "[[:space:]]+")) -- pre-check before we sent: -- * Drafts is empty before sending -- * Sent folder doesn't exist yet -- step "Drafts is empty before sending" assertFileAmountInMaildir (mdir "Drafts" "new") 0 step "Sent folder doesn't exist yet" files <- liftIO $ listDirectory mdir liftIO $ assertEqual "expected no maildir directories" (sort ["Drafts", ".notmuch", "notmuch-config", "new", "cur"]) (sort files) step "send mail and go back to threads" sendKeys "y" (Regex ("Query:[[:space:]]" <> buildAnsiRegex [] ["34"] [] <> "tag:inbox")) -- check that the sent mail can be parsed without errors step "parse mail with purebred-email" assertMailSuccessfullyParsed (testdir "sentMail") -- check that the sent mail is indexed step "focus query" sendKeys ":" (Regex (buildAnsiRegex [] ["37"] [] <> "tag")) step "delete all input" sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [])) step "enter sent tags" sendLine "tag:sent" (Substring "Draft mail subject") -- check that a copy of the sent mail has been copied to our Maildir step "Drafts directory is empty" assertFileAmountInMaildir (mdir "Drafts" "new") 0 step "Sent directory has a new entry" assertFileAmountInMaildir (mdir "Sent" "cur") 1 testSendFailureHandling :: PurebredTestCase testSendFailureHandling = purebredTmuxSession "send failure does not lose mail" $ \step -> do mdir <- view envMaildir setEnvVarInSession "PUREBRED_SEND_FAIL" "1" startApplication composeNewMail step step "send mail attempt #1 fails" sendKeys "y" (Substring "PUREBRED_SEND_FAIL") step "compose view remains active" assertSubstringS "From: \"Joe Bloggs\" " step "Sent folder doesn't exist yet" files <- liftIO $ listDirectory mdir liftIO $ assertEqual "expected no maildir directories" (sort ["Drafts", ".notmuch", "notmuch-config", "new", "cur"]) (sort files) step "send mail attempt #2 succeeds" sendKeys "y" (Regex ("Query:[[:space:]]" <> buildAnsiRegex [] ["34"] [] <> "tag:inbox")) -- check that the sent mail is indexed step "focus query" sendKeys ":" (Regex (buildAnsiRegex [] ["37"] [] <> "tag")) step "delete all input" sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [])) step "enter sent tags" sendLine "tag:sent" (Substring "Draft mail subject") step "Sent directory has a new entry" assertFileAmountInMaildir (mdir "Sent" "cur") 1 testReplyRendersNonASCIIHeadersCorrectly :: PurebredTestCase testReplyRendersNonASCIIHeadersCorrectly = purebredTmuxSession "reply to msg w/ utf8 From; mailbox renders properly" $ \step -> do startApplication step "focus search edit" sendKeys ":" (Regex (buildAnsiRegex [] ["37"] [] <> "tag")) step "delete all input" sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [])) step "search for msg <1234@url>" sendLine "id:1234@url" (Substring "Item 1 of 1") step "open thread" sendKeys "Enter" (Substring "Beginning of large text") step "start replying" sendKeys "r" (Substring "> Beginning of large text") step "exit vim" sendLine ": x" (Substring "Attachments") >>= put assertRegexS $ T.encodeUtf8 "To: \"Róman Joost\" " testGroupReply :: PurebredTestCase testGroupReply = purebredTmuxSession "group reply Cc's recipients of parent" $ \step -> do startApplication step "focus search edit" sendKeys ":" (Regex (buildAnsiRegex [] ["37"] [] <> "tag")) step "delete all input" sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [])) step "search for msg <20170817035004.55C4580B8F@host.example>" sendLine "id:20170817035004.55C4580B8F@host.example" (Substring "Item 1 of 1") step "open thread" sendKeys "Enter" (Substring "This is a test mail for purebred") step "start replying" sendKeys "g" (Substring "> This is a test mail for purebred") step "exit vim" sendLine ": x" (Substring "Attachments") >>= put assertRegexS $ T.encodeUtf8 "To: frase@host.example" assertRegexS $ T.encodeUtf8 "Cc: roman@host.example, joe@host.example" findMail :: ( HasTmuxSession testEnv , MonadReader testEnv m , MonadState Capture m , MonadIO m ) => (String -> m ()) -> String -- ^ query -> m Capture findMail step query = do step ("search for mail with query: " <> query) sendKeys ":" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "tag:inbox")) sendKeys "C-u" (Regex ("Query: " <> buildAnsiRegex [] ["37"] [] <> "[[:space:]]+")) step "enter free text search" sendLine query (Substring "Item 1 of 1") assertEditorResetsToInitialValue :: HasTmuxSession testEnv => (MonadReader testEnv m, MonadState Capture m, MonadIO m) => (String -> m ()) -> String -> B.ByteString -> B.ByteString -> m () assertEditorResetsToInitialValue step key focused unfocused = do step "focusing editor" sendKeys key (Regex $ focused <> "[[:space:]]+") step "entering bogus characters" sendKeys "asdf" (Regex $ focused <> "asdf" <> "[[:space:]]+") step "abort and expect old editor value is reset" sendKeys "Escape" (Regex $ unfocused <> "[[:space:]]+") >>= put assertConditionS (Not (Substring "Failed reading")) composeNewMail :: HasTmuxSession testEnv => (MonadReader testEnv m, MonadState Capture m, MonadIO m) => (String -> m ()) -> m () composeNewMail step = do step "start composition" sendKeys "m" (Substring "From") step "accept default" sendKeys "Enter" (Substring "To") step "enter to: email" sendKeys "user@to.test\r" (Substring "Subject") step "leave default" sendKeys "Draft mail subject\r" (Substring "~") step "enter mail body" sendKeys "iThis is a test body" (Substring "body") step "exit insert mode in vim" sendKeys "Escape" (Substring "body") step "exit vim" sendKeys ": x\r" (Substring "text/plain") >>= put assertSubstringS "From: \"Joe Bloggs\" " parseMail :: B.ByteString -> Either String MIMEMessage parseMail = parse (message mime) assertSubstr :: MonadIO m => String -> String -> m () assertSubstr needle haystack = liftIO $ assertBool (needle <> " not found in\n\n" <> haystack) (needle `isInfixOf` haystack) assertMailSuccessfullyParsed :: (MonadIO m) => String -> m () assertMailSuccessfullyParsed fp = do contents <- liftIO $ B.readFile fp let result = parseMail contents liftIO $ assertBool "expected successful MIMEMessage" (isRight result) -- | Check number of files in dir matches expectation. -- -- To allow for lag in filesystem change visibility, we initially -- wait 62.5ms, and retry up to 3 times, doubling the delay each -- time, for a total delay of ~937.5 ms. -- assertFileAmountInMaildir :: (MonadIO m) => FilePath -> Int -> m () assertFileAmountInMaildir dir expected = liftIO (go (3 :: Int) 62500) where errmsg fs = "expecting " <> show expected <> " file(s), dir contents: " <> show fs go n d = do threadDelay d files <- listDirectory dir case length files of l | l == expected -> pure () _ | n > 0 -> go (n - 1) (d * 2) _ -> assertFailure (errmsg files) -- Global test environment (shared by all test cases) newtype GlobalEnv = GlobalEnv FilePath -- Session test environment data Env = Env { _envConfigDir :: FilePath , _envMaildir :: FilePath , _envNotmuchConfig :: FilePath , _envSessionName :: String } instance HasTmuxSession Env where tmuxSession = envSessionName -- | Session-specific config dir envConfigDir :: Lens' Env FilePath envConfigDir = lens _envConfigDir (\s b -> s { _envConfigDir = b }) envMaildir :: Lens' Env FilePath envMaildir = lens _envMaildir (\s b -> s { _envMaildir = b }) envNotmuchConfig :: Lens' Env FilePath envNotmuchConfig = lens _envNotmuchConfig (\s b -> s { _envNotmuchConfig = b }) envSessionName :: Lens' Env String envSessionName = lens _envSessionName (\s b -> s { _envSessionName = b }) -- | Tear down a test session tearDown :: Env -> IO () tearDown (Env confdir mdir _ _) = do removeDirectoryRecursive confdir removeDirectoryRecursive mdir -- | Set up a test session. setUp :: GlobalEnv -> TmuxSession -> IO Env setUp (GlobalEnv globalConfigDir) sessionName = do maildir <- setUpTempMaildir nmCfg <- setUpNotmuchCfg maildir setUpNotmuch nmCfg confdir <- mkTempDir runProcess_ $ proc "sh" ["-c", "cp -a " <> globalConfigDir <> "/* " <> confdir] flip runReaderT sessionName $ do -- a) Make the regex less color code dependent by setting the TERM to 'screen'. -- This can happen if different environments support more than 16 colours (e.g. -- background values > 37), while our CI environment only supports 16 colours. -- -- Previously we used value "ansi". But we changed this because -- "ansi" can have different capabilities on different platforms, -- including missing ones. On the other hand, "screen" triggers -- special handling within vty. setEnvVarInSession "TERM" "screen" -- set the config dir setEnvVarInSession "PUREBRED_CONFIG_DIR" confdir setEnvVarInSession "NOTMUCH_CONFIG" nmCfg pure $ Env confdir maildir nmCfg sessionName precompileConfig :: FilePath -> IO () precompileConfig testdir = do env <- getEnvironment let systemEnv = ("PUREBRED_CONFIG_DIR", testdir) : env config = setEnv systemEnv $ proc "purebred" ["--version"] runProcess_ config -- | Get the explicitly-specified source directory via SRCDIR -- env var, or fall back to CWD. getSourceDirectory :: IO FilePath getSourceDirectory = lookupEnv "SRCDIR" >>= maybe getCurrentDirectory pure setUpPurebredConfig :: FilePath -> IO () setUpPurebredConfig testdir = do c <- getSourceDirectory copyFile (c <> "/configs/purebred.hs") (testdir <> "/purebred.hs") mkTempDir :: IO FilePath mkTempDir = getCanonicalTemporaryDirectory >>= flip createTempDirectory "purebredtest" -- | Set up a temporary Maildir containing the test database -- The returned directory contains the 'Maildir' subdirectory. setUpTempMaildir :: IO FilePath setUpTempMaildir = do basedir <- mkTempDir cwd <- getSourceDirectory runProcess_ $ proc "cp" ["-R", cwd <> "/test/data/Maildir", basedir] let mdir = basedir "Maildir" -- Rename files with maildir flags ; these had to be renamed (':' replaced -- with '_') to appease Hackage requirement that tarballs only contain -- filenames that are valid on both POSIX and Windows. We have to fix the -- filenames here before using them. -- -- In a Nix system the PATH environment may contain relative paths. -- For security reasons find(1) refuses to run when -execdir is given -- and PATH contains relative paths. So we have to remove relative -- dirs from PATH. -- path <- intercalate [searchPathSeparator] . filter isAbsolute <$> getSearchPath let f (k, _) | k == "PATH" = (k, path) f x = x env <- fmap f <$> getEnvironment runProcess_ $ setEnv env $ proc "find" [ mdir, "-name", "*_2,*" , "-execdir", "sh", "-c", "mv {} $(echo {} | sed s/_2,/:2,/)", ";" ] pure mdir -- | run notmuch to create the notmuch database -- Note: discard stdout which otherwise clobbers the test output setUpNotmuch :: FilePath -> IO () setUpNotmuch notmuchcfg = void $ readProcess_ $ proc "notmuch" ["--config=" <> notmuchcfg, "new" ] -- | Write a minimal notmuch config pointing to the given maildir. -- Returns the path to the notmuch configuration file (which is -- created under the given maildir directory). -- setUpNotmuchCfg :: FilePath -> IO FilePath setUpNotmuchCfg dir = do let cfgData = "[database]\npath=" <> dir <> "\n" cfgFile = dir <> "/notmuch-config" writeFile cfgFile cfgData $> cfgFile purebredTmuxSession = withTmuxSession setUp tearDown -- | convenience function to print captured output to STDERR debugOutput :: String -> IO () debugOutput out = do d <- lookupEnv "DEBUG" when (isJust d) $ hPutStr stderr ("\n\n" <> out) -- | start the application -- Note: this is currently defined as an additional test step for no good -- reason. startApplication :: (MonadReader Env m, MonadIO m) => m () startApplication = do srcdir <- liftIO getSourceDirectory tmuxSendKeys LiteralKeys ("cd " <> srcdir <> "\r") tmuxSendKeys InterpretKeys ("purebred\r") void $ waitForCondition (Substring "Purebred: Item") defaultRetries defaultBackoff -- | A list item which is toggled for a batch operation -- selectedListItem :: B.ByteString selectedListItem = buildAnsiRegex [] ["37"] ["43"] toggledListItem :: B.ByteString toggledListItem = buildAnsiRegex [] ["36"] [] newListItem :: B.ByteString newListItem = buildAnsiRegex [] ["37"] ["49"]