S2S: Project Add handler
This commit is contained in:
parent
89185164b8
commit
6ae079a310
5 changed files with 284 additions and 1 deletions
|
@ -19,6 +19,7 @@ module Vervis.Actor.Project
|
||||||
where
|
where
|
||||||
|
|
||||||
import Control.Applicative
|
import Control.Applicative
|
||||||
|
import Control.Exception.Base
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
import Control.Monad.IO.Class
|
import Control.Monad.IO.Class
|
||||||
import Control.Monad.Logger.CallStack
|
import Control.Monad.Logger.CallStack
|
||||||
|
@ -36,9 +37,11 @@ import Data.Time.Clock
|
||||||
import Data.Traversable
|
import Data.Traversable
|
||||||
import Database.Persist
|
import Database.Persist
|
||||||
import Database.Persist.Sql
|
import Database.Persist.Sql
|
||||||
|
import Optics.Core
|
||||||
import Yesod.Persist.Core
|
import Yesod.Persist.Core
|
||||||
|
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
|
import qualified Database.Esqueleto as E
|
||||||
|
|
||||||
import Control.Concurrent.Actor
|
import Control.Concurrent.Actor
|
||||||
import Network.FedURI
|
import Network.FedURI
|
||||||
|
@ -65,6 +68,7 @@ import Vervis.Federation.Util
|
||||||
import Vervis.Foundation
|
import Vervis.Foundation
|
||||||
import Vervis.Model hiding (projectCreate)
|
import Vervis.Model hiding (projectCreate)
|
||||||
import Vervis.Recipient (makeRecipientSet, LocalStageBy (..), Aud (..), collectAudience)
|
import Vervis.Recipient (makeRecipientSet, LocalStageBy (..), Aud (..), collectAudience)
|
||||||
|
import Vervis.RemoteActorStore
|
||||||
import Vervis.Persist.Actor
|
import Vervis.Persist.Actor
|
||||||
import Vervis.Persist.Collab
|
import Vervis.Persist.Collab
|
||||||
import Vervis.Persist.Discussion
|
import Vervis.Persist.Discussion
|
||||||
|
@ -93,6 +97,198 @@ projectAccept
|
||||||
-> ActE (Text, Act (), Next)
|
-> ActE (Text, Act (), Next)
|
||||||
projectAccept = topicAccept projectActor GrantResourceProject
|
projectAccept = topicAccept projectActor GrantResourceProject
|
||||||
|
|
||||||
|
-- Meaning: An actor is adding some object to some target
|
||||||
|
-- Behavior:
|
||||||
|
-- * Verify I'm the target
|
||||||
|
-- * Verify the object is a component, find in DB if local
|
||||||
|
-- * Verify it's not already an active component of mine
|
||||||
|
-- * Verify it's not already in a Add-Accept process waiting for project
|
||||||
|
-- collab to accept too
|
||||||
|
-- * Verify it's not already in an Invite-Accept process waiting for
|
||||||
|
-- component (or its collaborator) to accept too
|
||||||
|
-- * Insert the Add to my inbox
|
||||||
|
-- * Create a Component record in DB
|
||||||
|
-- * Forward the Add to my followers
|
||||||
|
projectAdd
|
||||||
|
:: UTCTime
|
||||||
|
-> ProjectId
|
||||||
|
-> Verse
|
||||||
|
-> AP.Add URIMode
|
||||||
|
-> ActE (Text, Act (), Next)
|
||||||
|
projectAdd now projectID (Verse authorIdMsig body) add = do
|
||||||
|
|
||||||
|
-- Check input
|
||||||
|
component <- do
|
||||||
|
let author = bimap (view _1) (remoteAuthorURI . view _1) authorIdMsig
|
||||||
|
(component, project) <- parseAdd author add
|
||||||
|
unless (project == Left projectID) $ throwE "Add target isn't me"
|
||||||
|
return component
|
||||||
|
|
||||||
|
-- If component is local, find it in our DB
|
||||||
|
-- If component is remote, HTTP GET it, verify it's an actor of a component
|
||||||
|
-- type, and store in our DB (if it's already there, no need for HTTP)
|
||||||
|
--
|
||||||
|
-- NOTE: This is a blocking HTTP GET done right here in the handler,
|
||||||
|
-- which is NOT a good idea. Ideally, it would be done async, and the
|
||||||
|
-- handler result would be sent later in a separate (e.g. Accept) activity.
|
||||||
|
-- But for the PoC level, the current situation will hopefully do.
|
||||||
|
componentDB <-
|
||||||
|
bitraverse
|
||||||
|
(withDBExcept . flip getComponentE "Component not found in DB")
|
||||||
|
(\ u@(ObjURI h lu) -> do
|
||||||
|
instanceID <-
|
||||||
|
lift $ withDB $ either entityKey id <$> insertBy' (Instance h)
|
||||||
|
result <-
|
||||||
|
ExceptT $ first (T.pack . displayException) <$>
|
||||||
|
fetchRemoteActor' instanceID h lu
|
||||||
|
case result of
|
||||||
|
Left Nothing -> throwE "Target @id mismatch"
|
||||||
|
Left (Just err) -> throwE $ T.pack $ displayException err
|
||||||
|
Right Nothing -> throwE "Target isn't an actor"
|
||||||
|
Right (Just actor) -> do
|
||||||
|
case remoteActorType $ entityVal actor of
|
||||||
|
AP.ActorTypeRepo -> pure ()
|
||||||
|
AP.ActorTypeTicketTracker -> pure ()
|
||||||
|
AP.ActorTypePatchTracker -> pure ()
|
||||||
|
_ -> throwE "Remote component type isn't repo/tt/pt"
|
||||||
|
return $ entityKey actor
|
||||||
|
)
|
||||||
|
component
|
||||||
|
|
||||||
|
maybeNew <- withDBExcept $ do
|
||||||
|
|
||||||
|
-- Grab me from DB
|
||||||
|
(project, actorRecip) <- lift $ do
|
||||||
|
p <- getJust projectID
|
||||||
|
(p,) <$> getJust (projectActor p)
|
||||||
|
|
||||||
|
-- Find existing Component records I have for this component
|
||||||
|
componentIDs <- lift $ getExistingComponents componentDB
|
||||||
|
|
||||||
|
-- Grab all the enabled ones, make sure none are enabled, and even if
|
||||||
|
-- any are enabled, make sure there's at most one (otherwise it's a
|
||||||
|
-- bug)
|
||||||
|
byEnabled <-
|
||||||
|
lift $ for componentIDs $ \ (componentID, _) ->
|
||||||
|
isJust <$> runMaybeT (tryComponentEnabled componentID)
|
||||||
|
case length $ filter id byEnabled of
|
||||||
|
0 -> return ()
|
||||||
|
1 -> throwE "I already have a ComponentEnable for this component"
|
||||||
|
_ -> error "Multiple ComponentEnable for a component"
|
||||||
|
|
||||||
|
-- Verify none of the Component records are already in
|
||||||
|
-- Add-waiting-for-project or Invite-waiting-for-component state
|
||||||
|
anyStarted <-
|
||||||
|
lift $ runMaybeT $ asum $
|
||||||
|
map (\ (componentID, identID) ->
|
||||||
|
tryComponentAddAccept componentID identID <|>
|
||||||
|
tryComponentInviteAccept componentID
|
||||||
|
)
|
||||||
|
componentIDs
|
||||||
|
unless (isNothing anyStarted) $
|
||||||
|
throwE
|
||||||
|
"One of the Component records is already in Add-Accept or \
|
||||||
|
\Invite-Accept state"
|
||||||
|
|
||||||
|
-- Insert the Add to my inbox
|
||||||
|
mractid <- lift $ insertToInbox now authorIdMsig body (actorInbox actorRecip) False
|
||||||
|
lift $ for mractid $ \ addDB -> do
|
||||||
|
|
||||||
|
-- Create a Component record in DB
|
||||||
|
insertComponent componentDB addDB
|
||||||
|
|
||||||
|
return $ projectActor project
|
||||||
|
|
||||||
|
case maybeNew of
|
||||||
|
Nothing -> done "I already have this activity in my inbox"
|
||||||
|
Just actorID -> do
|
||||||
|
projectHash <- encodeKeyHashid projectID
|
||||||
|
let sieve =
|
||||||
|
makeRecipientSet
|
||||||
|
[]
|
||||||
|
[LocalStageProjectFollowers projectHash]
|
||||||
|
forwardActivity
|
||||||
|
authorIdMsig body (LocalActorProject projectID) actorID sieve
|
||||||
|
done
|
||||||
|
"Recorded a Component record; Inserted the Add to inbox; \
|
||||||
|
\Forwarded to followers if addressed"
|
||||||
|
|
||||||
|
where
|
||||||
|
|
||||||
|
getExistingComponents (Left (ComponentRepo (Entity repoID _))) =
|
||||||
|
fmap (map $ bimap E.unValue (Left . E.unValue)) $
|
||||||
|
E.select $ E.from $ \ (ident `E.InnerJoin` local `E.InnerJoin` comp) -> do
|
||||||
|
E.on $ local E.^. ComponentLocalComponent E.==. comp E.^. ComponentId
|
||||||
|
E.on $ ident E.^. ComponentLocalRepoComponent E.==. local E.^. ComponentLocalId
|
||||||
|
E.where_ $
|
||||||
|
ident E.^. ComponentLocalRepoRepo E.==. E.val repoID E.&&.
|
||||||
|
comp E.^. ComponentProject E.==. E.val projectID
|
||||||
|
return (comp E.^. ComponentId, local E.^. ComponentLocalId)
|
||||||
|
getExistingComponents (Left (ComponentDeck (Entity deckID _))) =
|
||||||
|
fmap (map $ bimap E.unValue (Left . E.unValue)) $
|
||||||
|
E.select $ E.from $ \ (ident `E.InnerJoin` local `E.InnerJoin` comp) -> do
|
||||||
|
E.on $ local E.^. ComponentLocalComponent E.==. comp E.^. ComponentId
|
||||||
|
E.on $ ident E.^. ComponentLocalDeckComponent E.==. local E.^. ComponentLocalId
|
||||||
|
E.where_ $
|
||||||
|
ident E.^. ComponentLocalDeckDeck E.==. E.val deckID E.&&.
|
||||||
|
comp E.^. ComponentProject E.==. E.val projectID
|
||||||
|
return (comp E.^. ComponentId, local E.^. ComponentLocalId)
|
||||||
|
getExistingComponents (Left (ComponentLoom (Entity loomID _))) =
|
||||||
|
fmap (map $ bimap E.unValue (Left . E.unValue)) $
|
||||||
|
E.select $ E.from $ \ (ident `E.InnerJoin` local `E.InnerJoin` comp) -> do
|
||||||
|
E.on $ local E.^. ComponentLocalComponent E.==. comp E.^. ComponentId
|
||||||
|
E.on $ ident E.^. ComponentLocalLoomComponent E.==. local E.^. ComponentLocalId
|
||||||
|
E.where_ $
|
||||||
|
ident E.^. ComponentLocalLoomLoom E.==. E.val loomID E.&&.
|
||||||
|
comp E.^. ComponentProject E.==. E.val projectID
|
||||||
|
return (comp E.^. ComponentId, local E.^. ComponentLocalId)
|
||||||
|
getExistingComponents (Right remoteActorID) =
|
||||||
|
fmap (map $ bimap E.unValue (Right . E.unValue)) $
|
||||||
|
E.select $ E.from $ \ (ident `E.InnerJoin` comp) -> do
|
||||||
|
E.on $ ident E.^. ComponentRemoteComponent E.==. comp E.^. ComponentId
|
||||||
|
E.where_ $
|
||||||
|
ident E.^. ComponentRemoteActor E.==. E.val remoteActorID E.&&.
|
||||||
|
comp E.^. ComponentProject E.==. E.val projectID
|
||||||
|
return (comp E.^. ComponentId, ident E.^. ComponentRemoteId)
|
||||||
|
|
||||||
|
tryComponentEnabled componentID =
|
||||||
|
const () <$> MaybeT (getBy $ UniqueComponentEnable componentID)
|
||||||
|
|
||||||
|
tryComponentAddAccept componentID identID = do
|
||||||
|
_ <- MaybeT $ getBy $ UniqueComponentOriginAdd componentID
|
||||||
|
case identID of
|
||||||
|
Left localID ->
|
||||||
|
const () <$>
|
||||||
|
MaybeT (getBy $ UniqueComponentAcceptLocal localID)
|
||||||
|
Right remoteID ->
|
||||||
|
const () <$>
|
||||||
|
MaybeT (getBy $ UniqueComponentAcceptRemote remoteID)
|
||||||
|
|
||||||
|
tryComponentInviteAccept componentID = do
|
||||||
|
originID <- MaybeT $ getKeyBy $ UniqueComponentOriginInvite componentID
|
||||||
|
const () <$> MaybeT (getBy $ UniqueComponentProjectAccept originID)
|
||||||
|
|
||||||
|
insertComponent componentDB addDB = do
|
||||||
|
componentID <- insert $ Component projectID
|
||||||
|
originID <- insert $ ComponentOriginAdd componentID
|
||||||
|
case addDB of
|
||||||
|
Left (_, _, addID) ->
|
||||||
|
insert_ $ ComponentGestureLocal originID addID
|
||||||
|
Right (author, _, addID) ->
|
||||||
|
insert_ $ ComponentGestureRemote originID (remoteAuthorId author) addID
|
||||||
|
case componentDB of
|
||||||
|
Left l -> do
|
||||||
|
identID <- insert $ ComponentLocal componentID
|
||||||
|
case l of
|
||||||
|
ComponentRepo (Entity repoID _) ->
|
||||||
|
insert_ $ ComponentLocalRepo identID repoID
|
||||||
|
ComponentDeck (Entity deckID _) ->
|
||||||
|
insert_ $ ComponentLocalDeck identID deckID
|
||||||
|
ComponentLoom (Entity loomID _) ->
|
||||||
|
insert_ $ ComponentLocalLoom identID loomID
|
||||||
|
Right remoteActorID ->
|
||||||
|
insert_ $ ComponentRemote componentID remoteActorID
|
||||||
|
|
||||||
-- Meaning: Someone has created a project with my ID URI
|
-- Meaning: Someone has created a project with my ID URI
|
||||||
-- Behavior:
|
-- Behavior:
|
||||||
-- * Verify I'm in a just-been-created state
|
-- * Verify I'm in a just-been-created state
|
||||||
|
@ -368,6 +564,7 @@ projectBehavior :: UTCTime -> ProjectId -> VerseExt -> ActE (Text, Act (), Next)
|
||||||
projectBehavior now projectID (Left verse@(Verse _authorIdMsig body)) =
|
projectBehavior now projectID (Left verse@(Verse _authorIdMsig body)) =
|
||||||
case AP.activitySpecific $ actbActivity body of
|
case AP.activitySpecific $ actbActivity body of
|
||||||
AP.AcceptActivity accept -> projectAccept now projectID verse accept
|
AP.AcceptActivity accept -> projectAccept now projectID verse accept
|
||||||
|
AP.AddActivity add -> projectAdd now projectID verse add
|
||||||
AP.CreateActivity create -> projectCreate now projectID verse create
|
AP.CreateActivity create -> projectCreate now projectID verse create
|
||||||
AP.FollowActivity follow -> projectFollow now projectID verse follow
|
AP.FollowActivity follow -> projectFollow now projectID verse follow
|
||||||
AP.InviteActivity invite -> projectInvite now projectID verse invite
|
AP.InviteActivity invite -> projectInvite now projectID verse invite
|
||||||
|
|
|
@ -28,6 +28,7 @@ module Vervis.Data.Collab
|
||||||
, parseAccept
|
, parseAccept
|
||||||
, parseReject
|
, parseReject
|
||||||
, parseRemove
|
, parseRemove
|
||||||
|
, parseAdd
|
||||||
|
|
||||||
, grantResourceActorID
|
, grantResourceActorID
|
||||||
|
|
||||||
|
@ -44,6 +45,8 @@ module Vervis.Data.Collab
|
||||||
, getGrantResource404
|
, getGrantResource404
|
||||||
|
|
||||||
, grantResourceLocalActor
|
, grantResourceLocalActor
|
||||||
|
|
||||||
|
, ComponentBy (..)
|
||||||
)
|
)
|
||||||
where
|
where
|
||||||
|
|
||||||
|
@ -269,6 +272,60 @@ parseRemove sender (AP.Remove object origin) =
|
||||||
(,) <$> nameExceptT "Remove origin" (parseTopic origin)
|
(,) <$> nameExceptT "Remove origin" (parseTopic origin)
|
||||||
<*> nameExceptT "Remove object" (parseRecipient sender object)
|
<*> nameExceptT "Remove object" (parseRecipient sender object)
|
||||||
|
|
||||||
|
parseAdd
|
||||||
|
:: StageRoute Env ~ Route App
|
||||||
|
=> Either (LocalActorBy Key) FedURI
|
||||||
|
-> AP.Add URIMode
|
||||||
|
-> ActE
|
||||||
|
( Either (ComponentBy Key) FedURI
|
||||||
|
, Either ProjectId FedURI
|
||||||
|
)
|
||||||
|
parseAdd sender (AP.Add object target) = do
|
||||||
|
result@(component, project) <-
|
||||||
|
(,) <$> nameExceptT "Add.object" (parseComponent' object)
|
||||||
|
<*> nameExceptT "Add.target" (parseProject target)
|
||||||
|
case result of
|
||||||
|
(Right u, Right v) | u == v -> throwE "Object and target are the same"
|
||||||
|
_ -> pure ()
|
||||||
|
when (sender == first componentActor component) $
|
||||||
|
throwE "Sender and component are the same"
|
||||||
|
when (sender == first LocalActorProject project) $
|
||||||
|
throwE "Sender and project are the same"
|
||||||
|
return result
|
||||||
|
where
|
||||||
|
componentActor (ComponentRepo r) = LocalActorRepo r
|
||||||
|
componentActor (ComponentDeck d) = LocalActorDeck d
|
||||||
|
componentActor (ComponentLoom l) = LocalActorLoom l
|
||||||
|
parseComponent' (Right _) = throwE "Not a component URI"
|
||||||
|
parseComponent' (Left u) = do
|
||||||
|
routeOrRemote <- parseFedURI u
|
||||||
|
bitraverse
|
||||||
|
(\ route -> do
|
||||||
|
componentHash <-
|
||||||
|
fromMaybeE
|
||||||
|
(parseComponent route)
|
||||||
|
"Not a component route"
|
||||||
|
unhashComponentE
|
||||||
|
componentHash
|
||||||
|
"Contains invalid hashid"
|
||||||
|
)
|
||||||
|
pure
|
||||||
|
routeOrRemote
|
||||||
|
where
|
||||||
|
parseComponent (RepoR r) = Just $ ComponentRepo r
|
||||||
|
parseComponent (DeckR d) = Just $ ComponentDeck d
|
||||||
|
parseComponent (LoomR l) = Just $ ComponentLoom l
|
||||||
|
parseComponent _ = Nothing
|
||||||
|
parseProject u = do
|
||||||
|
routeOrRemote <- parseFedURI u
|
||||||
|
bitraverse
|
||||||
|
(\case
|
||||||
|
ProjectR j -> WAP.decodeKeyHashidE j "Inavlid hashid"
|
||||||
|
_ -> throwE "Not a project route"
|
||||||
|
)
|
||||||
|
pure
|
||||||
|
routeOrRemote
|
||||||
|
|
||||||
grantResourceActorID :: GrantResourceBy Identity -> ActorId
|
grantResourceActorID :: GrantResourceBy Identity -> ActorId
|
||||||
grantResourceActorID (GrantResourceRepo (Identity r)) = repoActor r
|
grantResourceActorID (GrantResourceRepo (Identity r)) = repoActor r
|
||||||
grantResourceActorID (GrantResourceDeck (Identity d)) = deckActor d
|
grantResourceActorID (GrantResourceDeck (Identity d)) = deckActor d
|
||||||
|
@ -354,3 +411,26 @@ grantResourceLocalActor (GrantResourceRepo r) = LocalActorRepo r
|
||||||
grantResourceLocalActor (GrantResourceDeck d) = LocalActorDeck d
|
grantResourceLocalActor (GrantResourceDeck d) = LocalActorDeck d
|
||||||
grantResourceLocalActor (GrantResourceLoom l) = LocalActorLoom l
|
grantResourceLocalActor (GrantResourceLoom l) = LocalActorLoom l
|
||||||
grantResourceLocalActor (GrantResourceProject l) = LocalActorProject l
|
grantResourceLocalActor (GrantResourceProject l) = LocalActorProject l
|
||||||
|
|
||||||
|
data ComponentBy f
|
||||||
|
= ComponentRepo (f Repo)
|
||||||
|
| ComponentDeck (f Deck)
|
||||||
|
| ComponentLoom (f Loom)
|
||||||
|
deriving (Generic, FunctorB, TraversableB, ConstraintsB)
|
||||||
|
|
||||||
|
deriving instance AllBF Eq f ComponentBy => Eq (ComponentBy f)
|
||||||
|
|
||||||
|
unhashComponentPure ctx = f
|
||||||
|
where
|
||||||
|
f (ComponentRepo r) =
|
||||||
|
ComponentRepo <$> decodeKeyHashidPure ctx r
|
||||||
|
f (ComponentDeck d) =
|
||||||
|
ComponentDeck <$> decodeKeyHashidPure ctx d
|
||||||
|
f (ComponentLoom l) =
|
||||||
|
ComponentLoom <$> decodeKeyHashidPure ctx l
|
||||||
|
|
||||||
|
unhashComponent c = do
|
||||||
|
ctx <- asksEnv WAP.stageHashidsContext
|
||||||
|
return $ unhashComponentPure ctx c
|
||||||
|
|
||||||
|
unhashComponentE c e = ExceptT $ maybe (Left e) Right <$> unhashComponent c
|
||||||
|
|
|
@ -2962,6 +2962,8 @@ changes hLocal ctx =
|
||||||
, addEntities model_542_component
|
, addEntities model_542_component
|
||||||
-- 543
|
-- 543
|
||||||
, addFieldPrimRequired "RemoteActor" ("" :: Text) "type"
|
, addFieldPrimRequired "RemoteActor" ("" :: Text) "type"
|
||||||
|
-- 544
|
||||||
|
, removeField "ComponentRemote" "object"
|
||||||
]
|
]
|
||||||
|
|
||||||
migrateDB
|
migrateDB
|
||||||
|
|
|
@ -17,6 +17,7 @@ module Vervis.Persist.Collab
|
||||||
( getCollabTopic
|
( getCollabTopic
|
||||||
, getCollabTopic'
|
, getCollabTopic'
|
||||||
, getGrantRecip
|
, getGrantRecip
|
||||||
|
, getComponentE
|
||||||
, getTopicGrants
|
, getTopicGrants
|
||||||
, getTopicInvites
|
, getTopicInvites
|
||||||
, getTopicJoins
|
, getTopicJoins
|
||||||
|
@ -99,6 +100,10 @@ getCollabTopic' collabID = do
|
||||||
|
|
||||||
getGrantRecip (GrantRecipPerson k) e = GrantRecipPerson <$> getEntityE k e
|
getGrantRecip (GrantRecipPerson k) e = GrantRecipPerson <$> getEntityE k e
|
||||||
|
|
||||||
|
getComponentE (ComponentRepo k) e = ComponentRepo <$> getEntityE k e
|
||||||
|
getComponentE (ComponentDeck k) e = ComponentDeck <$> getEntityE k e
|
||||||
|
getComponentE (ComponentLoom k) e = ComponentLoom <$> getEntityE k e
|
||||||
|
|
||||||
getTopicGrants
|
getTopicGrants
|
||||||
:: ( MonadIO m
|
:: ( MonadIO m
|
||||||
, PersistRecordBackend topic SqlBackend
|
, PersistRecordBackend topic SqlBackend
|
||||||
|
|
|
@ -830,7 +830,6 @@ ComponentLocalLoom
|
||||||
ComponentRemote
|
ComponentRemote
|
||||||
component ComponentId
|
component ComponentId
|
||||||
actor RemoteActorId
|
actor RemoteActorId
|
||||||
object RemoteObjectId
|
|
||||||
|
|
||||||
UniqueComponentRemote component
|
UniqueComponentRemote component
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue