module Imm.Hooks.SendMail (module Imm.Hooks.SendMail, module Reexport) where
import Imm.Feed
import Imm.Hooks
import Imm.Prelude
import Imm.Pretty
import Data.NonNull
import Data.Time
import Network.HaskellNet.SMTP as Reexport
import Network.HaskellNet.SMTP.SSL as Reexport
import Network.Mail.Mime as Reexport hiding (sendmail)
import Network.Socket
import Text.Atom.Types
import Text.RSS.Types
type Username = String
type Password = String
type ServerName = String
data ConnectionSettings = Plain ServerName PortNumber | Ssl ServerName Settings | StartTls ServerName Settings
deriving(Eq, Show)
data Authentication = Authentication AuthType Username Password
deriving(Eq, Show)
data SMTPServer = SMTPServer (Maybe Authentication) ConnectionSettings
deriving (Eq, Show)
data FormatMail = FormatMail
{ formatFrom :: Feed -> FeedElement -> Address
, formatSubject :: Feed -> FeedElement -> Text
, formatBody :: Feed -> FeedElement -> Text
, formatTo :: Feed -> FeedElement -> [Address]
}
data SendMailSettings = SendMailSettings (Feed -> FeedElement -> SMTPServer) FormatMail
mkCoHooks :: (MonadIO m) => SendMailSettings -> CoHooksF m SendMailSettings
mkCoHooks a@(SendMailSettings connectionSettings formatMail) = CoHooksF coOnNewElement where
coOnNewElement feed element = do
timezone <- io getCurrentTimeZone
currentTime <- io getCurrentTime
let mail = buildMail formatMail currentTime timezone feed element
io $ withSMTPConnection (connectionSettings feed element) $ sendMimeMail2 mail
return a
defaultFormatFrom :: Feed -> FeedElement -> Address
defaultFormatFrom (Rss doc) (RssElement item) = Address (Just $ channelTitle doc <> " (" <> itemAuthor item <> ")") ""
defaultFormatFrom (Atom feed) (AtomElement entry) = Address (Just $ title <> " (" <> authors <> ")") ""
where title = show . prettyAtomText $ feedTitle feed
authors = intercalate ", " $ map (toNullable . personName) $ entryAuthors entry <> feedAuthors feed
defaultFormatFrom _ _ = Address (Just "Unknown") ""
defaultFormatSubject :: Feed -> FeedElement -> Text
defaultFormatSubject _ element = getTitle element
defaultFormatBody :: Feed -> FeedElement -> Text
defaultFormatBody _ (RssElement item) = "<p>" <> maybe "<no link>" (withRssURI (show . prettyURI)) (itemLink item) <> "</p><p>" <> itemDescription item <> "</p>"
defaultFormatBody _ (AtomElement entry) = "<p>" <> intercalate "<br/>" links <> "</p><p>" <> fromMaybe "<empty>" (content <|> summary) <> "</p>"
where links = map (withAtomURI (show . prettyURI) . linkHref) $ entryLinks entry
content = show . prettyAtomContent <$> entryContent entry
summary = show . prettyAtomText <$> entrySummary entry
authenticate_ :: SMTPConnection -> Authentication -> IO Bool
authenticate_ connection (Authentication t u p) = do
result <- authenticate t u p connection
unless result $ putStrLn "Authentication failed"
return result
withSMTPConnection :: SMTPServer -> (SMTPConnection -> IO a) -> IO a
withSMTPConnection (SMTPServer authentication (Plain server port)) f =
doSMTPPort server port $ \connection -> do
forM_ authentication (authenticate_ connection)
f connection
withSMTPConnection (SMTPServer authentication (Ssl server settings)) f =
doSMTPSSLWithSettings server settings $ \connection -> do
forM_ authentication (authenticate_ connection)
f connection
withSMTPConnection (SMTPServer authentication (StartTls server settings)) f =
doSMTPSTARTTLSWithSettings server settings $ \connection -> do
forM_ authentication (authenticate_ connection)
f connection
buildMail :: FormatMail -> UTCTime -> TimeZone -> Feed -> FeedElement -> Mail
buildMail format currentTime timeZone feed element =
let date = formatTime defaultTimeLocale "%a, %e %b %Y %T %z" $ utcToZonedTime timeZone $ fromMaybe currentTime $ getDate element
in Mail
{ mailFrom = formatFrom format feed element
, mailTo = formatTo format feed element
, mailCc = []
, mailBcc = []
, mailHeaders =
[ ("Return-Path", "<imm@noreply>")
, ("Date", fromString date)
, ("Subject", formatSubject format feed element)
, ("Content-disposition", "inline")
]
, mailParts = [[htmlPart $ fromStrict $ formatBody format feed element]]
}