2024-09-04 19:10:39 +00:00
|
|
|
{- git-annex simulator
|
|
|
|
-
|
|
|
|
- Copyright 2024 Joey Hess <id@joeyh.name>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU AGPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
|
|
|
|
module Annex.Sim where
|
|
|
|
|
2024-09-06 16:53:51 +00:00
|
|
|
import Annex.Common
|
2024-09-04 19:10:39 +00:00
|
|
|
import Utility.DataUnits
|
|
|
|
import Types.NumCopies
|
2024-09-06 16:53:51 +00:00
|
|
|
import Types.Group
|
2024-09-05 19:22:41 +00:00
|
|
|
import Types.StandardGroups
|
2024-09-06 16:53:51 +00:00
|
|
|
import Types.TrustLevel
|
2024-09-06 18:23:29 +00:00
|
|
|
import Types.Difference
|
2024-09-06 16:53:51 +00:00
|
|
|
import Git
|
2024-09-04 19:10:39 +00:00
|
|
|
import Backend.Hash (genTestKey)
|
|
|
|
import Annex.UUID
|
2024-09-05 19:22:41 +00:00
|
|
|
import Annex.FileMatcher
|
2024-09-06 18:23:29 +00:00
|
|
|
import Annex.Init
|
|
|
|
import Annex.Startup
|
2024-09-09 15:28:30 +00:00
|
|
|
import Annex.Link
|
2024-09-06 16:53:51 +00:00
|
|
|
import Logs.Group
|
|
|
|
import Logs.Trust
|
|
|
|
import Logs.PreferredContent
|
2024-09-09 13:35:42 +00:00
|
|
|
import Logs.NumCopies
|
2024-09-06 16:53:51 +00:00
|
|
|
import Logs.Remote
|
|
|
|
import Logs.MaxSize
|
2024-09-06 18:23:29 +00:00
|
|
|
import Logs.Difference
|
2024-09-09 15:06:42 +00:00
|
|
|
import Logs.UUID
|
2024-09-09 18:52:24 +00:00
|
|
|
import Logs.Location
|
2024-09-06 18:23:29 +00:00
|
|
|
import qualified Annex
|
2024-09-05 14:50:04 +00:00
|
|
|
import qualified Remote
|
2024-09-06 18:23:29 +00:00
|
|
|
import qualified Git.Construct
|
|
|
|
import qualified Git.Remote.Remove
|
2024-09-09 15:28:30 +00:00
|
|
|
import qualified Annex.Queue
|
2024-09-04 19:10:39 +00:00
|
|
|
|
|
|
|
import System.Random
|
|
|
|
import Data.Word
|
|
|
|
import qualified Data.Map.Strict as M
|
|
|
|
import qualified Data.Set as S
|
|
|
|
import qualified Data.ByteString as B
|
|
|
|
import qualified Data.ByteString.Lazy as L
|
|
|
|
import qualified Data.UUID as U
|
|
|
|
import qualified Data.UUID.V5 as U5
|
2024-09-09 15:28:30 +00:00
|
|
|
import qualified Utility.RawFilePath as R
|
|
|
|
import qualified System.FilePath.ByteString as P
|
2024-09-04 19:10:39 +00:00
|
|
|
|
|
|
|
data SimState = SimState
|
|
|
|
{ simRepos :: M.Map RepoName UUID
|
2024-09-09 20:06:45 +00:00
|
|
|
, simRepoList :: [RepoName]
|
2024-09-04 19:10:39 +00:00
|
|
|
, simRepoState :: M.Map RepoName SimRepoState
|
2024-09-09 21:20:13 +00:00
|
|
|
, simConnections :: M.Map UUID (S.Set RemoteName)
|
2024-09-09 15:28:30 +00:00
|
|
|
, simFiles :: M.Map RawFilePath Key
|
2024-09-04 19:10:39 +00:00
|
|
|
, simRng :: StdGen
|
2024-09-09 13:35:42 +00:00
|
|
|
, simTrustLevels :: M.Map UUID TrustLevel
|
2024-09-04 19:10:39 +00:00
|
|
|
, simNumCopies :: NumCopies
|
2024-09-09 13:35:42 +00:00
|
|
|
, simMinCopies :: MinCopies
|
|
|
|
, simGroups :: M.Map UUID (S.Set Group)
|
|
|
|
, simWanted :: M.Map UUID PreferredContentExpression
|
|
|
|
, simRequired :: M.Map UUID PreferredContentExpression
|
2024-09-06 16:53:51 +00:00
|
|
|
, simGroupWanted :: M.Map Group PreferredContentExpression
|
2024-09-09 13:35:42 +00:00
|
|
|
, simMaxSize :: M.Map UUID MaxSize
|
2024-09-04 19:10:39 +00:00
|
|
|
, simRebalance :: Bool
|
2024-09-06 16:53:51 +00:00
|
|
|
, simGetExistingRepoByName :: GetExistingRepoByName
|
2024-09-09 20:06:45 +00:00
|
|
|
, simHistory :: [SimCommand]
|
2024-09-04 19:10:39 +00:00
|
|
|
}
|
|
|
|
deriving (Show)
|
|
|
|
|
2024-09-06 18:23:29 +00:00
|
|
|
emptySimState :: StdGen -> GetExistingRepoByName -> SimState
|
|
|
|
emptySimState rng repobyname = SimState
|
2024-09-04 19:10:39 +00:00
|
|
|
{ simRepos = mempty
|
2024-09-09 20:06:45 +00:00
|
|
|
, simRepoList = mempty
|
2024-09-04 19:10:39 +00:00
|
|
|
, simRepoState = mempty
|
|
|
|
, simConnections = mempty
|
|
|
|
, simFiles = mempty
|
2024-09-06 18:23:29 +00:00
|
|
|
, simRng = rng
|
2024-09-06 16:53:51 +00:00
|
|
|
, simTrustLevels = mempty
|
2024-09-04 19:10:39 +00:00
|
|
|
, simNumCopies = configuredNumCopies 1
|
2024-09-09 13:35:42 +00:00
|
|
|
, simMinCopies = configuredMinCopies 1
|
2024-09-04 19:10:39 +00:00
|
|
|
, simGroups = mempty
|
|
|
|
, simWanted = mempty
|
|
|
|
, simRequired = mempty
|
|
|
|
, simGroupWanted = mempty
|
|
|
|
, simMaxSize = mempty
|
|
|
|
, simRebalance = False
|
2024-09-06 16:53:51 +00:00
|
|
|
, simGetExistingRepoByName = repobyname
|
2024-09-09 20:06:45 +00:00
|
|
|
, simHistory = []
|
2024-09-04 19:10:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
-- State that can vary between different repos in the simulation.
|
|
|
|
data SimRepoState = SimRepoState
|
2024-09-09 18:52:24 +00:00
|
|
|
{ simLocations :: M.Map Key (S.Set UUID)
|
2024-09-06 16:53:51 +00:00
|
|
|
, simIsSpecialRemote :: Bool
|
2024-09-06 18:23:29 +00:00
|
|
|
, simRepo :: Maybe SimRepo
|
2024-09-04 19:10:39 +00:00
|
|
|
}
|
2024-09-06 18:23:29 +00:00
|
|
|
|
|
|
|
instance Show SimRepoState where
|
|
|
|
show _ = "SimRepoState"
|
2024-09-04 19:10:39 +00:00
|
|
|
|
2024-09-09 18:52:24 +00:00
|
|
|
setPresentKey :: UUID -> Key -> SimRepoState -> SimRepoState
|
|
|
|
setPresentKey u k rst = rst
|
2024-09-04 19:10:39 +00:00
|
|
|
{ simLocations =
|
2024-09-09 18:52:24 +00:00
|
|
|
M.insertWith S.union k (S.singleton u) (simLocations rst)
|
2024-09-04 19:10:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
newtype RepoName = RepoName { fromRepoName :: String }
|
|
|
|
deriving (Show, Eq, Ord)
|
|
|
|
|
2024-09-09 20:06:45 +00:00
|
|
|
newtype RemoteName = RemoteName { fromRemoteName :: String }
|
|
|
|
deriving (Show, Eq, Ord)
|
|
|
|
|
2024-09-04 19:10:39 +00:00
|
|
|
data SimCommand
|
|
|
|
= CommandInit RepoName
|
|
|
|
| CommandInitRemote RepoName
|
|
|
|
| CommandUse RepoName String
|
2024-09-09 20:06:45 +00:00
|
|
|
| CommandConnect RepoName RemoteName
|
|
|
|
| CommandDisconnect RepoName RemoteName
|
2024-09-05 19:22:41 +00:00
|
|
|
| CommandAddTree RepoName PreferredContentExpression
|
2024-09-04 19:10:39 +00:00
|
|
|
| CommandAdd FilePath ByteSize RepoName
|
|
|
|
| CommandStep Int
|
2024-09-09 20:06:45 +00:00
|
|
|
| CommandAction RepoName SimAction
|
2024-09-04 19:10:39 +00:00
|
|
|
| CommandSeed Int
|
|
|
|
| CommandPresent RepoName FilePath
|
|
|
|
| CommandNotPresent RepoName FilePath
|
|
|
|
| CommandNumCopies Int
|
2024-09-09 13:35:42 +00:00
|
|
|
| CommandMinCopies Int
|
2024-09-06 16:53:51 +00:00
|
|
|
| CommandTrustLevel RepoName String
|
|
|
|
| CommandGroup RepoName Group
|
|
|
|
| CommandUngroup RepoName Group
|
2024-09-05 19:22:41 +00:00
|
|
|
| CommandWanted RepoName PreferredContentExpression
|
|
|
|
| CommandRequired RepoName PreferredContentExpression
|
2024-09-06 16:53:51 +00:00
|
|
|
| CommandGroupWanted Group PreferredContentExpression
|
2024-09-04 19:10:39 +00:00
|
|
|
| CommandMaxSize RepoName MaxSize
|
|
|
|
| CommandRebalance Bool
|
|
|
|
deriving (Show)
|
|
|
|
|
2024-09-09 20:06:45 +00:00
|
|
|
data SimAction
|
|
|
|
= ActionPull RemoteName
|
|
|
|
| ActionPush RemoteName
|
|
|
|
| ActionGetWanted RemoteName
|
|
|
|
| ActionDropUnwanted (Maybe RemoteName)
|
|
|
|
| ActionSendWanted RemoteName
|
|
|
|
| ActionGitPush RemoteName
|
|
|
|
| ActionGitPull RemoteName
|
|
|
|
deriving (Show)
|
|
|
|
|
2024-09-09 14:59:01 +00:00
|
|
|
runSimCommand :: SimCommand -> SimState -> Annex SimState
|
|
|
|
runSimCommand cmd st = case applySimCommand cmd st of
|
|
|
|
Left err -> giveup err
|
|
|
|
Right (Right st') -> return st'
|
|
|
|
Right (Left mkst) -> mkst
|
|
|
|
|
2024-09-05 19:22:41 +00:00
|
|
|
applySimCommand
|
|
|
|
:: SimCommand
|
|
|
|
-> SimState
|
|
|
|
-> Either String (Either (Annex SimState) SimState)
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand (CommandStep n) st
|
|
|
|
| n > 0 = case randomRepo st of
|
|
|
|
(Just (_repo, u), st') ->
|
2024-09-09 21:20:13 +00:00
|
|
|
let (act, st'') = randomAction u st'
|
2024-09-09 21:04:32 +00:00
|
|
|
st''' = applySimAction u act st''
|
|
|
|
in applySimCommand (CommandStep (pred n)) st'''
|
|
|
|
(Nothing, st') -> Right $ Right st'
|
|
|
|
| otherwise = Right $ Right st
|
|
|
|
applySimCommand c st =
|
|
|
|
applySimCommand' c $ st { simHistory = c : simHistory st }
|
|
|
|
|
|
|
|
applySimCommand'
|
|
|
|
:: SimCommand
|
|
|
|
-> SimState
|
|
|
|
-> Either String (Either (Annex SimState) SimState)
|
|
|
|
applySimCommand' (CommandInit reponame) st =
|
2024-09-05 20:22:08 +00:00
|
|
|
checkNonexistantRepo reponame st $
|
|
|
|
let (u, st') = genSimUUID st reponame
|
2024-09-06 16:53:51 +00:00
|
|
|
in Right $ Right $ addRepo reponame (newSimRepoConfig u False) st'
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandInitRemote reponame) st =
|
2024-09-05 20:22:08 +00:00
|
|
|
checkNonexistantRepo reponame st $
|
|
|
|
let (u, st') = genSimUUID st reponame
|
2024-09-06 16:53:51 +00:00
|
|
|
in Right $ Right $ addRepo reponame (newSimRepoConfig u True) st'
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandUse reponame s) st =
|
2024-09-06 16:53:51 +00:00
|
|
|
case getExistingRepoByName (simGetExistingRepoByName st) s of
|
|
|
|
Right existingrepo -> checkNonexistantRepo reponame st $
|
|
|
|
Right $ Right $ addRepo reponame existingrepo st
|
|
|
|
Left msg -> Left $ "Unable to use a repository \""
|
2024-09-05 14:50:04 +00:00
|
|
|
++ fromRepoName reponame
|
|
|
|
++ "\" in the simulation because " ++ msg
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandConnect repo remote) st =
|
2024-09-09 21:20:13 +00:00
|
|
|
checkKnownRepo repo st $ \u -> Right $ Right $ st
|
2024-09-05 19:22:41 +00:00
|
|
|
{ simConnections =
|
2024-09-09 21:20:13 +00:00
|
|
|
let s = case M.lookup u (simConnections st) of
|
2024-09-05 19:22:41 +00:00
|
|
|
Just cs -> S.insert remote cs
|
|
|
|
Nothing -> S.singleton remote
|
2024-09-09 21:20:13 +00:00
|
|
|
in M.insert u s (simConnections st)
|
2024-09-05 19:22:41 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandDisconnect repo remote) st =
|
2024-09-09 21:20:13 +00:00
|
|
|
checkKnownRepo repo st $ \u -> Right $ Right $ st
|
2024-09-05 19:22:41 +00:00
|
|
|
{ simConnections =
|
2024-09-09 21:20:13 +00:00
|
|
|
let sc = case M.lookup u (simConnections st) of
|
2024-09-05 19:22:41 +00:00
|
|
|
Just s -> S.delete remote s
|
|
|
|
Nothing -> S.empty
|
2024-09-09 21:20:13 +00:00
|
|
|
in M.insert u sc (simConnections st)
|
2024-09-05 19:22:41 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandAddTree repo expr) st =
|
2024-09-09 13:35:42 +00:00
|
|
|
checkKnownRepo repo st $ const $
|
2024-09-05 19:22:41 +00:00
|
|
|
checkValidPreferredContentExpression expr $ Left $
|
|
|
|
error "TODO" -- XXX
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandAdd file sz repo) st = checkKnownRepo repo st $ \u ->
|
2024-09-04 19:10:39 +00:00
|
|
|
let (k, st') = genSimKey sz st
|
2024-09-05 19:22:41 +00:00
|
|
|
in Right $ Right $ st'
|
2024-09-09 15:28:30 +00:00
|
|
|
{ simFiles = M.insert (toRawFilePath file) k (simFiles st')
|
2024-09-06 16:53:51 +00:00
|
|
|
, simRepoState = case M.lookup repo (simRepoState st') of
|
|
|
|
Just rst -> M.insert repo
|
2024-09-09 18:52:24 +00:00
|
|
|
(setPresentKey u k rst)
|
2024-09-06 16:53:51 +00:00
|
|
|
(simRepoState st')
|
|
|
|
Nothing -> error "no simRepoState in applySimCommand CommandAdd"
|
2024-09-04 19:10:39 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandStep _) _ = error "applySimCommand' CommandStep"
|
|
|
|
applySimCommand' (CommandAction repo act) st =
|
|
|
|
checkKnownRepo repo st $ \u ->
|
|
|
|
Right $ Right $ applySimAction u act st
|
|
|
|
applySimCommand' (CommandSeed rngseed) st = Right $ Right $ st
|
2024-09-04 19:10:39 +00:00
|
|
|
{ simRng = mkStdGen rngseed
|
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandPresent repo file) st = checkKnownRepo repo st $ \u ->
|
2024-09-09 15:28:30 +00:00
|
|
|
case (M.lookup (toRawFilePath file) (simFiles st), M.lookup repo (simRepoState st)) of
|
2024-09-04 19:10:39 +00:00
|
|
|
(Just k, Just rst) -> case M.lookup k (simLocations rst) of
|
2024-09-09 18:52:24 +00:00
|
|
|
Just locs | S.member u locs -> Right $ Right st
|
2024-09-04 19:10:39 +00:00
|
|
|
_ -> missing
|
2024-09-05 19:22:41 +00:00
|
|
|
(Just _, Nothing) -> missing
|
2024-09-04 19:10:39 +00:00
|
|
|
(Nothing, _) -> Left $ "Expected " ++ file
|
|
|
|
++ " to be present in " ++ fromRepoName repo
|
|
|
|
++ ", but the simulation does not include that file."
|
|
|
|
where
|
|
|
|
missing = Left $ "Expected " ++ file ++ " to be present in "
|
|
|
|
++ fromRepoName repo ++ ", but it is not."
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandNotPresent repo file) st = checkKnownRepo repo st $ \u ->
|
2024-09-09 15:28:30 +00:00
|
|
|
case (M.lookup (toRawFilePath file) (simFiles st), M.lookup repo (simRepoState st)) of
|
2024-09-04 19:10:39 +00:00
|
|
|
(Just k, Just rst) -> case M.lookup k (simLocations rst) of
|
2024-09-09 18:52:24 +00:00
|
|
|
Just locs | S.notMember u locs -> Right $ Right st
|
2024-09-04 19:10:39 +00:00
|
|
|
_ -> present
|
2024-09-05 19:22:41 +00:00
|
|
|
(Just _, Nothing) -> present
|
2024-09-04 19:10:39 +00:00
|
|
|
(Nothing, _) -> Left $ "Expected " ++ file
|
|
|
|
++ " to not be present in " ++ fromRepoName repo
|
|
|
|
++ ", but the simulation does not include that file."
|
|
|
|
where
|
|
|
|
present = Left $ "Expected " ++ file ++ " not to be present in "
|
|
|
|
++ fromRepoName repo ++ ", but it is present."
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandNumCopies n) st = Right $ Right $ st
|
2024-09-04 19:10:39 +00:00
|
|
|
{ simNumCopies = configuredNumCopies n
|
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandMinCopies n) st = Right $ Right $ st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simMinCopies = configuredMinCopies n
|
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandTrustLevel repo s) st = checkKnownRepo repo st $ \u ->
|
2024-09-06 16:53:51 +00:00
|
|
|
case readTrustLevel s of
|
|
|
|
Just trustlevel -> Right $ Right $ st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simTrustLevels = M.insert u trustlevel
|
2024-09-06 16:53:51 +00:00
|
|
|
(simTrustLevels st)
|
|
|
|
}
|
|
|
|
Nothing -> Left $ "Unknown trust level \"" ++ s ++ "\"."
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandGroup repo groupname) st = checkKnownRepo repo st $ \u ->
|
2024-09-05 19:22:41 +00:00
|
|
|
Right $ Right $ st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simGroups = M.insertWith S.union u
|
2024-09-06 16:53:51 +00:00
|
|
|
(S.singleton groupname)
|
2024-09-05 19:22:41 +00:00
|
|
|
(simGroups st)
|
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandUngroup repo groupname) st = checkKnownRepo repo st $ \u ->
|
2024-09-05 19:22:41 +00:00
|
|
|
Right $ Right $ st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simGroups = M.adjust (S.delete groupname) u (simGroups st)
|
2024-09-05 19:22:41 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandWanted repo expr) st = checkKnownRepo repo st $ \u ->
|
2024-09-05 19:22:41 +00:00
|
|
|
checkValidPreferredContentExpression expr $ Right $ st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simWanted = M.insert u expr (simWanted st)
|
2024-09-05 19:22:41 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandRequired repo expr) st = checkKnownRepo repo st $ \u ->
|
2024-09-05 19:22:41 +00:00
|
|
|
checkValidPreferredContentExpression expr $ Right $ st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simRequired = M.insert u expr (simRequired st)
|
2024-09-05 19:22:41 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandGroupWanted groupname expr) st =
|
2024-09-05 19:22:41 +00:00
|
|
|
checkValidPreferredContentExpression expr $ Right $ st
|
2024-09-06 16:53:51 +00:00
|
|
|
{ simGroupWanted = M.insert groupname expr (simGroupWanted st)
|
2024-09-05 19:22:41 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandMaxSize repo sz) st = checkKnownRepo repo st $ \u ->
|
2024-09-05 19:22:41 +00:00
|
|
|
Right $ Right $ st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simMaxSize = M.insert u sz (simMaxSize st)
|
2024-09-05 19:22:41 +00:00
|
|
|
}
|
2024-09-09 21:04:32 +00:00
|
|
|
applySimCommand' (CommandRebalance b) st = Right $ Right $ st
|
2024-09-04 19:10:39 +00:00
|
|
|
{ simRebalance = b
|
|
|
|
}
|
|
|
|
|
2024-09-09 21:04:32 +00:00
|
|
|
-- XXX todo
|
|
|
|
applySimAction :: UUID -> SimAction -> SimState -> SimState
|
|
|
|
applySimAction u (ActionPull remote) st = undefined
|
|
|
|
applySimAction u (ActionPush remote) st = undefined
|
|
|
|
applySimAction u (ActionGetWanted remote) st = undefined
|
|
|
|
applySimAction u (ActionDropUnwanted Nothing) st = undefined
|
|
|
|
applySimAction u (ActionDropUnwanted (Just remote)) st = undefined
|
|
|
|
applySimAction u (ActionSendWanted remote) st = undefined
|
|
|
|
applySimAction u (ActionGitPush remote) st = undefined
|
|
|
|
applySimAction u (ActionGitPull remote) st = undefined
|
|
|
|
|
2024-09-05 20:22:08 +00:00
|
|
|
checkNonexistantRepo :: RepoName -> SimState -> Either String a -> Either String a
|
|
|
|
checkNonexistantRepo reponame st a = case M.lookup reponame (simRepos st) of
|
|
|
|
Nothing -> a
|
|
|
|
Just _ -> Left $ "There is already a repository in the simulation named \""
|
|
|
|
++ fromRepoName reponame ++ "\"."
|
|
|
|
|
2024-09-09 13:35:42 +00:00
|
|
|
checkKnownRepo :: RepoName -> SimState -> (UUID -> Either String a) -> Either String a
|
2024-09-05 19:22:41 +00:00
|
|
|
checkKnownRepo reponame st a = case M.lookup reponame (simRepos st) of
|
2024-09-09 13:35:42 +00:00
|
|
|
Just u -> a u
|
2024-09-05 19:22:41 +00:00
|
|
|
Nothing -> Left $ "No repository in the simulation is named \""
|
|
|
|
++ fromRepoName reponame ++ "\"."
|
|
|
|
|
|
|
|
checkValidPreferredContentExpression :: PreferredContentExpression -> v -> Either String v
|
|
|
|
checkValidPreferredContentExpression expr v =
|
|
|
|
case checkPreferredContentExpression expr of
|
|
|
|
Nothing -> Right v
|
|
|
|
Just e -> Left $ "Failed parsing \"" ++ expr ++ "\": " ++ e
|
|
|
|
|
2024-09-04 19:10:39 +00:00
|
|
|
simRandom :: SimState -> (StdGen -> (v, StdGen)) -> (v -> r) -> (r, SimState)
|
|
|
|
simRandom st mk f =
|
|
|
|
let (v, rng) = mk (simRng st)
|
|
|
|
in (f v, st { simRng = rng })
|
|
|
|
|
2024-09-09 21:04:32 +00:00
|
|
|
randomRepo :: SimState -> (Maybe (RepoName, UUID), SimState)
|
2024-09-09 20:06:45 +00:00
|
|
|
randomRepo st
|
|
|
|
| null (simRepoList st) = (Nothing, st)
|
|
|
|
| otherwise = simRandom st
|
2024-09-09 21:04:32 +00:00
|
|
|
(randomR (0, length (simRepoList st) - 1)) $ \n -> do
|
|
|
|
let r = simRepoList st !! n
|
|
|
|
u <- M.lookup r (simRepos st)
|
|
|
|
return (r, u)
|
2024-09-09 20:06:45 +00:00
|
|
|
|
2024-09-09 21:20:13 +00:00
|
|
|
randomAction :: UUID -> SimState -> (SimAction, SimState)
|
|
|
|
randomAction u st = case M.lookup u (simConnections st) of
|
|
|
|
Just cs | not (S.null cs) ->
|
|
|
|
let (mkact, st') = simRandom st (randomR (0, length mkactions - 1))
|
|
|
|
(mkactions !!)
|
|
|
|
(remote, st'') = simRandom st' (randomR (0, S.size cs - 1))
|
|
|
|
(`S.elemAt` cs)
|
|
|
|
in (mkact remote, st'')
|
|
|
|
-- When there are no remotes, this is the only possible action.
|
|
|
|
_ -> (ActionDropUnwanted Nothing, st)
|
|
|
|
where
|
|
|
|
mkactions =
|
|
|
|
[ ActionPull
|
|
|
|
, ActionPush
|
|
|
|
, ActionGetWanted
|
|
|
|
, ActionDropUnwanted . Just
|
|
|
|
, const (ActionDropUnwanted Nothing)
|
|
|
|
, ActionSendWanted
|
|
|
|
, ActionGitPush
|
|
|
|
, ActionGitPull
|
|
|
|
]
|
2024-09-09 20:06:45 +00:00
|
|
|
|
2024-09-04 19:10:39 +00:00
|
|
|
randomWords :: Int -> StdGen -> ([Word8], StdGen)
|
|
|
|
randomWords = go []
|
|
|
|
where
|
|
|
|
go c n g
|
|
|
|
| n < 1 = (c, g)
|
|
|
|
| otherwise =
|
|
|
|
let (w, g') = random g
|
|
|
|
in go (w:c) (pred n) g'
|
|
|
|
|
|
|
|
genSimKey :: ByteSize -> SimState -> (Key, SimState)
|
|
|
|
genSimKey sz st = simRandom st (randomWords 1024) mk
|
|
|
|
where
|
|
|
|
mk b =
|
|
|
|
let tk = genTestKey $ L.pack b
|
|
|
|
in alterKey tk $ \kd -> kd { keySize = Just sz }
|
|
|
|
|
|
|
|
genSimUUID :: SimState -> RepoName -> (UUID, SimState)
|
|
|
|
genSimUUID st (RepoName reponame) = simRandom st (randomWords 1024)
|
|
|
|
(\l -> genUUIDInNameSpace simUUIDNameSpace (encodeBS reponame <> B.pack l))
|
|
|
|
|
|
|
|
simUUIDNameSpace :: U.UUID
|
|
|
|
simUUIDNameSpace = U5.generateNamed U5.namespaceURL $
|
|
|
|
B.unpack "http://git-annex.branchable.com/git-annex-sim/"
|
2024-09-05 14:50:04 +00:00
|
|
|
|
2024-09-06 16:53:51 +00:00
|
|
|
newtype GetExistingRepoByName = GetExistingRepoByName
|
|
|
|
{ getExistingRepoByName :: String -> Either String SimRepoConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
instance Show GetExistingRepoByName where
|
|
|
|
show _ = "GetExistingRepoByName"
|
|
|
|
|
|
|
|
data SimRepoConfig = SimRepoConfig
|
|
|
|
{ simRepoUUID :: UUID
|
|
|
|
, simRepoIsSpecialRemote :: Bool
|
|
|
|
, simRepoGroups :: S.Set Group
|
|
|
|
, simRepoTrustLevel :: TrustLevel
|
|
|
|
, simRepoPreferredContent :: Maybe PreferredContentExpression
|
|
|
|
, simRepoRequiredContent :: Maybe PreferredContentExpression
|
|
|
|
, simRepoGroupPreferredContent :: M.Map Group PreferredContentExpression
|
|
|
|
, simRepoMaxSize :: Maybe MaxSize
|
|
|
|
}
|
|
|
|
deriving (Show)
|
|
|
|
|
|
|
|
newSimRepoConfig :: UUID -> Bool -> SimRepoConfig
|
|
|
|
newSimRepoConfig u isspecialremote = SimRepoConfig
|
|
|
|
{ simRepoUUID = u
|
|
|
|
, simRepoIsSpecialRemote = isspecialremote
|
|
|
|
, simRepoGroups = mempty
|
|
|
|
, simRepoTrustLevel = def
|
|
|
|
, simRepoPreferredContent = Nothing
|
|
|
|
, simRepoRequiredContent = Nothing
|
|
|
|
, simRepoGroupPreferredContent = mempty
|
|
|
|
, simRepoMaxSize = Nothing
|
2024-09-05 14:50:04 +00:00
|
|
|
}
|
|
|
|
|
2024-09-06 16:53:51 +00:00
|
|
|
addRepo :: RepoName -> SimRepoConfig -> SimState -> SimState
|
|
|
|
addRepo reponame simrepo st = st
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simRepos = M.insert reponame u (simRepos st)
|
2024-09-09 20:06:45 +00:00
|
|
|
, simRepoList = if reponame `elem` simRepoList st
|
|
|
|
then simRepoList st
|
|
|
|
else reponame : simRepoList st
|
2024-09-06 16:53:51 +00:00
|
|
|
, simRepoState = M.insert reponame rst (simRepoState st)
|
2024-09-09 21:20:13 +00:00
|
|
|
, simConnections = M.insert u mempty (simConnections st)
|
2024-09-09 13:35:42 +00:00
|
|
|
, simGroups = M.insert u (simRepoGroups simrepo) (simGroups st)
|
|
|
|
, simTrustLevels = M.insert u
|
2024-09-06 16:53:51 +00:00
|
|
|
(simRepoTrustLevel simrepo)
|
|
|
|
(simTrustLevels st)
|
|
|
|
, simWanted = M.alter
|
|
|
|
(const $ simRepoPreferredContent simrepo)
|
2024-09-09 13:35:42 +00:00
|
|
|
u
|
2024-09-06 16:53:51 +00:00
|
|
|
(simWanted st)
|
|
|
|
, simRequired = M.alter
|
|
|
|
(const $ simRepoRequiredContent simrepo)
|
2024-09-09 13:35:42 +00:00
|
|
|
u
|
2024-09-06 16:53:51 +00:00
|
|
|
(simRequired st)
|
|
|
|
, simGroupWanted = M.union
|
|
|
|
(simRepoGroupPreferredContent simrepo)
|
|
|
|
(simGroupWanted st)
|
|
|
|
, simMaxSize = M.alter
|
|
|
|
(const $ simRepoMaxSize simrepo)
|
2024-09-09 13:35:42 +00:00
|
|
|
u
|
2024-09-06 16:53:51 +00:00
|
|
|
(simMaxSize st)
|
|
|
|
}
|
|
|
|
where
|
2024-09-09 13:35:42 +00:00
|
|
|
u = simRepoUUID simrepo
|
2024-09-06 16:53:51 +00:00
|
|
|
rst = SimRepoState
|
|
|
|
{ simLocations = mempty
|
|
|
|
, simIsSpecialRemote = simRepoIsSpecialRemote simrepo
|
2024-09-06 18:23:29 +00:00
|
|
|
, simRepo = Nothing
|
2024-09-06 16:53:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mkGetExistingRepoByName :: Annex GetExistingRepoByName
|
|
|
|
mkGetExistingRepoByName = do
|
|
|
|
groupmap <- groupMap
|
|
|
|
trustmap <- trustMap
|
|
|
|
pcmap <- preferredContentMapRaw
|
|
|
|
rcmap <- requiredContentMapRaw
|
|
|
|
gpcmap <- groupPreferredContentMapRaw
|
|
|
|
maxsizes <- getMaxSizes
|
|
|
|
nametouuid <- Remote.nameToUUID''
|
|
|
|
remoteconfigmap <- readRemoteLog
|
|
|
|
return $ GetExistingRepoByName $ \name ->
|
|
|
|
case nametouuid name of
|
|
|
|
(u:[], _) -> Right $
|
|
|
|
let gs = fromMaybe S.empty $
|
|
|
|
M.lookup u (groupsByUUID groupmap)
|
|
|
|
in SimRepoConfig
|
|
|
|
{ simRepoUUID = u
|
|
|
|
, simRepoIsSpecialRemote =
|
|
|
|
M.member u remoteconfigmap
|
|
|
|
, simRepoGroups = gs
|
|
|
|
, simRepoTrustLevel =
|
|
|
|
lookupTrust' u trustmap
|
|
|
|
, simRepoPreferredContent =
|
|
|
|
M.lookup u pcmap
|
|
|
|
, simRepoRequiredContent =
|
|
|
|
M.lookup u rcmap
|
|
|
|
, simRepoGroupPreferredContent =
|
|
|
|
M.restrictKeys gpcmap gs
|
|
|
|
, simRepoMaxSize =
|
|
|
|
M.lookup u maxsizes
|
|
|
|
}
|
|
|
|
(_, msg) -> Left msg
|
2024-09-05 14:50:04 +00:00
|
|
|
|
2024-09-06 18:23:29 +00:00
|
|
|
-- Information about a git repository that is cloned and used to represent
|
|
|
|
-- a repository in the simulation
|
|
|
|
data SimRepo = SimRepo
|
|
|
|
{ simRepoGitRepo :: Repo
|
|
|
|
, simRepoAnnex :: (Annex.AnnexState, Annex.AnnexRead)
|
|
|
|
, simRepoCurrState :: SimState
|
2024-09-09 18:52:24 +00:00
|
|
|
, simRepoName :: RepoName
|
2024-09-06 18:23:29 +00:00
|
|
|
}
|
|
|
|
|
2024-09-09 14:59:01 +00:00
|
|
|
{- Clones and updates SimRepos to reflect the SimState. -}
|
|
|
|
updateSimRepos :: Repo -> (UUID -> FilePath) -> SimState -> IO SimState
|
|
|
|
updateSimRepos parent getdest st = do
|
|
|
|
st' <- updateSimRepoStates st
|
|
|
|
cloneNewSimRepos parent getdest st'
|
|
|
|
|
|
|
|
updateSimRepoStates :: SimState -> IO SimState
|
|
|
|
updateSimRepoStates st = go st (M.toList $ simRepoState st)
|
|
|
|
where
|
|
|
|
go st' [] = return st
|
|
|
|
go st' ((reponame, rst):rest) = case simRepo rst of
|
|
|
|
Just sr -> do
|
|
|
|
sr' <- updateSimRepoState st sr
|
|
|
|
let rst' = rst { simRepo = Just sr' }
|
|
|
|
let st'' = st
|
|
|
|
{ simRepoState = M.insert reponame rst'
|
|
|
|
(simRepoState st)
|
|
|
|
}
|
|
|
|
go st'' rest
|
|
|
|
Nothing -> go st' rest
|
|
|
|
|
|
|
|
cloneNewSimRepos :: Repo -> (UUID -> FilePath) -> SimState -> IO SimState
|
|
|
|
cloneNewSimRepos parent getdest = \st -> go st (M.toList $ simRepoState st)
|
|
|
|
where
|
|
|
|
go st [] = return st
|
|
|
|
go st ((reponame, rst):rest) =
|
|
|
|
case (simRepo rst, M.lookup reponame (simRepos st)) of
|
|
|
|
(Nothing, Just u) -> do
|
|
|
|
sr <- cloneSimRepo reponame u parent
|
|
|
|
(getdest u) st
|
|
|
|
let rst' = rst { simRepo = Just sr }
|
|
|
|
let st' = st
|
|
|
|
{ simRepoState = M.insert reponame rst'
|
|
|
|
(simRepoState st)
|
|
|
|
}
|
|
|
|
go st' rest
|
|
|
|
_ -> go st rest
|
|
|
|
|
2024-09-06 18:23:29 +00:00
|
|
|
cloneSimRepo :: RepoName -> UUID -> Repo -> FilePath -> SimState -> IO SimRepo
|
|
|
|
cloneSimRepo simreponame u parent dest st = do
|
2024-09-06 16:53:51 +00:00
|
|
|
cloned <- boolSystem "git"
|
|
|
|
[ Param "clone"
|
|
|
|
, Param "--shared"
|
|
|
|
, Param "--quiet"
|
|
|
|
-- Avoid overhead of checking out the working tree.
|
|
|
|
, Param "--no-checkout"
|
2024-09-06 18:23:29 +00:00
|
|
|
-- Make sure the origin gets that name.
|
|
|
|
, Param "--origin", Param "origin"
|
2024-09-06 16:53:51 +00:00
|
|
|
, File (fromRawFilePath (repoPath parent))
|
|
|
|
, File dest
|
|
|
|
]
|
2024-09-06 18:23:29 +00:00
|
|
|
unless cloned $
|
|
|
|
giveup "git clone failed"
|
|
|
|
simrepo <- Git.Construct.fromPath (toRawFilePath dest)
|
|
|
|
ast <- Annex.new simrepo
|
|
|
|
((), ast') <- Annex.run ast $ doQuietAction $ do
|
|
|
|
-- Disconnect simulated repository from origin, so its
|
|
|
|
-- git-annex branch is not used, and also to prevent any
|
|
|
|
-- accidental foot shooting pushes to it.
|
|
|
|
inRepo $ Git.Remote.Remove.remove "origin"
|
|
|
|
storeUUID u
|
|
|
|
-- Prevent merging this simulated git-annex branch with
|
|
|
|
-- any real one. Writing to the git-annex branch here also
|
|
|
|
-- avoids checkSharedClone enabling the shared clone
|
|
|
|
-- setting, which is not wanted here.
|
|
|
|
recordDifferences simulationDifferences u
|
2024-09-09 15:06:42 +00:00
|
|
|
let desc = simulatedRepositoryDescription simreponame
|
2024-09-06 18:23:29 +00:00
|
|
|
initialize startupAnnex (Just desc) Nothing
|
|
|
|
updateSimRepoState st $ SimRepo
|
|
|
|
{ simRepoGitRepo = simrepo
|
|
|
|
, simRepoAnnex = ast'
|
|
|
|
, simRepoCurrState = emptySimState
|
|
|
|
(simRng st)
|
|
|
|
(simGetExistingRepoByName st)
|
2024-09-09 18:52:24 +00:00
|
|
|
, simRepoName = simreponame
|
2024-09-06 18:23:29 +00:00
|
|
|
}
|
|
|
|
|
2024-09-09 15:06:42 +00:00
|
|
|
simulatedRepositoryDescription :: RepoName -> String
|
|
|
|
simulatedRepositoryDescription simreponame =
|
|
|
|
"simulated repository " ++ fromRepoName simreponame
|
|
|
|
|
2024-09-09 13:35:42 +00:00
|
|
|
simulationDifferences :: Differences
|
|
|
|
simulationDifferences = mkDifferences $ S.singleton Simulation
|
|
|
|
|
2024-09-06 18:23:29 +00:00
|
|
|
updateSimRepoState :: SimState -> SimRepo -> IO SimRepo
|
2024-09-09 13:35:42 +00:00
|
|
|
updateSimRepoState newst sr = do
|
|
|
|
((), (ast, ard)) <- Annex.run (simRepoAnnex sr) $ doQuietAction $ do
|
2024-09-06 18:23:29 +00:00
|
|
|
let oldst = simRepoCurrState sr
|
2024-09-09 15:06:42 +00:00
|
|
|
updateField oldst newst simRepos $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = const . setdesc
|
2024-09-09 15:06:42 +00:00
|
|
|
, addDiff = setdesc
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const $ const noop
|
2024-09-09 15:06:42 +00:00
|
|
|
}
|
2024-09-09 13:35:42 +00:00
|
|
|
updateField oldst newst simTrustLevels $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = const . trustSet
|
2024-09-09 13:35:42 +00:00
|
|
|
, addDiff = trustSet
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const . flip trustSet def
|
2024-09-09 13:35:42 +00:00
|
|
|
}
|
|
|
|
when (simNumCopies oldst /= simNumCopies newst) $
|
|
|
|
setGlobalNumCopies (simNumCopies newst)
|
|
|
|
when (simMinCopies oldst /= simMinCopies newst) $
|
|
|
|
setGlobalMinCopies (simMinCopies newst)
|
|
|
|
updateField oldst newst simGroups $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = \u -> const . groupChange u . const
|
2024-09-09 13:35:42 +00:00
|
|
|
, addDiff = \u -> groupChange u . const
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const . flip groupChange (const mempty)
|
2024-09-09 13:35:42 +00:00
|
|
|
}
|
|
|
|
updateField oldst newst simWanted $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = const . preferredContentSet
|
2024-09-09 13:35:42 +00:00
|
|
|
, addDiff = preferredContentSet
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const . flip preferredContentSet mempty
|
2024-09-09 13:35:42 +00:00
|
|
|
}
|
|
|
|
updateField oldst newst simRequired $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = const . requiredContentSet
|
2024-09-09 13:35:42 +00:00
|
|
|
, addDiff = requiredContentSet
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const . flip requiredContentSet mempty
|
2024-09-09 13:35:42 +00:00
|
|
|
}
|
|
|
|
updateField oldst newst simGroupWanted $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = const . groupPreferredContentSet
|
2024-09-09 13:35:42 +00:00
|
|
|
, addDiff = groupPreferredContentSet
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const . flip groupPreferredContentSet mempty
|
2024-09-09 13:35:42 +00:00
|
|
|
}
|
|
|
|
updateField oldst newst simMaxSize $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = const . recordMaxSize
|
2024-09-09 13:35:42 +00:00
|
|
|
, addDiff = recordMaxSize
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const . flip recordMaxSize (MaxSize 0)
|
|
|
|
}
|
|
|
|
updateField oldst newst getlocations $ DiffUpdate
|
|
|
|
{ replaceDiff = \k oldls newls -> do
|
|
|
|
setlocations InfoPresent k
|
|
|
|
(S.difference newls oldls)
|
|
|
|
setlocations InfoMissing k
|
|
|
|
(S.difference oldls newls)
|
|
|
|
, addDiff = setlocations InfoPresent
|
|
|
|
, removeDiff = setlocations InfoMissing
|
2024-09-09 13:35:42 +00:00
|
|
|
}
|
2024-09-09 15:28:30 +00:00
|
|
|
updateField oldst newst simFiles $ DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff = const . stageannexedfile
|
2024-09-09 18:07:52 +00:00
|
|
|
, addDiff = stageannexedfile
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff = const . unstageannexedfile
|
2024-09-09 15:28:30 +00:00
|
|
|
}
|
|
|
|
Annex.Queue.flush
|
2024-09-09 13:35:42 +00:00
|
|
|
let ard' = ard { Annex.rebalance = simRebalance newst }
|
2024-09-06 18:23:29 +00:00
|
|
|
return $ sr
|
2024-09-09 13:35:42 +00:00
|
|
|
{ simRepoAnnex = (ast, ard')
|
|
|
|
, simRepoCurrState = newst
|
2024-09-06 18:23:29 +00:00
|
|
|
}
|
2024-09-09 15:28:30 +00:00
|
|
|
where
|
|
|
|
setdesc r u = describeUUID u $ toUUIDDesc $
|
|
|
|
simulatedRepositoryDescription r
|
2024-09-09 18:07:52 +00:00
|
|
|
stageannexedfile f k = do
|
|
|
|
let f' = annexedfilepath f
|
2024-09-09 15:28:30 +00:00
|
|
|
l <- calcRepo $ gitAnnexLink f' k
|
|
|
|
addAnnexLink l f'
|
2024-09-09 18:07:52 +00:00
|
|
|
unstageannexedfile f = do
|
|
|
|
liftIO $ removeWhenExistsWith R.removeLink $
|
|
|
|
annexedfilepath f
|
|
|
|
annexedfilepath f = repoPath (simRepoGitRepo sr) P.</> f
|
2024-09-09 18:52:24 +00:00
|
|
|
getlocations = maybe mempty simLocations
|
|
|
|
. M.lookup (simRepoName sr)
|
|
|
|
. simRepoState
|
|
|
|
setlocations s k ls =
|
|
|
|
mapM_ (\l -> logChange NoLiveUpdate k l s) (S.toList ls)
|
2024-09-06 18:23:29 +00:00
|
|
|
|
2024-09-09 13:35:42 +00:00
|
|
|
data DiffUpdate a b m = DiffUpdate
|
2024-09-09 18:52:24 +00:00
|
|
|
{ replaceDiff :: a -> b -> b -> m ()
|
|
|
|
-- ^ The first value is the new one, the second is the old one.
|
2024-09-09 13:35:42 +00:00
|
|
|
, addDiff :: a -> b -> m ()
|
2024-09-09 18:52:24 +00:00
|
|
|
, removeDiff :: a -> b -> m ()
|
2024-09-09 13:35:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
updateMap
|
|
|
|
:: (Monad m, Ord a, Eq b)
|
|
|
|
=> M.Map a b
|
|
|
|
-> M.Map a b
|
|
|
|
-> DiffUpdate a b m
|
|
|
|
-> m ()
|
|
|
|
updateMap old new diffupdate = do
|
|
|
|
forM_ (M.toList $ M.intersectionWith (,) new old) $
|
|
|
|
\(k, (newv, oldv))->
|
|
|
|
when (newv /= oldv) $
|
2024-09-09 18:52:24 +00:00
|
|
|
replaceDiff diffupdate k newv oldv
|
2024-09-09 13:35:42 +00:00
|
|
|
forM_ (M.toList $ M.difference new old) $
|
|
|
|
uncurry (addDiff diffupdate)
|
2024-09-09 18:52:24 +00:00
|
|
|
forM_ (M.toList $ M.difference old new) $
|
|
|
|
\(k, oldv) -> removeDiff diffupdate k oldv
|
2024-09-09 13:35:42 +00:00
|
|
|
|
|
|
|
updateField
|
|
|
|
:: (Monad m, Ord a, Eq b)
|
|
|
|
=> v
|
|
|
|
-> v
|
|
|
|
-> (v -> M.Map a b)
|
|
|
|
-> DiffUpdate a b m
|
|
|
|
-> m ()
|
|
|
|
updateField old new f = updateMap (f old) (f new)
|