2016-05-20 01:58:23 +09:00
|
|
|
{- This file is part of Vervis.
|
|
|
|
-
|
2019-03-16 01:36:02 +09:00
|
|
|
- Written in 2016, 2019 by fr33domlover <fr33domlover@riseup.net>.
|
2016-05-20 01:58:23 +09:00
|
|
|
-
|
|
|
|
- ♡ Copying is an act of love. Please copy, reuse and share.
|
|
|
|
-
|
|
|
|
- The author(s) have dedicated all copyright and related and neighboring
|
|
|
|
- rights to this software to the public domain worldwide. This software is
|
|
|
|
- distributed without any warranty.
|
|
|
|
-
|
|
|
|
- You should have received a copy of the CC0 Public Domain Dedication along
|
|
|
|
- with this software. If not, see
|
|
|
|
- <http://creativecommons.org/publicdomain/zero/1.0/>.
|
|
|
|
-}
|
|
|
|
|
|
|
|
module Vervis.Handler.Discussion
|
|
|
|
( getDiscussion
|
2019-03-16 01:36:02 +09:00
|
|
|
, getDiscussionMessage
|
2016-05-22 05:01:31 +09:00
|
|
|
, getTopReply
|
|
|
|
, postTopReply
|
2016-05-20 07:07:25 +09:00
|
|
|
, getReply
|
|
|
|
, postReply
|
2016-05-20 01:58:23 +09:00
|
|
|
)
|
|
|
|
where
|
|
|
|
|
|
|
|
import Prelude
|
|
|
|
|
2019-03-16 01:36:02 +09:00
|
|
|
import Control.Monad
|
2016-05-20 01:58:23 +09:00
|
|
|
import Control.Monad.IO.Class (liftIO)
|
2019-04-21 06:34:45 +09:00
|
|
|
import Control.Monad.Trans.Except
|
2019-03-23 05:46:42 +09:00
|
|
|
import Data.Maybe
|
2016-05-20 01:58:23 +09:00
|
|
|
import Data.Time.Clock (getCurrentTime)
|
2016-05-20 07:07:25 +09:00
|
|
|
import Database.Persist
|
2019-03-23 05:46:42 +09:00
|
|
|
import Database.Persist.Sql
|
|
|
|
import Data.Traversable
|
2016-05-20 01:58:23 +09:00
|
|
|
import Text.Blaze.Html (Html)
|
2019-04-21 06:34:45 +09:00
|
|
|
import Data.Text (Text)
|
|
|
|
import Yesod.Auth
|
2019-03-23 05:46:42 +09:00
|
|
|
import Yesod.Core
|
2019-03-16 01:36:02 +09:00
|
|
|
import Yesod.Core.Handler
|
2016-05-20 07:07:25 +09:00
|
|
|
import Yesod.Form.Functions (runFormPost)
|
|
|
|
import Yesod.Form.Types (FormResult (..))
|
2016-05-20 01:58:23 +09:00
|
|
|
import Yesod.Persist.Core (runDB, get404, getBy404)
|
|
|
|
|
2019-06-02 23:41:51 +09:00
|
|
|
import qualified Data.Text as T
|
|
|
|
|
2019-05-25 01:09:58 +09:00
|
|
|
import Data.Aeson.Encode.Pretty.ToEncoding
|
2019-05-25 22:01:15 +09:00
|
|
|
import Database.Persist.JSON
|
2019-03-20 17:07:37 +09:00
|
|
|
import Network.FedURI
|
2019-03-23 05:46:42 +09:00
|
|
|
import Web.ActivityPub
|
2019-04-21 06:34:45 +09:00
|
|
|
import Yesod.Auth.Unverified
|
2019-03-23 05:46:42 +09:00
|
|
|
import Yesod.FedURI
|
2019-03-29 12:25:32 +09:00
|
|
|
import Yesod.Hashids
|
2019-03-23 05:46:42 +09:00
|
|
|
|
|
|
|
import Database.Persist.Local
|
|
|
|
import Yesod.Persist.Local
|
2019-03-20 17:07:37 +09:00
|
|
|
|
|
|
|
import Vervis.Discussion
|
2016-05-20 07:07:25 +09:00
|
|
|
import Vervis.Form.Discussion
|
2019-04-21 06:34:45 +09:00
|
|
|
import Vervis.Federation
|
2019-03-23 05:46:42 +09:00
|
|
|
import Vervis.Foundation
|
2016-05-20 01:58:23 +09:00
|
|
|
import Vervis.Model
|
2019-03-23 05:46:42 +09:00
|
|
|
import Vervis.Model.Ident
|
2019-06-02 23:41:51 +09:00
|
|
|
import Vervis.Render
|
2019-03-23 05:46:42 +09:00
|
|
|
import Vervis.Settings
|
2016-05-20 01:58:23 +09:00
|
|
|
import Vervis.Widget.Discussion
|
|
|
|
|
2016-05-22 06:27:12 +09:00
|
|
|
getDiscussion
|
2019-03-16 01:36:02 +09:00
|
|
|
:: (MessageId -> Route App)
|
|
|
|
-> Route App
|
|
|
|
-> AppDB DiscussionId
|
|
|
|
-> Handler Html
|
2016-05-22 06:27:12 +09:00
|
|
|
getDiscussion reply topic getdid =
|
|
|
|
defaultLayout $ discussionW getdid topic reply
|
2016-05-20 01:58:23 +09:00
|
|
|
|
2019-03-20 17:07:37 +09:00
|
|
|
getNode :: AppDB DiscussionId -> MessageId -> AppDB MessageTreeNode
|
|
|
|
getNode getdid mid = do
|
|
|
|
did <- getdid
|
|
|
|
m <- get404 mid
|
|
|
|
unless (messageRoot m == did) notFound
|
|
|
|
mlocal <- getBy $ UniqueLocalMessage mid
|
|
|
|
mremote <- getBy $ UniqueRemoteMessage mid
|
|
|
|
author <- case (mlocal, mremote) of
|
|
|
|
(Nothing, Nothing) -> fail "Message with no author"
|
|
|
|
(Just _, Just _) -> fail "Message used as both local and remote"
|
|
|
|
(Just (Entity lmid lm), Nothing) -> do
|
|
|
|
p <- getJust $ localMessageAuthor lm
|
|
|
|
s <- getJust $ personIdent p
|
|
|
|
return $ MessageTreeNodeLocal lmid s
|
|
|
|
(Nothing, Just (Entity _rmid rm)) -> do
|
|
|
|
rs <- getJust $ remoteMessageAuthor rm
|
2019-04-12 09:56:27 +09:00
|
|
|
i <- getJust $ remoteActorInstance rs
|
2019-05-07 11:54:45 +09:00
|
|
|
return $
|
|
|
|
MessageTreeNodeRemote
|
|
|
|
(instanceHost i)
|
|
|
|
(remoteMessageIdent rm)
|
|
|
|
(remoteActorIdent rs)
|
2019-05-21 18:11:13 +09:00
|
|
|
(remoteActorName rs)
|
2019-03-20 17:07:37 +09:00
|
|
|
return $ MessageTreeNode mid m author
|
|
|
|
|
2019-03-23 05:46:42 +09:00
|
|
|
{-
|
2019-03-22 04:06:52 +09:00
|
|
|
getNodeL :: AppDB DiscussionId -> LocalMessageId -> AppDB MessageTreeNode
|
|
|
|
getNodeL getdid lmid = do
|
|
|
|
did <- getdid
|
|
|
|
lm <- get404 lmid
|
|
|
|
let mid = localMessageRest lm
|
|
|
|
m <- getJust mid
|
|
|
|
unless (messageRoot m == did) notFound
|
|
|
|
p <- getJust $ localMessageAuthor lm
|
|
|
|
s <- getJust $ personIdent p
|
|
|
|
return $ MessageTreeNode mid m $ MessageTreeNodeLocal lmid s
|
2019-03-23 05:46:42 +09:00
|
|
|
-}
|
2019-03-22 04:06:52 +09:00
|
|
|
|
2019-03-23 05:46:42 +09:00
|
|
|
getDiscussionMessage :: ShrIdent -> LocalMessageId -> Handler TypedContent
|
2019-05-25 01:09:58 +09:00
|
|
|
getDiscussionMessage shr lmid = do
|
|
|
|
doc <- runDB $ do
|
|
|
|
sid <- getKeyBy404 $ UniqueSharer shr
|
|
|
|
pid <- getKeyBy404 $ UniquePersonIdent sid
|
|
|
|
lm <- get404 lmid
|
|
|
|
unless (localMessageAuthor lm == pid) notFound
|
|
|
|
m <- getJust $ localMessageRest lm
|
|
|
|
route2fed <- getEncodeRouteHome
|
|
|
|
uContext <- do
|
|
|
|
let did = messageRoot m
|
|
|
|
mt <- getValBy $ UniqueTicketDiscussion did
|
|
|
|
mrd <- getValBy $ UniqueRemoteDiscussion did
|
|
|
|
case (mt, mrd) of
|
|
|
|
(Nothing, Nothing) -> fail $ "DiscussionId #" ++ show did ++ " has no context"
|
|
|
|
(Just _, Just _) -> fail $ "DiscussionId #" ++ show did ++ " has both ticket and remote contexts"
|
|
|
|
(Just t, Nothing) -> do
|
|
|
|
j <- getJust $ ticketProject t
|
|
|
|
s <- getJust $ projectSharer j
|
|
|
|
let shr = sharerIdent s
|
|
|
|
prj = projectIdent j
|
|
|
|
return $ route2fed $ TicketR shr prj $ ticketNumber t
|
|
|
|
(Nothing, Just rd) -> do
|
|
|
|
i <- getJust $ remoteDiscussionInstance rd
|
|
|
|
return $ l2f (instanceHost i) (remoteDiscussionIdent rd)
|
|
|
|
muParent <- for (messageParent m) $ \ midParent -> do
|
|
|
|
mlocal <- getBy $ UniqueLocalMessage midParent
|
|
|
|
mremote <- getValBy $ UniqueRemoteMessage midParent
|
|
|
|
case (mlocal, mremote) of
|
|
|
|
(Nothing, Nothing) -> fail "Message with no author"
|
|
|
|
(Just _, Just _) -> fail "Message used as both local and remote"
|
|
|
|
(Just (Entity lmidParent lmParent), Nothing) -> do
|
|
|
|
p <- getJust $ localMessageAuthor lmParent
|
|
|
|
s <- getJust $ personIdent p
|
|
|
|
lmhidParent <- encodeKeyHashid lmidParent
|
|
|
|
return $ route2fed $ MessageR (sharerIdent s) lmhidParent
|
|
|
|
(Nothing, Just rmParent) -> do
|
|
|
|
rs <- getJust $ remoteMessageAuthor rmParent
|
|
|
|
i <- getJust $ remoteActorInstance rs
|
|
|
|
return $ l2f (instanceHost i) (remoteActorIdent rs)
|
2019-05-25 22:01:15 +09:00
|
|
|
ob <- getJust $ localMessageCreate lm
|
|
|
|
let activity = docValue $ persistJSONValue $ outboxItemActivity ob
|
2019-03-23 05:46:42 +09:00
|
|
|
|
2019-05-25 01:09:58 +09:00
|
|
|
host <- getsYesod $ appInstanceHost . appSettings
|
|
|
|
route2local <- getEncodeRouteLocal
|
|
|
|
lmhid <- encodeKeyHashid lmid
|
|
|
|
return $ Doc host Note
|
|
|
|
{ noteId = Just $ route2local $ MessageR shr lmhid
|
|
|
|
, noteAttrib = route2local $ SharerR shr
|
2019-05-25 22:01:15 +09:00
|
|
|
, noteAudience =
|
|
|
|
case activitySpecific activity of
|
|
|
|
CreateActivity (Create note) -> noteAudience note
|
|
|
|
_ -> error $ "lmid#" ++ show (fromSqlKey lmid) ++ "'s create isn't a Create activity!"
|
2019-05-25 01:09:58 +09:00
|
|
|
, noteReplyTo = Just $ fromMaybe uContext muParent
|
|
|
|
, noteContext = Just uContext
|
|
|
|
, notePublished = Just $ messageCreated m
|
2019-06-02 23:41:51 +09:00
|
|
|
, noteSource = messageSource m
|
|
|
|
, noteContent = messageContent m
|
2019-05-25 01:09:58 +09:00
|
|
|
}
|
|
|
|
selectRep $ do
|
|
|
|
provideAP $ pure doc
|
|
|
|
provideRep $
|
|
|
|
defaultLayout
|
|
|
|
[whamlet|
|
|
|
|
<div><pre>#{encodePrettyToLazyText doc}
|
|
|
|
|]
|
2016-05-20 07:07:25 +09:00
|
|
|
|
2016-05-22 05:01:31 +09:00
|
|
|
getTopReply :: Route App -> Handler Html
|
|
|
|
getTopReply replyP = do
|
|
|
|
((_result, widget), enctype) <- runFormPost newMessageForm
|
|
|
|
defaultLayout $(widgetFile "discussion/top-reply")
|
|
|
|
|
|
|
|
postTopReply
|
2019-04-21 06:34:45 +09:00
|
|
|
:: Text
|
|
|
|
-> [Route App]
|
2019-05-18 07:42:01 +09:00
|
|
|
-> [Route App]
|
2019-04-21 06:34:45 +09:00
|
|
|
-> Route App
|
|
|
|
-> Route App
|
2019-03-20 17:07:37 +09:00
|
|
|
-> (LocalMessageId -> Route App)
|
2016-05-22 05:01:31 +09:00
|
|
|
-> Handler Html
|
2019-05-18 07:42:01 +09:00
|
|
|
postTopReply hDest recipsA recipsC context replyP after = do
|
2016-05-22 05:01:31 +09:00
|
|
|
((result, widget), enctype) <- runFormPost newMessageForm
|
2019-04-21 06:34:45 +09:00
|
|
|
elmid <- runExceptT $ do
|
|
|
|
msg <- case result of
|
|
|
|
FormMissing -> throwE "Field(s) missing."
|
|
|
|
FormFailure _l -> throwE "Message submission failed, see errors below."
|
|
|
|
FormSuccess nm -> return $ nmContent nm
|
2019-05-21 08:51:06 +09:00
|
|
|
encodeRouteFed <- getEncodeRouteHome
|
2019-04-21 06:34:45 +09:00
|
|
|
encodeRouteLocal <- getEncodeRouteLocal
|
|
|
|
let encodeRecipRoute = l2f hDest . encodeRouteLocal
|
|
|
|
shrAuthor <- do
|
|
|
|
Entity _ p <- requireVerifiedAuth
|
|
|
|
lift $ runDB $ sharerIdent <$> get404 (personIdent p)
|
2019-06-02 23:41:51 +09:00
|
|
|
let msg' = T.filter (/= '\r') msg
|
|
|
|
contentHtml <- ExceptT . pure $ renderPandocMarkdown msg'
|
2019-04-21 06:34:45 +09:00
|
|
|
let (hLocal, luAuthor) = f2l $ encodeRouteFed $ SharerR shrAuthor
|
|
|
|
uContext = encodeRecipRoute context
|
2019-05-18 07:42:01 +09:00
|
|
|
recips = recipsA ++ recipsC
|
2019-04-21 06:34:45 +09:00
|
|
|
note = Note
|
|
|
|
{ noteId = Nothing
|
|
|
|
, noteAttrib = luAuthor
|
|
|
|
, noteAudience = Audience
|
2019-05-18 07:42:01 +09:00
|
|
|
{ audienceTo = map encodeRecipRoute recips
|
|
|
|
, audienceBto = []
|
|
|
|
, audienceCc = []
|
|
|
|
, audienceBcc = []
|
|
|
|
, audienceGeneral = []
|
|
|
|
, audienceNonActors = map encodeRecipRoute recipsC
|
2019-03-20 17:07:37 +09:00
|
|
|
}
|
2019-04-21 06:34:45 +09:00
|
|
|
, noteReplyTo = Just uContext
|
|
|
|
, noteContext = Just uContext
|
2019-05-07 10:51:21 +09:00
|
|
|
, notePublished = Nothing
|
2019-06-02 23:41:51 +09:00
|
|
|
, noteSource = msg'
|
|
|
|
, noteContent = contentHtml
|
2019-04-21 06:34:45 +09:00
|
|
|
}
|
|
|
|
ExceptT $ handleOutboxNote hLocal note
|
|
|
|
case elmid of
|
|
|
|
Left e -> do
|
|
|
|
setMessage $ toHtml e
|
2016-05-22 05:01:31 +09:00
|
|
|
defaultLayout $(widgetFile "discussion/top-reply")
|
2019-04-21 06:34:45 +09:00
|
|
|
Right lmid -> do
|
|
|
|
setMessage "Message submitted."
|
|
|
|
redirect $ after lmid
|
2016-05-22 05:01:31 +09:00
|
|
|
|
2016-05-20 07:07:25 +09:00
|
|
|
getReply
|
2019-03-16 01:36:02 +09:00
|
|
|
:: (MessageId -> Route App)
|
|
|
|
-> (MessageId -> Route App)
|
2016-05-20 07:40:54 +09:00
|
|
|
-> AppDB DiscussionId
|
2019-03-16 01:36:02 +09:00
|
|
|
-> MessageId
|
2016-05-20 07:07:25 +09:00
|
|
|
-> Handler Html
|
2019-04-21 06:34:45 +09:00
|
|
|
getReply replyG replyP getdid midParent = do
|
|
|
|
mtn <- runDB $ getNode getdid midParent
|
2016-05-20 07:07:25 +09:00
|
|
|
now <- liftIO getCurrentTime
|
|
|
|
((_result, widget), enctype) <- runFormPost newMessageForm
|
|
|
|
defaultLayout $(widgetFile "discussion/reply")
|
|
|
|
|
|
|
|
postReply
|
2019-04-21 06:34:45 +09:00
|
|
|
:: Text
|
|
|
|
-> [Route App]
|
2019-05-18 07:42:01 +09:00
|
|
|
-> [Route App]
|
2019-04-21 06:34:45 +09:00
|
|
|
-> Route App
|
|
|
|
-> (MessageId -> Route App)
|
2019-03-16 01:36:02 +09:00
|
|
|
-> (MessageId -> Route App)
|
2019-03-20 17:07:37 +09:00
|
|
|
-> (LocalMessageId -> Route App)
|
2016-05-20 07:40:54 +09:00
|
|
|
-> AppDB DiscussionId
|
2019-03-16 01:36:02 +09:00
|
|
|
-> MessageId
|
2016-05-20 07:07:25 +09:00
|
|
|
-> Handler Html
|
2019-05-18 07:42:01 +09:00
|
|
|
postReply hDest recipsA recipsC context replyG replyP after getdid midParent = do
|
2016-05-20 07:07:25 +09:00
|
|
|
((result, widget), enctype) <- runFormPost newMessageForm
|
2019-04-21 06:34:45 +09:00
|
|
|
elmid <- runExceptT $ do
|
|
|
|
msg <- case result of
|
|
|
|
FormMissing -> throwE "Field(s) missing."
|
|
|
|
FormFailure _l -> throwE "Message submission failed, see errors below."
|
|
|
|
FormSuccess nm -> return $ nmContent nm
|
2019-05-21 08:51:06 +09:00
|
|
|
encodeRouteFed <- getEncodeRouteHome
|
2019-04-21 06:34:45 +09:00
|
|
|
encodeRouteLocal <- getEncodeRouteLocal
|
|
|
|
let encodeRecipRoute = l2f hDest . encodeRouteLocal
|
|
|
|
(shrAuthor, uParent) <- do
|
|
|
|
Entity _ p <- requireVerifiedAuth
|
|
|
|
lift $ runDB $ do
|
|
|
|
_m <- get404 midParent
|
|
|
|
shr <- sharerIdent <$> get404 (personIdent p)
|
|
|
|
mlocal <- getBy $ UniqueLocalMessage midParent
|
|
|
|
mremote <- getValBy $ UniqueRemoteMessage midParent
|
|
|
|
parent <- case (mlocal, mremote) of
|
|
|
|
(Nothing, Nothing) -> error "Message with no author"
|
|
|
|
(Just _, Just _) -> error "Message used as both local and remote"
|
|
|
|
(Just (Entity lmidParent lm), Nothing) -> do
|
|
|
|
p <- getJust $ localMessageAuthor lm
|
|
|
|
s <- getJust $ personIdent p
|
|
|
|
lmkhid <- encodeKeyHashid lmidParent
|
|
|
|
return $ encodeRouteFed $ MessageR (sharerIdent s) lmkhid
|
|
|
|
(Nothing, Just rm) -> do
|
|
|
|
i <- getJust $ remoteMessageInstance rm
|
|
|
|
return $ l2f (instanceHost i) (remoteMessageIdent rm)
|
|
|
|
return (shr, parent)
|
2019-06-02 23:41:51 +09:00
|
|
|
let msg' = T.filter (/= '\r') msg
|
|
|
|
contentHtml <- ExceptT . pure $ renderPandocMarkdown msg'
|
2019-04-21 06:34:45 +09:00
|
|
|
let (hLocal, luAuthor) = f2l $ encodeRouteFed $ SharerR shrAuthor
|
|
|
|
uContext = encodeRecipRoute context
|
2019-05-18 07:42:01 +09:00
|
|
|
recips = recipsA ++ recipsC
|
2019-04-21 06:34:45 +09:00
|
|
|
note = Note
|
|
|
|
{ noteId = Nothing
|
|
|
|
, noteAttrib = luAuthor
|
|
|
|
, noteAudience = Audience
|
2019-05-18 07:42:01 +09:00
|
|
|
{ audienceTo = map encodeRecipRoute recips
|
|
|
|
, audienceBto = []
|
|
|
|
, audienceCc = []
|
|
|
|
, audienceBcc = []
|
|
|
|
, audienceGeneral = []
|
|
|
|
, audienceNonActors = map encodeRecipRoute recipsC
|
2019-03-20 17:07:37 +09:00
|
|
|
}
|
2019-04-21 06:34:45 +09:00
|
|
|
, noteReplyTo = Just uParent
|
|
|
|
, noteContext = Just uContext
|
2019-05-07 10:51:21 +09:00
|
|
|
, notePublished = Nothing
|
2019-06-02 23:41:51 +09:00
|
|
|
, noteSource = msg'
|
|
|
|
, noteContent = contentHtml
|
2019-04-21 06:34:45 +09:00
|
|
|
}
|
|
|
|
ExceptT $ handleOutboxNote hLocal note
|
|
|
|
case elmid of
|
|
|
|
Left e -> do
|
|
|
|
setMessage $ toHtml e
|
|
|
|
mtn <- runDB $ getNode getdid midParent
|
|
|
|
now <- liftIO getCurrentTime
|
2016-05-20 07:07:25 +09:00
|
|
|
defaultLayout $(widgetFile "discussion/reply")
|
2019-04-21 06:34:45 +09:00
|
|
|
Right lmid -> do
|
|
|
|
setMessage "Message submitted."
|
|
|
|
redirect $ after lmid
|