2024-06-13 20:00:58 +00:00
|
|
|
{- git-annex cluster log
|
|
|
|
-
|
|
|
|
- Copyright 2024 Joey Hess <id@joeyh.name>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU AGPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
{-# LANGUAGE OverloadedStrings, TupleSections #-}
|
|
|
|
|
|
|
|
module Logs.Cluster (
|
2024-06-14 15:07:31 +00:00
|
|
|
ClusterUUID,
|
2024-06-13 20:00:58 +00:00
|
|
|
ClusterNodeUUID(..),
|
|
|
|
getClusters,
|
|
|
|
recordCluster,
|
|
|
|
) where
|
|
|
|
|
|
|
|
import qualified Annex
|
|
|
|
import Annex.Common
|
|
|
|
import qualified Annex.Branch
|
|
|
|
import Types.Cluster
|
|
|
|
import Logs
|
|
|
|
import Logs.UUIDBased
|
|
|
|
import Logs.MapLog
|
|
|
|
|
|
|
|
import qualified Data.Set as S
|
|
|
|
import qualified Data.Map as M
|
|
|
|
import Data.ByteString.Builder
|
|
|
|
import qualified Data.Attoparsec.ByteString as A
|
|
|
|
import qualified Data.Attoparsec.ByteString.Char8 as A8
|
|
|
|
import qualified Data.ByteString.Lazy as L
|
|
|
|
|
|
|
|
getClusters :: Annex Clusters
|
2024-06-14 15:16:01 +00:00
|
|
|
getClusters = maybe loadClusters return =<< Annex.getState Annex.clusters
|
|
|
|
|
|
|
|
loadClusters :: Annex Clusters
|
|
|
|
loadClusters = do
|
2024-06-14 15:07:31 +00:00
|
|
|
m <- convclusteruuids . M.map value . fromMapLog . parseClusterLog
|
2024-06-13 20:00:58 +00:00
|
|
|
<$> Annex.Branch.get clusterLog
|
2024-06-14 15:16:01 +00:00
|
|
|
let clusters = Clusters
|
2024-06-13 20:00:58 +00:00
|
|
|
{ clusterUUIDs = m
|
|
|
|
, clusterNodeUUIDs = M.foldlWithKey inverter mempty m
|
|
|
|
}
|
2024-06-14 15:16:01 +00:00
|
|
|
Annex.changeState $ \s -> s { Annex.clusters = Just clusters }
|
|
|
|
return clusters
|
2024-06-13 20:00:58 +00:00
|
|
|
where
|
2024-06-14 15:07:31 +00:00
|
|
|
convclusteruuids :: M.Map UUID (S.Set ClusterNodeUUID) -> M.Map ClusterUUID (S.Set ClusterNodeUUID)
|
|
|
|
convclusteruuids = M.fromList
|
|
|
|
. mapMaybe (\(mk, v) -> (, v) <$> mk)
|
|
|
|
. M.toList . M.mapKeys mkClusterUUID
|
2024-06-13 20:00:58 +00:00
|
|
|
inverter m k v = M.unionWith (<>) m
|
|
|
|
(M.fromList (map (, S.singleton k) (S.toList v)))
|
|
|
|
|
|
|
|
recordCluster :: ClusterUUID -> S.Set ClusterNodeUUID -> Annex ()
|
|
|
|
recordCluster clusteruuid nodeuuids = do
|
|
|
|
-- If a private UUID has been configured as a cluster node,
|
|
|
|
-- avoid leaking it into the git-annex log.
|
|
|
|
privateuuids <- annexPrivateRepos <$> Annex.getGitConfig
|
|
|
|
let nodeuuids' = S.filter
|
|
|
|
(\(ClusterNodeUUID n) -> S.notMember n privateuuids)
|
|
|
|
nodeuuids
|
|
|
|
|
|
|
|
c <- currentVectorClock
|
|
|
|
Annex.Branch.change (Annex.Branch.RegardingUUID [fromClusterUUID clusteruuid]) clusterLog $
|
|
|
|
(buildLogNew buildClusterNodeList)
|
2024-06-14 18:21:50 +00:00
|
|
|
. changeLog c (fromClusterUUID clusteruuid) nodeuuids'
|
2024-06-13 20:00:58 +00:00
|
|
|
. parseClusterLog
|
|
|
|
|
|
|
|
buildClusterNodeList :: S.Set ClusterNodeUUID -> Builder
|
|
|
|
buildClusterNodeList = assemble
|
|
|
|
. map (buildUUID . fromClusterNodeUUID)
|
|
|
|
. S.toList
|
|
|
|
where
|
|
|
|
assemble [] = mempty
|
|
|
|
assemble (x:[]) = x
|
|
|
|
assemble (x:y:l) = x <> " " <> assemble (y:l)
|
|
|
|
|
|
|
|
parseClusterLog :: L.ByteString -> Log (S.Set ClusterNodeUUID)
|
|
|
|
parseClusterLog = parseLogNew parseClusterNodeList
|
|
|
|
|
|
|
|
parseClusterNodeList :: A.Parser (S.Set ClusterNodeUUID)
|
|
|
|
parseClusterNodeList = S.fromList <$> many parseword
|
|
|
|
where
|
|
|
|
parseword = parsenode
|
|
|
|
<* ((const () <$> A8.char ' ') <|> A.endOfInput)
|
|
|
|
parsenode = ClusterNodeUUID
|
|
|
|
<$> (toUUID <$> A8.takeWhile1 (/= ' '))
|
|
|
|
|