2016-05-19 16:58:23 +00:00
|
|
|
{- This file is part of Vervis.
|
|
|
|
-
|
2019-03-15 16:36:02 +00:00
|
|
|
- Written in 2016, 2019 by fr33domlover <fr33domlover@riseup.net>.
|
2016-05-19 16:58:23 +00: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-15 16:36:02 +00:00
|
|
|
, getDiscussionMessage
|
2016-05-21 20:01:31 +00:00
|
|
|
, getTopReply
|
|
|
|
, postTopReply
|
2016-05-19 22:07:25 +00:00
|
|
|
, getReply
|
|
|
|
, postReply
|
2016-05-19 16:58:23 +00:00
|
|
|
)
|
|
|
|
where
|
|
|
|
|
|
|
|
import Prelude
|
|
|
|
|
2019-03-15 16:36:02 +00:00
|
|
|
import Control.Monad
|
2016-05-19 16:58:23 +00:00
|
|
|
import Control.Monad.IO.Class (liftIO)
|
2019-03-22 20:46:42 +00:00
|
|
|
import Data.Maybe
|
2016-05-19 16:58:23 +00:00
|
|
|
import Data.Time.Clock (getCurrentTime)
|
2016-05-19 22:07:25 +00:00
|
|
|
import Database.Persist
|
2019-03-22 20:46:42 +00:00
|
|
|
import Database.Persist.Sql
|
|
|
|
import Data.Traversable
|
2016-05-19 16:58:23 +00:00
|
|
|
import Text.Blaze.Html (Html)
|
2016-05-19 22:07:25 +00:00
|
|
|
import Yesod.Auth (requireAuthId)
|
2019-03-22 20:46:42 +00:00
|
|
|
import Yesod.Core
|
2019-03-15 16:36:02 +00:00
|
|
|
import Yesod.Core.Handler
|
2016-05-19 22:07:25 +00:00
|
|
|
import Yesod.Form.Functions (runFormPost)
|
|
|
|
import Yesod.Form.Types (FormResult (..))
|
2016-05-19 16:58:23 +00:00
|
|
|
import Yesod.Persist.Core (runDB, get404, getBy404)
|
|
|
|
|
2019-03-20 08:07:37 +00:00
|
|
|
import Network.FedURI
|
2019-03-22 20:46:42 +00:00
|
|
|
import Web.ActivityPub
|
|
|
|
import Yesod.FedURI
|
|
|
|
|
|
|
|
import Database.Persist.Local
|
|
|
|
import Yesod.Persist.Local
|
2019-03-20 08:07:37 +00:00
|
|
|
|
|
|
|
import Vervis.Discussion
|
2016-05-19 22:07:25 +00:00
|
|
|
import Vervis.Form.Discussion
|
2019-03-22 20:46:42 +00:00
|
|
|
import Vervis.Foundation
|
2016-05-19 16:58:23 +00:00
|
|
|
import Vervis.Model
|
2019-03-22 20:46:42 +00:00
|
|
|
import Vervis.Model.Ident
|
|
|
|
import Vervis.Settings
|
2016-05-19 16:58:23 +00:00
|
|
|
import Vervis.Widget.Discussion
|
|
|
|
|
2016-05-21 21:27:12 +00:00
|
|
|
getDiscussion
|
2019-03-15 16:36:02 +00:00
|
|
|
:: (MessageId -> Route App)
|
|
|
|
-> Route App
|
|
|
|
-> AppDB DiscussionId
|
|
|
|
-> Handler Html
|
2016-05-21 21:27:12 +00:00
|
|
|
getDiscussion reply topic getdid =
|
|
|
|
defaultLayout $ discussionW getdid topic reply
|
2016-05-19 16:58:23 +00:00
|
|
|
|
2019-03-20 08:07:37 +00: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
|
|
|
|
i <- getJust $ remoteSharerInstance rs
|
|
|
|
return $ MessageTreeNodeRemote $
|
|
|
|
l2f (instanceHost i) (remoteSharerIdent rs)
|
|
|
|
return $ MessageTreeNode mid m author
|
|
|
|
|
2019-03-22 20:46:42 +00:00
|
|
|
{-
|
2019-03-21 19:06:52 +00: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-22 20:46:42 +00:00
|
|
|
-}
|
2019-03-21 19:06:52 +00:00
|
|
|
|
2019-03-22 20:46:42 +00:00
|
|
|
getDiscussionMessage :: ShrIdent -> LocalMessageId -> Handler TypedContent
|
|
|
|
getDiscussionMessage shr lmid = selectRep $ provideAP $ runDB $ do
|
|
|
|
sid <- getKeyBy404 $ UniqueSharer shr
|
|
|
|
pid <- getKeyBy404 $ UniquePersonIdent sid
|
|
|
|
lm <- get404 lmid
|
|
|
|
unless (localMessageAuthor lm == pid) notFound
|
|
|
|
m <- getJust $ localMessageRest lm
|
|
|
|
route2fed <- getEncodeRouteFed
|
|
|
|
encodeHid <- getsYesod appHashidEncode
|
2019-03-23 02:57:34 +00:00
|
|
|
(uRecip, uContext) <- do
|
2019-03-22 20:46:42 +00:00
|
|
|
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
|
2019-03-23 02:57:34 +00:00
|
|
|
let shr = sharerIdent s
|
|
|
|
prj = projectIdent j
|
|
|
|
return
|
|
|
|
( route2fed $ ProjectR shr prj
|
|
|
|
, route2fed $ TicketR shr prj $ ticketNumber t
|
|
|
|
)
|
2019-03-22 20:46:42 +00:00
|
|
|
(Nothing, Just rd) -> do
|
2019-03-23 02:57:34 +00:00
|
|
|
let iid = remoteDiscussionInstance rd
|
|
|
|
i <- getJust iid
|
|
|
|
rs <- getJust $ remoteDiscussionSharer rd
|
|
|
|
unless (iid == remoteSharerInstance rs) $
|
|
|
|
fail "RemoteDiscussion and its sharer on different hosts"
|
|
|
|
return
|
|
|
|
( l2f (instanceHost i) (remoteSharerIdent rs)
|
|
|
|
, l2f (instanceHost i) (remoteDiscussionIdent rd)
|
|
|
|
)
|
2019-03-22 20:46:42 +00:00
|
|
|
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
|
|
|
|
let lmhidParent = encodeHid $ fromSqlKey lmidParent
|
|
|
|
return $ route2fed $ MessageR (sharerIdent s) lmhidParent
|
|
|
|
(Nothing, Just rmParent) -> do
|
|
|
|
rs <- getJust $ remoteMessageAuthor rmParent
|
|
|
|
i <- getJust $ remoteSharerInstance rs
|
|
|
|
return $ l2f (instanceHost i) (remoteSharerIdent rs)
|
|
|
|
|
|
|
|
host <- getsYesod $ appInstanceHost . appSettings
|
|
|
|
route2local <- getEncodeRouteLocal
|
|
|
|
let lmhid = encodeHid $ fromSqlKey lmid
|
|
|
|
return $ Doc host Note
|
2019-03-23 02:05:30 +00:00
|
|
|
{ noteId = Just $ route2local $ MessageR shr lmhid
|
2019-03-22 20:46:42 +00:00
|
|
|
, noteAttrib = route2local $ SharerR shr
|
2019-03-23 02:57:34 +00:00
|
|
|
, noteAudience = deliverTo uRecip
|
2019-03-22 20:46:42 +00:00
|
|
|
, noteReplyTo = Just $ fromMaybe uContext muParent
|
|
|
|
, noteContext = Just uContext
|
|
|
|
, notePublished = Just $ messageCreated m
|
|
|
|
, noteContent = messageContent m
|
|
|
|
}
|
2016-05-19 22:07:25 +00:00
|
|
|
|
2016-05-21 20:01:31 +00:00
|
|
|
getTopReply :: Route App -> Handler Html
|
|
|
|
getTopReply replyP = do
|
|
|
|
((_result, widget), enctype) <- runFormPost newMessageForm
|
|
|
|
defaultLayout $(widgetFile "discussion/top-reply")
|
|
|
|
|
|
|
|
postTopReply
|
|
|
|
:: Route App
|
2019-03-20 08:07:37 +00:00
|
|
|
-> (LocalMessageId -> Route App)
|
2016-05-21 20:01:31 +00:00
|
|
|
-> AppDB DiscussionId
|
|
|
|
-> Handler Html
|
|
|
|
postTopReply replyP after getdid = do
|
|
|
|
((result, widget), enctype) <- runFormPost newMessageForm
|
|
|
|
now <- liftIO getCurrentTime
|
|
|
|
case result of
|
|
|
|
FormSuccess nm -> do
|
|
|
|
author <- requireAuthId
|
|
|
|
mnum <- runDB $ do
|
|
|
|
did <- getdid
|
2019-03-20 08:07:37 +00:00
|
|
|
mid <- insert Message
|
|
|
|
{ messageCreated = now
|
|
|
|
, messageContent = nmContent nm
|
|
|
|
, messageParent = Nothing
|
|
|
|
, messageRoot = did
|
|
|
|
}
|
|
|
|
lmid <- insert LocalMessage
|
|
|
|
{ localMessageAuthor = author
|
|
|
|
, localMessageRest = mid
|
|
|
|
}
|
|
|
|
return lmid
|
2016-05-21 20:01:31 +00:00
|
|
|
setMessage "Message submitted."
|
|
|
|
redirect $ after mnum
|
|
|
|
FormMissing -> do
|
|
|
|
setMessage "Field(s) missing."
|
|
|
|
defaultLayout $(widgetFile "discussion/top-reply")
|
|
|
|
FormFailure _l -> do
|
|
|
|
setMessage "Message submission failed, see errors below."
|
|
|
|
defaultLayout $(widgetFile "discussion/top-reply")
|
|
|
|
|
2016-05-19 22:07:25 +00:00
|
|
|
getReply
|
2019-03-15 16:36:02 +00:00
|
|
|
:: (MessageId -> Route App)
|
|
|
|
-> (MessageId -> Route App)
|
2016-05-19 22:40:54 +00:00
|
|
|
-> AppDB DiscussionId
|
2019-03-15 16:36:02 +00:00
|
|
|
-> MessageId
|
2016-05-19 22:07:25 +00:00
|
|
|
-> Handler Html
|
2019-03-15 16:36:02 +00:00
|
|
|
getReply replyG replyP getdid mid = do
|
2019-03-20 08:07:37 +00:00
|
|
|
mtn <- runDB $ getNode getdid mid
|
2016-05-19 22:07:25 +00:00
|
|
|
now <- liftIO getCurrentTime
|
|
|
|
((_result, widget), enctype) <- runFormPost newMessageForm
|
|
|
|
defaultLayout $(widgetFile "discussion/reply")
|
|
|
|
|
|
|
|
postReply
|
2019-03-15 16:36:02 +00:00
|
|
|
:: (MessageId -> Route App)
|
|
|
|
-> (MessageId -> Route App)
|
2019-03-20 08:07:37 +00:00
|
|
|
-> (LocalMessageId -> Route App)
|
2016-05-19 22:40:54 +00:00
|
|
|
-> AppDB DiscussionId
|
2019-03-15 16:36:02 +00:00
|
|
|
-> MessageId
|
2016-05-19 22:07:25 +00:00
|
|
|
-> Handler Html
|
2019-03-15 16:36:02 +00:00
|
|
|
postReply replyG replyP after getdid mid = do
|
2016-05-19 22:07:25 +00:00
|
|
|
((result, widget), enctype) <- runFormPost newMessageForm
|
|
|
|
now <- liftIO getCurrentTime
|
|
|
|
case result of
|
|
|
|
FormSuccess nm -> do
|
|
|
|
author <- requireAuthId
|
2019-03-15 16:36:02 +00:00
|
|
|
msgid <- runDB $ do
|
2016-05-19 22:40:54 +00:00
|
|
|
did <- getdid
|
2019-03-15 16:36:02 +00:00
|
|
|
parent <- do
|
|
|
|
message <- get404 mid
|
|
|
|
unless (messageRoot message == did) notFound
|
|
|
|
return mid
|
2019-03-20 08:07:37 +00:00
|
|
|
mid <- insert Message
|
|
|
|
{ messageCreated = now
|
|
|
|
, messageContent = nmContent nm
|
|
|
|
, messageParent = Just parent
|
|
|
|
, messageRoot = did
|
|
|
|
}
|
|
|
|
lmid <- insert LocalMessage
|
|
|
|
{ localMessageAuthor = author
|
|
|
|
, localMessageRest = mid
|
|
|
|
}
|
|
|
|
return lmid
|
2016-05-19 22:07:25 +00:00
|
|
|
setMessage "Message submitted."
|
2019-03-15 16:36:02 +00:00
|
|
|
redirect $ after msgid
|
2016-05-19 22:07:25 +00:00
|
|
|
FormMissing -> do
|
|
|
|
setMessage "Field(s) missing."
|
2019-03-20 08:07:37 +00:00
|
|
|
mtn <- runDB $ getNode getdid mid
|
2016-05-19 22:07:25 +00:00
|
|
|
defaultLayout $(widgetFile "discussion/reply")
|
|
|
|
FormFailure _l -> do
|
|
|
|
setMessage "Message submission failed, see errors below."
|
2019-03-20 08:07:37 +00:00
|
|
|
mtn <- runDB $ getNode getdid mid
|
2016-05-19 22:07:25 +00:00
|
|
|
defaultLayout $(widgetFile "discussion/reply")
|