added an optional cost= configuration to all special remotes

Note that when this is specified and an older git-annex is used to
enableremote such a special remote, it will simply ignore the cost= field
and use whatever the default cost is.

In passing, fixed adb to support the remote.name.cost and
remote.name.cost-command configs.

Sponsored-by: Dartmouth College's DANDI project
This commit is contained in:
Joey Hess 2023-01-12 13:42:28 -04:00
parent 400ce29a94
commit cfaae7e931
No known key found for this signature in database
GPG key ID: DB12DB0FF05F8F38
25 changed files with 83 additions and 42 deletions

View file

@ -1,6 +1,6 @@
{- git-annex special remote configuration {- git-annex special remote configuration
- -
- Copyright 2019-2020 Joey Hess <id@joeyh.name> - Copyright 2019-2023 Joey Hess <id@joeyh.name>
- -
- Licensed under the GNU AGPL version 3 or higher. - Licensed under the GNU AGPL version 3 or higher.
-} -}
@ -17,9 +17,11 @@ import Types.UUID
import Types.ProposedAccepted import Types.ProposedAccepted
import Types.RemoteConfig import Types.RemoteConfig
import Types.GitConfig import Types.GitConfig
import Config.Cost
import qualified Data.Map as M import qualified Data.Map as M
import qualified Data.Set as S import qualified Data.Set as S
import Text.Read
import Data.Typeable import Data.Typeable
import GHC.Stack import GHC.Stack
@ -56,6 +58,9 @@ typeField = Accepted "type"
autoEnableField :: RemoteConfigField autoEnableField :: RemoteConfigField
autoEnableField = Accepted "autoenable" autoEnableField = Accepted "autoenable"
costField :: RemoteConfigField
costField = Accepted "cost"
encryptionField :: RemoteConfigField encryptionField :: RemoteConfigField
encryptionField = Accepted "encryption" encryptionField = Accepted "encryption"
@ -106,6 +111,8 @@ commonFieldParsers =
(FieldDesc "type of special remote") (FieldDesc "type of special remote")
, trueFalseParser autoEnableField (Just False) , trueFalseParser autoEnableField (Just False)
(FieldDesc "automatically enable special remote") (FieldDesc "automatically enable special remote")
, costParser costField
(FieldDesc "default cost of this special remote")
, yesNoParser exportTreeField (Just False) , yesNoParser exportTreeField (Just False)
(FieldDesc "export trees of files to this remote") (FieldDesc "export trees of files to this remote")
, yesNoParser importTreeField (Just False) , yesNoParser importTreeField (Just False)
@ -252,6 +259,13 @@ trueFalseParser' "true" = Just True
trueFalseParser' "false" = Just False trueFalseParser' "false" = Just False
trueFalseParser' _ = Nothing trueFalseParser' _ = Nothing
costParser :: RemoteConfigField -> FieldDesc -> RemoteConfigFieldParser
costParser f fd = genParser readcost f Nothing fd
(Just (ValueDesc "a number"))
where
readcost :: String -> Maybe Cost
readcost = readMaybe
genParser genParser
:: Typeable t :: Typeable t
=> (String -> Maybe t) => (String -> Maybe t)

View file

@ -20,6 +20,8 @@ git-annex (10.20221213) UNRELEASED; urgency=medium
these provide additional names for the web special remote, and may these provide additional names for the web special remote, and may
also have their own additional configuration and cost. also have their own additional configuration and cost.
* web: Add urlinclude and urlexclude configuration settings. * web: Add urlinclude and urlexclude configuration settings.
* Added an optional cost= configuration to all special remotes.
* adb: Support the remote.name.cost and remote.name.cost-command configs.
-- Joey Hess <id@joeyh.name> Mon, 12 Dec 2022 13:04:54 -0400 -- Joey Hess <id@joeyh.name> Mon, 12 Dec 2022 13:04:54 -0400

View file

@ -1,6 +1,6 @@
{- Git configuration {- Git configuration
- -
- Copyright 2011-2020 Joey Hess <id@joeyh.name> - Copyright 2011-2023 Joey Hess <id@joeyh.name>
- -
- Licensed under the GNU AGPL version 3 or higher. - Licensed under the GNU AGPL version 3 or higher.
-} -}
@ -24,7 +24,9 @@ import Config.Cost
import Config.DynamicConfig import Config.DynamicConfig
import Types.Availability import Types.Availability
import Types.GitConfig import Types.GitConfig
import Types.RemoteConfig
import Git.Types import Git.Types
import Annex.SpecialRemote.Config
{- Looks up a setting in git config. This is not as efficient as using the {- Looks up a setting in git config. This is not as efficient as using the
- GitConfig type. -} - GitConfig type. -}
@ -51,14 +53,17 @@ reloadConfig = Annex.changeGitRepo =<< inRepo Git.Config.reRead
unsetConfig :: ConfigKey -> Annex () unsetConfig :: ConfigKey -> Annex ()
unsetConfig key = void $ inRepo $ Git.Config.unset key unsetConfig key = void $ inRepo $ Git.Config.unset key
{- Calculates cost for a remote. Either the specific default, or as configured {- Gets cost for a remote. As configured by
- by remote.<name>.annex-cost, or if remote.<name>.annex-cost-command - remote.<name>.annex-cost, or if remote.<name>.annex-cost-command
- is set and prints a number, that is used. -} - is set and prints a number, that is used. If neither is set,
remoteCost :: RemoteGitConfig -> Cost -> Annex Cost - using the cost field from the ParsedRemoteConfig, and if it is not set,
remoteCost c d = fromMaybe d <$> remoteCost' c - the specified default. -}
remoteCost :: RemoteGitConfig -> ParsedRemoteConfig -> Cost -> Annex Cost
remoteCost gc pc d = fromMaybe d <$> remoteCost' gc pc
remoteCost' :: RemoteGitConfig -> Annex (Maybe Cost) remoteCost' :: RemoteGitConfig -> ParsedRemoteConfig -> Annex (Maybe Cost)
remoteCost' = liftIO . getDynamicConfig . remoteAnnexCost remoteCost' gc pc = maybe (getRemoteConfigValue costField pc) Just
<$> liftIO (getDynamicConfig $ remoteAnnexCost gc)
setRemoteCost :: Git.Repo -> Cost -> Annex () setRemoteCost :: Git.Repo -> Cost -> Annex ()
setRemoteCost r c = setConfig (remoteAnnexConfig r "cost") (show c) setRemoteCost r c = setConfig (remoteAnnexConfig r "cost") (show c)

View file

@ -15,6 +15,7 @@ import Types.Creds
import Types.Export import Types.Export
import Types.Import import Types.Import
import qualified Git import qualified Git
import Config
import Config.Cost import Config.Cost
import Remote.Helper.Special import Remote.Helper.Special
import Remote.Helper.ExportImport import Remote.Helper.ExportImport
@ -70,11 +71,12 @@ oldandroidField = Accepted "oldandroid"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
let this = Remote
{ uuid = u
-- adb operates over USB or wifi, so is not as cheap -- adb operates over USB or wifi, so is not as cheap
-- as local, but not too expensive -- as local, but not too expensive
, cost = semiExpensiveRemoteCost cst <- remoteCost gc c semiExpensiveRemoteCost
let this = Remote
{ uuid = u
, cost = cst
, name = Git.repoDescribe r , name = Git.repoDescribe r
, storeKey = storeKeyDummy , storeKey = storeKeyDummy
, retrieveKeyFile = retrieveKeyFileDummy , retrieveKeyFile = retrieveKeyFileDummy

View file

@ -60,8 +60,8 @@ list _autoinit = do
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r _ rc gc rs = do gen r _ rc gc rs = do
cst <- remoteCost gc expensiveRemoteCost
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc c expensiveRemoteCost
return $ Just Remote return $ Just Remote
{ uuid = bitTorrentUUID { uuid = bitTorrentUUID
, cost = cst , cost = cst

View file

@ -75,7 +75,7 @@ appendonlyField = Accepted "appendonly"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc $ cst <- remoteCost gc c $
if borgLocal borgrepo if borgLocal borgrepo
then nearlyCheapRemoteCost then nearlyCheapRemoteCost
else expensiveRemoteCost else expensiveRemoteCost

View file

@ -65,7 +65,7 @@ gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
bupr <- liftIO $ bup2GitRemote buprepo bupr <- liftIO $ bup2GitRemote buprepo
cst <- remoteCost gc $ cst <- remoteCost gc c $
if bupLocal buprepo if bupLocal buprepo
then nearlyCheapRemoteCost then nearlyCheapRemoteCost
else expensiveRemoteCost else expensiveRemoteCost

View file

@ -55,7 +55,7 @@ ddarrepoField = Accepted "ddarrepo"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc $ cst <- remoteCost gc c $
if ddarLocal ddarrepo if ddarLocal ddarrepo
then nearlyCheapRemoteCost then nearlyCheapRemoteCost
else expensiveRemoteCost else expensiveRemoteCost

View file

@ -73,7 +73,7 @@ ignoreinodesField = Accepted "ignoreinodes"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc cheapRemoteCost cst <- remoteCost gc c cheapRemoteCost
let chunkconfig = getChunkConfig c let chunkconfig = getChunkConfig c
cow <- liftIO newCopyCoWTried cow <- liftIO newCopyCoWTried
let ii = IgnoreInodes $ fromMaybe True $ let ii = IgnoreInodes $ fromMaybe True $

View file

@ -67,7 +67,7 @@ gen r u rc gc rs
-- readonly mode only downloads urls; does not use external program -- readonly mode only downloads urls; does not use external program
| externaltype == "readonly" = do | externaltype == "readonly" = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc expensiveRemoteCost cst <- remoteCost gc c expensiveRemoteCost
let rmt = mk c cst GloballyAvailable let rmt = mk c cst GloballyAvailable
Nothing Nothing
(externalInfo externaltype) (externalInfo externaltype)
@ -86,7 +86,7 @@ gen r u rc gc rs
external <- newExternal externaltype (Just u) c (Just gc) external <- newExternal externaltype (Just u) c (Just gc)
(Git.remoteName r) (Just rs) (Git.remoteName r) (Just rs)
Annex.addCleanupAction (RemoteCleanup u) $ stopExternal external Annex.addCleanupAction (RemoteCleanup u) $ stopExternal external
cst <- getCost external r gc cst <- getCost external r gc c
avail <- getAvailability external r gc avail <- getAvailability external r gc
exportsupported <- if exportTree c exportsupported <- if exportTree c
then checkExportSupported' external then checkExportSupported' external
@ -755,9 +755,9 @@ respErrorMessage req err
{- Caches the cost in the git config to avoid needing to start up an {- Caches the cost in the git config to avoid needing to start up an
- external special remote every time time just to ask it what its - external special remote every time time just to ask it what its
- cost is. -} - cost is. -}
getCost :: External -> Git.Repo -> RemoteGitConfig -> Annex Cost getCost :: External -> Git.Repo -> RemoteGitConfig -> ParsedRemoteConfig -> Annex Cost
getCost external r gc = getCost external r gc pc =
(go =<< remoteCost' gc) `catchNonAsync` const (pure defcst) (go =<< remoteCost' gc pc) `catchNonAsync` const (pure defcst)
where where
go (Just c) = return c go (Just c) = return c
go Nothing = do go Nothing = do

View file

@ -127,8 +127,10 @@ gen baser u rc gc rs = do
gen' :: Git.Repo -> UUID -> ParsedRemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen' :: Git.Repo -> UUID -> ParsedRemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen' r u c gc rs = do gen' r u c gc rs = do
cst <- remoteCost gc $ cst <- remoteCost gc c $
if repoCheap r then nearlyCheapRemoteCost else expensiveRemoteCost if repoCheap r
then nearlyCheapRemoteCost
else expensiveRemoteCost
let (rsynctransport, rsyncurl, accessmethod) = rsyncTransportToObjects r gc let (rsynctransport, rsyncurl, accessmethod) = rsyncTransportToObjects r gc
protectsargs <- liftIO Remote.Rsync.probeRsyncProtectsArgs protectsargs <- liftIO Remote.Rsync.probeRsyncProtectsArgs
let rsyncopts = Remote.Rsync.genRsyncOpts protectsargs c gc rsynctransport rsyncurl let rsyncopts = Remote.Rsync.genRsyncOpts protectsargs c gc rsynctransport rsyncurl

View file

@ -174,7 +174,7 @@ gen r u rc gc rs
Nothing -> do Nothing -> do
st <- mkState r u gc st <- mkState r u gc
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
go st c <$> remoteCost gc defcst go st c <$> remoteCost gc c defcst
Just addr -> Remote.P2P.chainGen addr r u rc gc rs Just addr -> Remote.P2P.chainGen addr r u rc gc rs
where where
defcst = if repoCheap r then cheapRemoteCost else expensiveRemoteCost defcst = if repoCheap r then cheapRemoteCost else expensiveRemoteCost

View file

@ -93,7 +93,7 @@ gen r u rc gc rs = do
else pure r else pure r
sem <- liftIO $ MSemN.new 1 sem <- liftIO $ MSemN.new 1
h <- liftIO $ newTVarIO $ LFSHandle Nothing Nothing sem r' gc h <- liftIO $ newTVarIO $ LFSHandle Nothing Nothing sem r' gc
cst <- remoteCost gc expensiveRemoteCost cst <- remoteCost gc c expensiveRemoteCost
let specialcfg = (specialRemoteCfg c) let specialcfg = (specialRemoteCfg c)
-- chunking would not improve git-lfs -- chunking would not improve git-lfs
{ chunkConfig = NoChunks { chunkConfig = NoChunks

View file

@ -63,9 +63,10 @@ fileprefixField :: RemoteConfigField
fileprefixField = Accepted "fileprefix" fileprefixField = Accepted "fileprefix"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = new gen r u rc gc rs = do
<$> parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
<*> remoteCost gc veryExpensiveRemoteCost cst <- remoteCost gc c veryExpensiveRemoteCost
return (new c cst)
where where
new c cst = Just $ specialRemote' specialcfg c new c cst = Just $ specialRemote' specialcfg c
(store this) (store this)

View file

@ -50,7 +50,7 @@ hooktypeField = Accepted "hooktype"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc expensiveRemoteCost cst <- remoteCost gc c expensiveRemoteCost
return $ Just $ specialRemote c return $ Just $ specialRemote c
(store hooktype) (store hooktype)
(retrieve hooktype) (retrieve hooktype)

View file

@ -50,7 +50,7 @@ urlField = Accepted "url"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc expensiveRemoteCost cst <- remoteCost gc c expensiveRemoteCost
let url = getRemoteConfigValue urlField c let url = getRemoteConfigValue urlField c
ll <- liftIO newLearnedLayout ll <- liftIO newLearnedLayout
return $ Just $ this url ll c cst return $ Just $ this url ll c cst

View file

@ -48,7 +48,7 @@ chainGen :: P2PAddress -> Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig ->
chainGen addr r u rc gc rs = do chainGen addr r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
connpool <- mkConnectionPool connpool <- mkConnectionPool
cst <- remoteCost gc veryExpensiveRemoteCost cst <- remoteCost gc c veryExpensiveRemoteCost
let protorunner = runProto u addr connpool let protorunner = runProto u addr connpool
let withconn = withConnection u addr connpool let withconn = withConnection u addr connpool
let this = Remote let this = Remote

View file

@ -75,7 +75,7 @@ rsyncUrlField = Accepted "rsyncurl"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc expensiveRemoteCost cst <- remoteCost gc c expensiveRemoteCost
(transport, url) <- rsyncTransport gc $ (transport, url) <- rsyncTransport gc $
fromMaybe (giveup "missing rsyncurl") $ remoteAnnexRsyncUrl gc fromMaybe (giveup "missing rsyncurl") $ remoteAnnexRsyncUrl gc
protectsargs <- liftIO probeRsyncProtectsArgs protectsargs <- liftIO probeRsyncProtectsArgs

View file

@ -187,7 +187,7 @@ mungekeysField = Accepted "mungekeys"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc expensiveRemoteCost cst <- remoteCost gc c expensiveRemoteCost
info <- extractS3Info c info <- extractS3Info c
hdl <- mkS3HandleVar c gc u hdl <- mkS3HandleVar c gc u
magic <- liftIO initMagicMime magic <- liftIO initMagicMime

View file

@ -79,7 +79,7 @@ furlField = Accepted "introducer-furl"
gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote) gen :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> RemoteStateHandle -> Annex (Maybe Remote)
gen r u rc gc rs = do gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
cst <- remoteCost gc expensiveRemoteCost cst <- remoteCost gc c expensiveRemoteCost
hdl <- liftIO $ TahoeHandle hdl <- liftIO $ TahoeHandle
<$> maybe (defaultTahoeConfigDir u) return (remoteAnnexTahoe gc) <$> maybe (defaultTahoeConfigDir u) return (remoteAnnexTahoe gc)
<*> newEmptyTMVarIO <*> newEmptyTMVarIO

View file

@ -72,7 +72,7 @@ gen r u rc gc rs = do
c <- parsedRemoteConfig remote rc c <- parsedRemoteConfig remote rc
new new
<$> pure c <$> pure c
<*> remoteCost gc expensiveRemoteCost <*> remoteCost gc c expensiveRemoteCost
<*> mkDavHandleVar c gc u <*> mkDavHandleVar c gc u
where where
new c cst hdl = Just $ specialRemote c new c cst hdl = Just $ specialRemote c

View file

@ -106,6 +106,12 @@ want to use `git annex renameremote`.
when the special remote does not need anything special to be done to get when the special remote does not need anything special to be done to get
it enabled. it enabled.
* `cost`
Specify this to override the default cost of the special remote.
This configuration can be overridden by the local git config,
eg remote.name.annex-cost.
* `uuid` * `uuid`
Normally, git-annex initremote generates a new UUID for the new special Normally, git-annex initremote generates a new UUID for the new special

View file

@ -128,13 +128,12 @@ about, and git-annex may use any of those urls for downloading a file.
If some urls are especially fast, or especially slow, you might want to If some urls are especially fast, or especially slow, you might want to
configure which urls git-annex prefers to use first, or should only use as configure which urls git-annex prefers to use first, or should only use as
a last resory. To accomplish that, you can create additional remotes, that a last resory. To accomplish that, you can create additional remotes, that
are web special remotes, and are configured to only be used for some urls. are web special remotes, and are configured to only be used for some urls,
Then it's simply a matter of configuring the cost of those remotes. and have a different cost than the web special remote.
For example, suppose that you want to prioritize using urls on "fasthost.com". For example, suppose that you want to prioritize using urls on "fasthost.com".
git-annex initremote --sameas=web fasthost type=web urlinclude='*//fasthost.com/*' git-annex initremote --sameas=web fasthost type=web urlinclude='*//fasthost.com/*' cost=150
git config remote.fasthost.annex-cost 150
Now, `git-annex get` of a file that is on both fasthost.com and another url Now, `git-annex get` of a file that is on both fasthost.com and another url
will prefer to use the fasthost special remote, rather than the web special will prefer to use the fasthost special remote, rather than the web special
@ -145,8 +144,7 @@ remote, and use the other url.
Suppose that you want to avoid using urls on "slowhost.com", except Suppose that you want to avoid using urls on "slowhost.com", except
as a last resort. as a last resort.
git-annex initremote --sameas=web slowhost type=web urlinclude='*//slowhost.com/*' git-annex initremote --sameas=web slowhost type=web urlinclude='*//slowhost.com/*' cost=300
git config remote.slowhost.annex-cost 300
Now, `git-annex get` of a file that is on both slowhost.com and another url Now, `git-annex get` of a file that is on both slowhost.com and another url
will first try the fasthost remote. If fasthost does not support the url, will first try the fasthost remote. If fasthost does not support the url,

View file

@ -30,3 +30,5 @@ PS somehow I have some odd memory of seeing some config option to provide git-an
[[!meta author=yoh]] [[!meta author=yoh]]
[[!tag projects/dandi]] [[!tag projects/dandi]]
> [[fixed|done]] --[[Joey]]

View file

@ -0,0 +1,9 @@
[[!comment format=mdwn
username="joey"
subject="""comment 11"""
date="2023-01-12T17:38:38Z"
content="""
Went ahead and implememented cost=, so now all you need is:
git-annex initremote --sameas=web dandiapi type=web urlinclude='*//api.dandiarchive.org/*' cost=300
"""]]