Added remote.<name>.annex-push and remote.<name>.annex-pull

The former can be useful to make remotes that don't get fully synced with
local changes, which comes up in a lot of situations.

The latter was mostly added for symmetry, but could be useful (though less
likely to be).

Implementing `remote.<name>.annex-pull` was a bit tricky, as there's no one
place where git-annex pulls/fetches from remotes. I audited all
instances of "fetch" and "pull". A few cases were left not checking this
config:

* Git.Repair can try to pull missing refs from a remote, and if the local
  repo is corrupted, that seems a reasonable thing to do even though
  the config would normally prevent it.
* Assistant.WebApp.Gpg and Remote.Gcrypt and Remote.Git do fetches
  as part of the setup process of a remote. The config would probably not
  be set then, and having the setup fail seems worse than honoring it if it
  is already set.

I have not prevented all the code that does a "merge" from merging branches
from remotes with remote.<name>.annex-pull=false. That could perhaps
be done, but it would need a way to map from branch name to remote name,
and the way refspecs work makes that hard to get really correct. So if the
user fetches manually, the git-annex branch will get merged, for example.
Anther way of looking at/justifying this is that the setting is called
"annex-pull", not "annex-merge".

This commit was supported by the NSF-funded DataLad project.
This commit is contained in:
Joey Hess 2017-04-05 13:04:02 -04:00
parent eab6146e53
commit 29e73f76ef
No known key found for this signature in database
GPG key ID: DB12DB0FF05F8F38
11 changed files with 96 additions and 23 deletions

View file

@ -110,8 +110,14 @@ reconnectRemotes rs = void $ do
pushToRemotes :: [Remote] -> Assistant [Remote] pushToRemotes :: [Remote] -> Assistant [Remote]
pushToRemotes remotes = do pushToRemotes remotes = do
now <- liftIO getCurrentTime now <- liftIO getCurrentTime
let remotes' = filter (not . remoteAnnexReadOnly . Remote.gitconfig) remotes let remotes' = filter (wantpush . Remote.gitconfig) remotes
syncAction remotes' (pushToRemotes' now) syncAction remotes' (pushToRemotes' now)
where
wantpush gc
| remoteAnnexReadOnly gc = False
| not (remoteAnnexPush gc) = False
| otherwise = True
pushToRemotes' :: UTCTime -> [Remote] -> Assistant [Remote] pushToRemotes' :: UTCTime -> [Remote] -> Assistant [Remote]
pushToRemotes' now remotes = do pushToRemotes' now remotes = do
(g, branch, u) <- liftAnnex $ do (g, branch, u) <- liftAnnex $ do
@ -195,16 +201,20 @@ manualPull :: Command.Sync.CurrBranch -> [Remote] -> Assistant ([Remote], Bool)
manualPull currentbranch remotes = do manualPull currentbranch remotes = do
g <- liftAnnex gitRepo g <- liftAnnex gitRepo
let (_xmppremotes, normalremotes) = partition Remote.isXMPPRemote remotes let (_xmppremotes, normalremotes) = partition Remote.isXMPPRemote remotes
failed <- forM normalremotes $ \r -> do failed <- forM normalremotes $ \r -> if wantpull $ Remote.gitconfig r
then do
g' <- liftAnnex $ sshOptionsTo (Remote.repo r) (Remote.gitconfig r) g g' <- liftAnnex $ sshOptionsTo (Remote.repo r) (Remote.gitconfig r) g
ifM (liftIO $ Git.Command.runBool [Param "fetch", Param $ Remote.name r] g') ifM (liftIO $ Git.Command.runBool [Param "fetch", Param $ Remote.name r] g')
( return Nothing ( return Nothing
, return $ Just r , return $ Just r
) )
else return Nothing
haddiverged <- liftAnnex Annex.Branch.forceUpdate haddiverged <- liftAnnex Annex.Branch.forceUpdate
forM_ normalremotes $ \r -> forM_ normalremotes $ \r ->
liftAnnex $ Command.Sync.mergeRemote r currentbranch Command.Sync.mergeConfig liftAnnex $ Command.Sync.mergeRemote r currentbranch Command.Sync.mergeConfig
return (catMaybes failed, haddiverged) return (catMaybes failed, haddiverged)
where
wantpull gc = remoteAnnexPull gc
{- Start syncing a remote, using a background thread. -} {- Start syncing a remote, using a background thread. -}
syncRemote :: Remote -> Assistant () syncRemote :: Remote -> Assistant ()

View file

@ -4,6 +4,9 @@ git-annex (6.20170322) UNRELEASED; urgency=medium
about it once, not every time git-annex is run. about it once, not every time git-annex is run.
* multicast: New command, uses uftp to multicast annexed files, for eg * multicast: New command, uses uftp to multicast annexed files, for eg
a classroom setting. a classroom setting.
* Added remote.<name>.annex-push and remote.<name>.annex-pull
which can be useful to make remotes that don't get fully synced with
local changes.
-- Joey Hess <id@joeyh.name> Wed, 29 Mar 2017 12:41:46 -0400 -- Joey Hess <id@joeyh.name> Wed, 29 Mar 2017 12:41:46 -0400

View file

@ -360,7 +360,7 @@ updateBranch syncbranch updateto g =
] g ] g
pullRemote :: SyncOptions -> [Git.Merge.MergeConfig] -> Remote -> CurrBranch -> CommandStart pullRemote :: SyncOptions -> [Git.Merge.MergeConfig] -> Remote -> CurrBranch -> CommandStart
pullRemote o mergeconfig remote branch = stopUnless (pure $ pullOption o) $ do pullRemote o mergeconfig remote branch = stopUnless (pure $ pullOption o && wantpull) $ do
showStart "pull" (Remote.name remote) showStart "pull" (Remote.name remote)
next $ do next $ do
showOutput showOutput
@ -370,6 +370,7 @@ pullRemote o mergeconfig remote branch = stopUnless (pure $ pullOption o) $ do
fetch = inRepoWithSshOptionsTo (Remote.repo remote) (Remote.gitconfig remote) $ fetch = inRepoWithSshOptionsTo (Remote.repo remote) (Remote.gitconfig remote) $
Git.Command.runBool Git.Command.runBool
[Param "fetch", Param $ Remote.name remote] [Param "fetch", Param $ Remote.name remote]
wantpull = remoteAnnexPull (Remote.gitconfig remote)
{- The remote probably has both a master and a synced/master branch. {- The remote probably has both a master and a synced/master branch.
- Which to merge from? Well, the master has whatever latest changes - Which to merge from? Well, the master has whatever latest changes
@ -400,7 +401,7 @@ pushRemote o remote (Just branch, _) = stopUnless (pure (pushOption o) <&&> need
showStart "push" (Remote.name remote) showStart "push" (Remote.name remote)
next $ next $ do next $ next $ do
showOutput showOutput
ok <- inRepoWithSshOptionsTo (Remote.repo remote) (Remote.gitconfig remote) $ ok <- inRepoWithSshOptionsTo (Remote.repo remote) gc $
pushBranch remote branch pushBranch remote branch
if ok if ok
then postpushupdate then postpushupdate
@ -410,7 +411,8 @@ pushRemote o remote (Just branch, _) = stopUnless (pure (pushOption o) <&&> need
return ok return ok
where where
needpush needpush
| remoteAnnexReadOnly (Remote.gitconfig remote) = return False | remoteAnnexReadOnly gc = return False
| not (remoteAnnexPush gc) = return False
| otherwise = anyM (newer remote) [syncBranch branch, Annex.Branch.name] | otherwise = anyM (newer remote) [syncBranch branch, Annex.Branch.name]
-- Do updateInstead emulation for remotes on eg removable drives -- Do updateInstead emulation for remotes on eg removable drives
-- formatted FAT, where the post-update hook won't run. -- formatted FAT, where the post-update hook won't run.
@ -426,6 +428,7 @@ pushRemote o remote (Just branch, _) = stopUnless (pure (pushOption o) <&&> need
, return True , return True
) )
| otherwise = return True | otherwise = return True
gc = Remote.gitconfig remote
{- Pushes a regular branch like master to a remote. Also pushes the git-annex {- Pushes a regular branch like master to a remote. Also pushes the git-annex
- branch. - branch.

View file

@ -8,7 +8,7 @@
module RemoteDaemon.Common module RemoteDaemon.Common
( liftAnnex ( liftAnnex
, inLocalRepo , inLocalRepo
, checkNewShas , checkShouldFetch
, ConnectionStatus(..) , ConnectionStatus(..)
, robustConnection , robustConnection
) where ) where
@ -35,6 +35,13 @@ liftAnnex (TransportHandle _ annexstate) a = do
inLocalRepo :: TransportHandle -> (Git.Repo -> IO a) -> IO a inLocalRepo :: TransportHandle -> (Git.Repo -> IO a) -> IO a
inLocalRepo (TransportHandle (LocalRepo g) _) a = a g inLocalRepo (TransportHandle (LocalRepo g) _) a = a g
-- Check if some shas should be fetched from the remote,
-- and presumably later merged.
checkShouldFetch :: RemoteGitConfig -> TransportHandle -> [Git.Sha] -> IO Bool
checkShouldFetch gc transporthandle shas
| remoteAnnexPull gc = checkNewShas transporthandle shas
| otherwise = return False
-- Check if any of the shas are actally new in the local git repo, -- Check if any of the shas are actally new in the local git repo,
-- to avoid unnecessary fetching. -- to avoid unnecessary fetching.
checkNewShas :: TransportHandle -> [Git.Sha] -> IO Bool checkNewShas :: TransportHandle -> [Git.Sha] -> IO Bool

View file

@ -36,7 +36,7 @@ transportUsingCmd cmd params rr@(RemoteRepo r gc) url h@(TransportHandle (LocalR
transportUsingCmd' cmd params rr url transporthandle ichan ochan transportUsingCmd' cmd params rr url transporthandle ichan ochan
transportUsingCmd' :: FilePath -> [CommandParam] -> Transport transportUsingCmd' :: FilePath -> [CommandParam] -> Transport
transportUsingCmd' cmd params (RemoteRepo r _) url transporthandle ichan ochan = transportUsingCmd' cmd params (RemoteRepo r gc) url transporthandle ichan ochan =
robustConnection 1 $ do robustConnection 1 $ do
(Just toh, Just fromh, Just errh, pid) <- (Just toh, Just fromh, Just errh, pid) <-
createProcess (proc cmd (toCommand params)) createProcess (proc cmd (toCommand params))
@ -74,7 +74,7 @@ transportUsingCmd' cmd params (RemoteRepo r _) url transporthandle ichan ochan =
send (CONNECTED url) send (CONNECTED url)
handlestdout fromh handlestdout fromh
Just (SshRemote.CHANGED (ChangedRefs shas)) -> do Just (SshRemote.CHANGED (ChangedRefs shas)) -> do
whenM (checkNewShas transporthandle shas) $ whenM (checkShouldFetch gc transporthandle shas) $
fetch fetch
handlestdout fromh handlestdout fromh
-- avoid reconnect on protocol error -- avoid reconnect on protocol error

View file

@ -129,7 +129,7 @@ serveClient th u r q = bracket setup cleanup start
-- Connect to peer's tor hidden service. -- Connect to peer's tor hidden service.
transport :: Transport transport :: Transport
transport (RemoteRepo r _) url@(RemoteURI uri) th ichan ochan = transport (RemoteRepo r gc) url@(RemoteURI uri) th ichan ochan =
case unformatP2PAddress (show uri) of case unformatP2PAddress (show uri) of
Nothing -> return () Nothing -> return ()
Just addr -> robustConnection 1 $ do Just addr -> robustConnection 1 $ do
@ -168,7 +168,7 @@ transport (RemoteRepo r _) url@(RemoteURI uri) th ichan ochan =
v <- runNetProto conn P2P.notifyChange v <- runNetProto conn P2P.notifyChange
case v of case v of
Right (Just (ChangedRefs shas)) -> do Right (Just (ChangedRefs shas)) -> do
whenM (checkNewShas th shas) $ whenM (checkShouldFetch gc th shas) $
fetch fetch
handlepeer conn handlepeer conn
_ -> return ConnectionClosed _ -> return ConnectionClosed

View file

@ -181,6 +181,8 @@ data RemoteGitConfig = RemoteGitConfig
, remoteAnnexCostCommand :: Maybe String , remoteAnnexCostCommand :: Maybe String
, remoteAnnexIgnore :: Bool , remoteAnnexIgnore :: Bool
, remoteAnnexSync :: Bool , remoteAnnexSync :: Bool
, remoteAnnexPull :: Bool
, remoteAnnexPush :: Bool
, remoteAnnexReadOnly :: Bool , remoteAnnexReadOnly :: Bool
, remoteAnnexVerify :: Bool , remoteAnnexVerify :: Bool
, remoteAnnexTrustLevel :: Maybe String , remoteAnnexTrustLevel :: Maybe String
@ -218,6 +220,8 @@ extractRemoteGitConfig r remotename = RemoteGitConfig
, remoteAnnexCostCommand = notempty $ getmaybe "cost-command" , remoteAnnexCostCommand = notempty $ getmaybe "cost-command"
, remoteAnnexIgnore = getbool "ignore" False , remoteAnnexIgnore = getbool "ignore" False
, remoteAnnexSync = getbool "sync" True , remoteAnnexSync = getbool "sync" True
, remoteAnnexPull = getbool "pull" True
, remoteAnnexPush = getbool "push" True
, remoteAnnexReadOnly = getbool "readonly" False , remoteAnnexReadOnly = getbool "readonly" False
, remoteAnnexVerify = getbool "verify" True , remoteAnnexVerify = getbool "verify" True
, remoteAnnexTrustLevel = notempty $ getmaybe "trustlevel" , remoteAnnexTrustLevel = notempty $ getmaybe "trustlevel"

View file

@ -29,6 +29,9 @@ accepting connections from other nodes and serving up the contents of the
repository. This is only done if you first run `git annex enable-tor`. repository. This is only done if you first run `git annex enable-tor`.
Use `git annex p2p` to configure access to tor-annex remotes. Use `git annex p2p` to configure access to tor-annex remotes.
Note that when `remote.<name>.annex-pull` is set to false, the remotedaemon
will avoid fetching changes from that remote.
# OPTIONS # OPTIONS
* `--foreground` * `--foreground`

View file

@ -8,10 +8,7 @@ git annex sync `[remote ...]`
# DESCRIPTION # DESCRIPTION
Use this command when you want to synchronize the local repository with This command synchronizes the local repository with its remotes.
one or more of its remotes. You can specify the remotes (or remote
groups) to sync with by name; the default if none are specified is to
sync with all remotes.
The sync process involves first committing any local changes to files The sync process involves first committing any local changes to files
that have previously been added to the repository, that have previously been added to the repository,
@ -36,6 +33,12 @@ by running "git annex sync" on the remote.
# OPTIONS # OPTIONS
* `[remote]`
By default, all remotes are synced, except for remotes that have
`remote.<name>.annex-sync` set to false. By specifying the names
of remotes (or remote groups), you can control which ones to sync with.
* `--fast` * `--fast`
Only sync with the remotes with the lowest annex-cost value configured. Only sync with the remotes with the lowest annex-cost value configured.
@ -52,11 +55,21 @@ by running "git annex sync" on the remote.
* `--pull`, `--no-pull` * `--pull`, `--no-pull`
By default, git pulls from remotes. Use --no-pull to disable. By default, git pulls from remotes. Use --no-pull to disable all pulling.
When `remote.<name>.annex-pull` or `remote.<name>.annex-sync`
are set to false, pulling is disabled for those remotes, and using
`--pull` will not enable it.
* `--push`, `--no-push` * `--push`, `--no-push`
By default, git pushes to remotes. Use --no-push to disable. By default, git pushes changes to remotes.
Use --no-push to disable all pushing.
When `remote.<name>.annex-push` or `remote.<name>.annex-sync` are
set to false, or `remote.<name>.annex-readonly` is set to true,
pushing is disabled for those remotes, and using `--push` will not enable
it.
* `--content`, `--no-content` * `--content`, `--no-content`
@ -64,7 +77,7 @@ by running "git annex sync" on the remote.
The --content option causes the content of files in the work tree The --content option causes the content of files in the work tree
to also be uploaded and downloaded as necessary. to also be uploaded and downloaded as necessary.
The annex.synccontent configuration can be set to true to make content The `annex.synccontent` configuration can be set to true to make content
be synced by default. be synced by default.
Normally this tries to get each annexed file in the work tree Normally this tries to get each annexed file in the work tree

View file

@ -1156,7 +1156,18 @@ Here are all the supported configuration settings.
* `remote.<name>.annex-sync` * `remote.<name>.annex-sync`
If set to `false`, prevents git-annex sync (and the git-annex assistant) If set to `false`, prevents git-annex sync (and the git-annex assistant)
from syncing with this remote. from syncing with this remote by default. However, `git annex sync <name>`
can still be used to sync with the remote.
* `remote.<name>.annex-pull`
If set to `false`, prevents git-annex sync (and the git-annex assistant
etc) from ever pulling (or fetching) from the remote.
* `remote.<name>.annex-push`
If set to `false`, prevents git-annex sync (and the git-annex assistant
etc) from ever pushing to the remote.
* `remote.<name>.annex-readonly` * `remote.<name>.annex-readonly`

View file

@ -0,0 +1,19 @@
[[!comment format=mdwn
username="joey"
subject="""comment 1"""
date="2017-04-05T16:11:57Z"
content="""
Setting `remote.<name>.annex-readonly=true` prevents git-annex sync
from pushing changes to the remote. It also prevents any git-annex command
from copying annexed file contents to the remote, or deleting annexed file
contents. So I think it's ideal for this kind of situation.
There does seem to be room for configs to prevent sync from pulling/pushing
without making the remote fully readonly. For example, the remote might be
a source of content, that only knows about the files it added and not other
files in the local repository, so dropping files from it should be allowed
but not pushing to it.
So, I've added `remote.<name>.annex-push` and
`remote.<name>.annex-pull`.
"""]]