Urls can now be claimed by remotes. This will allow creating, for example, a external special remote that handles magnet: and *.torrent urls.
This commit is contained in:
parent
ee27298b91
commit
30bf112185
28 changed files with 346 additions and 114 deletions
3
Annex.hs
3
Annex.hs
|
@ -63,6 +63,7 @@ import Types.CleanupActions
|
||||||
import Utility.Quvi (QuviVersion)
|
import Utility.Quvi (QuviVersion)
|
||||||
#endif
|
#endif
|
||||||
import Utility.InodeCache
|
import Utility.InodeCache
|
||||||
|
import Utility.Url
|
||||||
|
|
||||||
import "mtl" Control.Monad.Reader
|
import "mtl" Control.Monad.Reader
|
||||||
import Control.Concurrent
|
import Control.Concurrent
|
||||||
|
@ -128,6 +129,7 @@ data AnnexState = AnnexState
|
||||||
, useragent :: Maybe String
|
, useragent :: Maybe String
|
||||||
, errcounter :: Integer
|
, errcounter :: Integer
|
||||||
, unusedkeys :: Maybe (S.Set Key)
|
, unusedkeys :: Maybe (S.Set Key)
|
||||||
|
, tempurls :: M.Map Key URLString
|
||||||
#ifdef WITH_QUVI
|
#ifdef WITH_QUVI
|
||||||
, quviversion :: Maybe QuviVersion
|
, quviversion :: Maybe QuviVersion
|
||||||
#endif
|
#endif
|
||||||
|
@ -173,6 +175,7 @@ newState c r = AnnexState
|
||||||
, useragent = Nothing
|
, useragent = Nothing
|
||||||
, errcounter = 0
|
, errcounter = 0
|
||||||
, unusedkeys = Nothing
|
, unusedkeys = Nothing
|
||||||
|
, tempurls = M.empty
|
||||||
#ifdef WITH_QUVI
|
#ifdef WITH_QUVI
|
||||||
, quviversion = Nothing
|
, quviversion = Nothing
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -80,7 +80,7 @@ startDistributionDownload d = go =<< liftIO . newVersionLocation d =<< liftIO ol
|
||||||
where
|
where
|
||||||
go Nothing = debug ["Skipping redundant upgrade"]
|
go Nothing = debug ["Skipping redundant upgrade"]
|
||||||
go (Just dest) = do
|
go (Just dest) = do
|
||||||
liftAnnex $ setUrlPresent k u
|
liftAnnex $ setUrlPresent webUUID k u
|
||||||
hook <- asIO1 $ distributionDownloadComplete d dest cleanup
|
hook <- asIO1 $ distributionDownloadComplete d dest cleanup
|
||||||
modifyDaemonStatus_ $ \s -> s
|
modifyDaemonStatus_ $ \s -> s
|
||||||
{ transferHook = M.insert k hook (transferHook s) }
|
{ transferHook = M.insert k hook (transferHook s) }
|
||||||
|
@ -97,7 +97,7 @@ startDistributionDownload d = go =<< liftIO . newVersionLocation d =<< liftIO ol
|
||||||
}
|
}
|
||||||
cleanup = liftAnnex $ do
|
cleanup = liftAnnex $ do
|
||||||
lockContent k removeAnnex
|
lockContent k removeAnnex
|
||||||
setUrlMissing k u
|
setUrlMissing webUUID k u
|
||||||
logStatus k InfoMissing
|
logStatus k InfoMissing
|
||||||
|
|
||||||
{- Called once the download is done.
|
{- Called once the download is done.
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{- git-annex command
|
{- git-annex command
|
||||||
-
|
-
|
||||||
- Copyright 2011-2013 Joey Hess <joey@kitenet.net>
|
- Copyright 2011-2014 Joey Hess <joey@kitenet.net>
|
||||||
-
|
-
|
||||||
- Licensed under the GNU GPL version 3 or higher.
|
- Licensed under the GNU GPL version 3 or higher.
|
||||||
-}
|
-}
|
||||||
|
@ -19,6 +19,8 @@ import qualified Annex
|
||||||
import qualified Annex.Queue
|
import qualified Annex.Queue
|
||||||
import qualified Annex.Url as Url
|
import qualified Annex.Url as Url
|
||||||
import qualified Backend.URL
|
import qualified Backend.URL
|
||||||
|
import qualified Remote
|
||||||
|
import qualified Types.Remote as Remote
|
||||||
import Annex.Content
|
import Annex.Content
|
||||||
import Logs.Web
|
import Logs.Web
|
||||||
import Types.Key
|
import Types.Key
|
||||||
|
@ -26,6 +28,7 @@ import Types.KeySource
|
||||||
import Config
|
import Config
|
||||||
import Annex.Content.Direct
|
import Annex.Content.Direct
|
||||||
import Logs.Location
|
import Logs.Location
|
||||||
|
import Utility.Metered
|
||||||
import qualified Annex.Transfer as Transfer
|
import qualified Annex.Transfer as Transfer
|
||||||
#ifdef WITH_QUVI
|
#ifdef WITH_QUVI
|
||||||
import Annex.Quvi
|
import Annex.Quvi
|
||||||
|
@ -54,7 +57,71 @@ seek ps = do
|
||||||
withStrings (start relaxed f d) ps
|
withStrings (start relaxed f d) ps
|
||||||
|
|
||||||
start :: Bool -> Maybe FilePath -> Maybe Int -> String -> CommandStart
|
start :: Bool -> Maybe FilePath -> Maybe Int -> String -> CommandStart
|
||||||
start relaxed optfile pathdepth s = go $ fromMaybe bad $ parseURI s
|
start relaxed optfile pathdepth s = do
|
||||||
|
r <- Remote.claimingUrl s
|
||||||
|
if Remote.uuid r == webUUID
|
||||||
|
then startWeb relaxed optfile pathdepth s
|
||||||
|
else startRemote r relaxed optfile pathdepth s
|
||||||
|
|
||||||
|
startRemote :: Remote -> Bool -> Maybe FilePath -> Maybe Int -> String -> CommandStart
|
||||||
|
startRemote r relaxed optfile pathdepth s = do
|
||||||
|
url <- case Url.parseURIRelaxed s of
|
||||||
|
Nothing -> error $ "bad uri " ++ s
|
||||||
|
Just u -> pure u
|
||||||
|
pathmax <- liftIO $ fileNameLengthLimit "."
|
||||||
|
let file = choosefile $ url2file url pathdepth pathmax
|
||||||
|
showStart "addurl" file
|
||||||
|
showNote $ "using " ++ Remote.name r
|
||||||
|
next $ performRemote r relaxed s file
|
||||||
|
where
|
||||||
|
choosefile = flip fromMaybe optfile
|
||||||
|
|
||||||
|
performRemote :: Remote -> Bool -> URLString -> FilePath -> CommandPerform
|
||||||
|
performRemote r relaxed uri file = ifAnnexed file adduri geturi
|
||||||
|
where
|
||||||
|
loguri = setDownloader uri OtherDownloader
|
||||||
|
adduri = addUrlChecked relaxed loguri (Remote.uuid r) checkexistssize
|
||||||
|
checkexistssize key = do
|
||||||
|
res <- tryNonAsync $ Remote.checkUrl r uri
|
||||||
|
case res of
|
||||||
|
Left e -> do
|
||||||
|
warning (show e)
|
||||||
|
return (False, False)
|
||||||
|
Right Nothing ->
|
||||||
|
return (True, True)
|
||||||
|
Right (Just sz) ->
|
||||||
|
return (True, sz == fromMaybe sz (keySize key))
|
||||||
|
geturi = do
|
||||||
|
dummykey <- Backend.URL.fromUrl uri =<<
|
||||||
|
if relaxed
|
||||||
|
then return Nothing
|
||||||
|
else Remote.checkUrl r uri
|
||||||
|
liftIO $ createDirectoryIfMissing True (parentDir file)
|
||||||
|
next $ ifM (Annex.getState Annex.fast <||> pure relaxed)
|
||||||
|
( do
|
||||||
|
res <- tryNonAsync $ Remote.checkUrl r uri
|
||||||
|
case res of
|
||||||
|
Left e -> do
|
||||||
|
warning (show e)
|
||||||
|
return False
|
||||||
|
Right size -> do
|
||||||
|
key <- Backend.URL.fromUrl uri size
|
||||||
|
cleanup (Remote.uuid r) loguri file key Nothing
|
||||||
|
return True
|
||||||
|
, do
|
||||||
|
-- Set temporary url for the dummy key
|
||||||
|
-- so that the remote knows what url it
|
||||||
|
-- should use to download it.
|
||||||
|
setTempUrl dummykey uri
|
||||||
|
let downloader = Remote.retrieveKeyFile r dummykey (Just file)
|
||||||
|
ok <- isJust <$>
|
||||||
|
downloadWith downloader dummykey (Remote.uuid r) loguri file
|
||||||
|
removeTempUrl dummykey
|
||||||
|
return ok
|
||||||
|
)
|
||||||
|
|
||||||
|
startWeb :: Bool -> Maybe FilePath -> Maybe Int -> String -> CommandStart
|
||||||
|
startWeb relaxed optfile pathdepth s = go $ fromMaybe bad $ parseURI s
|
||||||
where
|
where
|
||||||
(s', downloader) = getDownloader s
|
(s', downloader) = getDownloader s
|
||||||
bad = fromMaybe (error $ "bad url " ++ s') $
|
bad = fromMaybe (error $ "bad url " ++ s') $
|
||||||
|
@ -62,7 +129,7 @@ start relaxed optfile pathdepth s = go $ fromMaybe bad $ parseURI s
|
||||||
choosefile = flip fromMaybe optfile
|
choosefile = flip fromMaybe optfile
|
||||||
go url = case downloader of
|
go url = case downloader of
|
||||||
QuviDownloader -> usequvi
|
QuviDownloader -> usequvi
|
||||||
DefaultDownloader ->
|
_ ->
|
||||||
#ifdef WITH_QUVI
|
#ifdef WITH_QUVI
|
||||||
ifM (quviSupported s')
|
ifM (quviSupported s')
|
||||||
( usequvi
|
( usequvi
|
||||||
|
@ -75,7 +142,7 @@ start relaxed optfile pathdepth s = go $ fromMaybe bad $ parseURI s
|
||||||
pathmax <- liftIO $ fileNameLengthLimit "."
|
pathmax <- liftIO $ fileNameLengthLimit "."
|
||||||
let file = choosefile $ url2file url pathdepth pathmax
|
let file = choosefile $ url2file url pathdepth pathmax
|
||||||
showStart "addurl" file
|
showStart "addurl" file
|
||||||
next $ perform relaxed s' file
|
next $ performWeb relaxed s' file
|
||||||
#ifdef WITH_QUVI
|
#ifdef WITH_QUVI
|
||||||
badquvi = error $ "quvi does not know how to download url " ++ s'
|
badquvi = error $ "quvi does not know how to download url " ++ s'
|
||||||
usequvi = do
|
usequvi = do
|
||||||
|
@ -96,7 +163,9 @@ performQuvi :: Bool -> URLString -> URLString -> FilePath -> CommandPerform
|
||||||
performQuvi relaxed pageurl videourl file = ifAnnexed file addurl geturl
|
performQuvi relaxed pageurl videourl file = ifAnnexed file addurl geturl
|
||||||
where
|
where
|
||||||
quviurl = setDownloader pageurl QuviDownloader
|
quviurl = setDownloader pageurl QuviDownloader
|
||||||
addurl key = next $ cleanup quviurl file key Nothing
|
addurl key = next $ do
|
||||||
|
cleanup webUUID quviurl file key Nothing
|
||||||
|
return True
|
||||||
geturl = next $ isJust <$> addUrlFileQuvi relaxed quviurl videourl file
|
geturl = next $ isJust <$> addUrlFileQuvi relaxed quviurl videourl file
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -106,7 +175,7 @@ addUrlFileQuvi relaxed quviurl videourl file = do
|
||||||
key <- Backend.URL.fromUrl quviurl Nothing
|
key <- Backend.URL.fromUrl quviurl Nothing
|
||||||
ifM (pure relaxed <||> Annex.getState Annex.fast)
|
ifM (pure relaxed <||> Annex.getState Annex.fast)
|
||||||
( do
|
( do
|
||||||
cleanup' quviurl file key Nothing
|
cleanup webUUID quviurl file key Nothing
|
||||||
return (Just key)
|
return (Just key)
|
||||||
, do
|
, do
|
||||||
{- Get the size, and use that to check
|
{- Get the size, and use that to check
|
||||||
|
@ -124,27 +193,31 @@ addUrlFileQuvi relaxed quviurl videourl file = do
|
||||||
downloadUrl [videourl] tmp
|
downloadUrl [videourl] tmp
|
||||||
if ok
|
if ok
|
||||||
then do
|
then do
|
||||||
cleanup' quviurl file key (Just tmp)
|
cleanup webUUID quviurl file key (Just tmp)
|
||||||
return (Just key)
|
return (Just key)
|
||||||
else return Nothing
|
else return Nothing
|
||||||
)
|
)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
perform :: Bool -> URLString -> FilePath -> CommandPerform
|
performWeb :: Bool -> URLString -> FilePath -> CommandPerform
|
||||||
perform relaxed url file = ifAnnexed file addurl geturl
|
performWeb relaxed url file = ifAnnexed file addurl geturl
|
||||||
where
|
where
|
||||||
geturl = next $ isJust <$> addUrlFile relaxed url file
|
geturl = next $ isJust <$> addUrlFile relaxed url file
|
||||||
addurl key
|
addurl = addUrlChecked relaxed url webUUID checkexistssize
|
||||||
|
checkexistssize = Url.withUrlOptions . Url.check url . keySize
|
||||||
|
|
||||||
|
addUrlChecked :: Bool -> URLString -> UUID -> (Key -> Annex (Bool, Bool)) -> Key -> CommandPerform
|
||||||
|
addUrlChecked relaxed url u checkexistssize key
|
||||||
| relaxed = do
|
| relaxed = do
|
||||||
setUrlPresent key url
|
setUrlPresent u key url
|
||||||
next $ return True
|
next $ return True
|
||||||
| otherwise = ifM (elem url <$> getUrls key)
|
| otherwise = ifM (elem url <$> getUrls key)
|
||||||
( stop
|
( stop
|
||||||
, do
|
, do
|
||||||
(exists, samesize) <- Url.withUrlOptions $ Url.check url (keySize key)
|
(exists, samesize) <- checkexistssize key
|
||||||
if exists && samesize
|
if exists && samesize
|
||||||
then do
|
then do
|
||||||
setUrlPresent key url
|
setUrlPresent u key url
|
||||||
next $ return True
|
next $ return True
|
||||||
else do
|
else do
|
||||||
warning $ "while adding a new url to an already annexed file, " ++ if exists
|
warning $ "while adding a new url to an already annexed file, " ++ if exists
|
||||||
|
@ -158,21 +231,27 @@ addUrlFile relaxed url file = do
|
||||||
liftIO $ createDirectoryIfMissing True (parentDir file)
|
liftIO $ createDirectoryIfMissing True (parentDir file)
|
||||||
ifM (Annex.getState Annex.fast <||> pure relaxed)
|
ifM (Annex.getState Annex.fast <||> pure relaxed)
|
||||||
( nodownload relaxed url file
|
( nodownload relaxed url file
|
||||||
, do
|
, downloadWeb url file
|
||||||
showAction $ "downloading " ++ url ++ " "
|
|
||||||
download url file
|
|
||||||
)
|
)
|
||||||
|
|
||||||
download :: URLString -> FilePath -> Annex (Maybe Key)
|
downloadWeb :: URLString -> FilePath -> Annex (Maybe Key)
|
||||||
download url file = do
|
downloadWeb url file = do
|
||||||
{- Generate a dummy key to use for this download, before we can
|
|
||||||
- examine the file and find its real key. This allows resuming
|
|
||||||
- downloads, as the dummy key for a given url is stable. -}
|
|
||||||
dummykey <- addSizeUrlKey url =<< Backend.URL.fromUrl url Nothing
|
dummykey <- addSizeUrlKey url =<< Backend.URL.fromUrl url Nothing
|
||||||
|
let downloader f _ = do
|
||||||
|
showOutput
|
||||||
|
downloadUrl [url] f
|
||||||
|
showAction $ "downloading " ++ url ++ " "
|
||||||
|
downloadWith downloader dummykey webUUID url file
|
||||||
|
|
||||||
|
{- The Key should be a dummy key, based on the URL, which is used
|
||||||
|
- for this download, before we can examine the file and find its real key.
|
||||||
|
- For resuming downloads to work, the dummy key for a given url should be
|
||||||
|
- stable. -}
|
||||||
|
downloadWith :: (FilePath -> MeterUpdate -> Annex Bool) -> Key -> UUID -> URLString -> FilePath -> Annex (Maybe Key)
|
||||||
|
downloadWith downloader dummykey u url file =
|
||||||
prepGetViaTmpChecked dummykey Nothing $ do
|
prepGetViaTmpChecked dummykey Nothing $ do
|
||||||
tmp <- fromRepo $ gitAnnexTmpObjectLocation dummykey
|
tmp <- fromRepo $ gitAnnexTmpObjectLocation dummykey
|
||||||
showOutput
|
ifM (runtransfer tmp)
|
||||||
ifM (runtransfer dummykey tmp)
|
|
||||||
( do
|
( do
|
||||||
backend <- chooseBackend file
|
backend <- chooseBackend file
|
||||||
let source = KeySource
|
let source = KeySource
|
||||||
|
@ -184,15 +263,15 @@ download url file = do
|
||||||
case k of
|
case k of
|
||||||
Nothing -> return Nothing
|
Nothing -> return Nothing
|
||||||
Just (key, _) -> do
|
Just (key, _) -> do
|
||||||
cleanup' url file key (Just tmp)
|
cleanup u url file key (Just tmp)
|
||||||
return (Just key)
|
return (Just key)
|
||||||
, return Nothing
|
, return Nothing
|
||||||
)
|
)
|
||||||
where
|
where
|
||||||
runtransfer dummykey tmp = Transfer.notifyTransfer Transfer.Download (Just file) $
|
runtransfer tmp = Transfer.notifyTransfer Transfer.Download (Just file) $
|
||||||
Transfer.download webUUID dummykey (Just file) Transfer.forwardRetry $ const $ do
|
Transfer.download webUUID dummykey (Just file) Transfer.forwardRetry $ \p -> do
|
||||||
liftIO $ createDirectoryIfMissing True (parentDir tmp)
|
liftIO $ createDirectoryIfMissing True (parentDir tmp)
|
||||||
downloadUrl [url] tmp
|
downloader tmp p
|
||||||
|
|
||||||
{- Hits the url to get the size, if available.
|
{- Hits the url to get the size, if available.
|
||||||
-
|
-
|
||||||
|
@ -204,16 +283,11 @@ addSizeUrlKey url key = do
|
||||||
size <- snd <$> Url.withUrlOptions (Url.exists url)
|
size <- snd <$> Url.withUrlOptions (Url.exists url)
|
||||||
return $ key { keySize = size }
|
return $ key { keySize = size }
|
||||||
|
|
||||||
cleanup :: URLString -> FilePath -> Key -> Maybe FilePath -> Annex Bool
|
cleanup :: UUID -> URLString -> FilePath -> Key -> Maybe FilePath -> Annex ()
|
||||||
cleanup url file key mtmp = do
|
cleanup u url file key mtmp = do
|
||||||
cleanup' url file key mtmp
|
|
||||||
return True
|
|
||||||
|
|
||||||
cleanup' :: URLString -> FilePath -> Key -> Maybe FilePath -> Annex ()
|
|
||||||
cleanup' url file key mtmp = do
|
|
||||||
when (isJust mtmp) $
|
when (isJust mtmp) $
|
||||||
logStatus key InfoPresent
|
logStatus key InfoPresent
|
||||||
setUrlPresent key url
|
setUrlPresent u key url
|
||||||
Command.Add.addLink file key Nothing
|
Command.Add.addLink file key Nothing
|
||||||
whenM isDirect $ do
|
whenM isDirect $ do
|
||||||
void $ addAssociatedFile key file
|
void $ addAssociatedFile key file
|
||||||
|
@ -230,7 +304,7 @@ nodownload relaxed url file = do
|
||||||
if exists
|
if exists
|
||||||
then do
|
then do
|
||||||
key <- Backend.URL.fromUrl url size
|
key <- Backend.URL.fromUrl url size
|
||||||
cleanup' url file key Nothing
|
cleanup webUUID url file key Nothing
|
||||||
return (Just key)
|
return (Just key)
|
||||||
else do
|
else do
|
||||||
warning $ "unable to access url: " ++ url
|
warning $ "unable to access url: " ++ url
|
||||||
|
@ -245,8 +319,11 @@ url2file url pathdepth pathmax = case pathdepth of
|
||||||
| depth < 0 -> frombits $ reverse . take (negate depth) . reverse
|
| depth < 0 -> frombits $ reverse . take (negate depth) . reverse
|
||||||
| otherwise -> error "bad --pathdepth"
|
| otherwise -> error "bad --pathdepth"
|
||||||
where
|
where
|
||||||
fullurl = uriRegName auth ++ uriPath url ++ uriQuery url
|
fullurl = concat
|
||||||
|
[ maybe "" uriRegName (uriAuthority url)
|
||||||
|
, uriPath url
|
||||||
|
, uriQuery url
|
||||||
|
]
|
||||||
frombits a = intercalate "/" $ a urlbits
|
frombits a = intercalate "/" $ a urlbits
|
||||||
urlbits = map (truncateFilePath pathmax . sanitizeFilePath) $
|
urlbits = map (truncateFilePath pathmax . sanitizeFilePath) $
|
||||||
filter (not . null) $ split "/" fullurl
|
filter (not . null) $ split "/" fullurl
|
||||||
auth = fromMaybe (error $ "bad url " ++ show url) $ uriAuthority url
|
|
||||||
|
|
|
@ -16,6 +16,7 @@ import qualified Command.Add
|
||||||
import Logs.Web
|
import Logs.Web
|
||||||
import Logs.Location
|
import Logs.Location
|
||||||
import Utility.CopyFile
|
import Utility.CopyFile
|
||||||
|
import qualified Remote
|
||||||
|
|
||||||
cmd :: [Command]
|
cmd :: [Command]
|
||||||
cmd = [notDirect $ command "rekey"
|
cmd = [notDirect $ command "rekey"
|
||||||
|
@ -61,8 +62,9 @@ cleanup file oldkey newkey = do
|
||||||
-- If the old key had some associated urls, record them for
|
-- If the old key had some associated urls, record them for
|
||||||
-- the new key as well.
|
-- the new key as well.
|
||||||
urls <- getUrls oldkey
|
urls <- getUrls oldkey
|
||||||
unless (null urls) $
|
forM_ urls $ \url -> do
|
||||||
mapM_ (setUrlPresent newkey) urls
|
r <- Remote.claimingUrl url
|
||||||
|
setUrlPresent (Remote.uuid r) newkey url
|
||||||
|
|
||||||
-- Update symlink to use the new key.
|
-- Update symlink to use the new key.
|
||||||
liftIO $ removeFile file
|
liftIO $ removeFile file
|
||||||
|
|
|
@ -10,6 +10,7 @@ module Command.RmUrl where
|
||||||
import Common.Annex
|
import Common.Annex
|
||||||
import Command
|
import Command
|
||||||
import Logs.Web
|
import Logs.Web
|
||||||
|
import qualified Remote
|
||||||
|
|
||||||
cmd :: [Command]
|
cmd :: [Command]
|
||||||
cmd = [notBareRepo $
|
cmd = [notBareRepo $
|
||||||
|
@ -26,5 +27,9 @@ start (file, url) = flip whenAnnexed file $ \_ key -> do
|
||||||
|
|
||||||
cleanup :: String -> Key -> CommandCleanup
|
cleanup :: String -> Key -> CommandCleanup
|
||||||
cleanup url key = do
|
cleanup url key = do
|
||||||
setUrlMissing key url
|
r <- Remote.claimingUrl url
|
||||||
|
let url' = if Remote.uuid r == webUUID
|
||||||
|
then url
|
||||||
|
else setDownloader url OtherDownloader
|
||||||
|
setUrlMissing (Remote.uuid r) key url'
|
||||||
return True
|
return True
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{- git-annex command
|
{- git-annex command
|
||||||
-
|
-
|
||||||
- Copyright 2010 Joey Hess <joey@kitenet.net>
|
- Copyright 2010-2014 Joey Hess <joey@kitenet.net>
|
||||||
-
|
-
|
||||||
- Licensed under the GNU GPL version 3 or higher.
|
- Licensed under the GNU GPL version 3 or higher.
|
||||||
-}
|
-}
|
||||||
|
@ -13,6 +13,7 @@ import Common.Annex
|
||||||
import Command
|
import Command
|
||||||
import Remote
|
import Remote
|
||||||
import Logs.Trust
|
import Logs.Trust
|
||||||
|
import Logs.Web
|
||||||
|
|
||||||
cmd :: [Command]
|
cmd :: [Command]
|
||||||
cmd = [noCommit $ withOptions (jsonOption : keyOptions) $
|
cmd = [noCommit $ withOptions (jsonOption : keyOptions) $
|
||||||
|
@ -57,9 +58,17 @@ perform remotemap key = do
|
||||||
untrustedheader = "The following untrusted locations may also have copies:\n"
|
untrustedheader = "The following untrusted locations may also have copies:\n"
|
||||||
|
|
||||||
performRemote :: Key -> Remote -> Annex ()
|
performRemote :: Key -> Remote -> Annex ()
|
||||||
performRemote key remote = maybe noop go $ whereisKey remote
|
performRemote key remote = do
|
||||||
where
|
ls <- (++)
|
||||||
go a = do
|
<$> askremote
|
||||||
ls <- a key
|
<*> claimedurls
|
||||||
unless (null ls) $ showLongNote $ unlines $
|
unless (null ls) $ showLongNote $ unlines $
|
||||||
map (\l -> name remote ++ ": " ++ l) ls
|
map (\l -> name remote ++ ": " ++ l) ls
|
||||||
|
where
|
||||||
|
askremote = maybe (pure []) (flip id key) (whereisKey remote)
|
||||||
|
claimedurls = do
|
||||||
|
us <- map fst
|
||||||
|
. filter (\(_, d) -> d == OtherDownloader)
|
||||||
|
. map getDownloader
|
||||||
|
<$> getUrls key
|
||||||
|
filterM (\u -> (==) <$> pure remote <*> claimingUrl u) us
|
||||||
|
|
55
Logs/Web.hs
55
Logs/Web.hs
|
@ -1,6 +1,6 @@
|
||||||
{- Web url logs.
|
{- Web url logs.
|
||||||
-
|
-
|
||||||
- Copyright 2011, 2013 Joey Hess <joey@kitenet.net>
|
- Copyright 2011-2014 Joey Hess <joey@kitenet.net>
|
||||||
-
|
-
|
||||||
- Licensed under the GNU GPL version 3 or higher.
|
- Licensed under the GNU GPL version 3 or higher.
|
||||||
-}
|
-}
|
||||||
|
@ -16,12 +16,16 @@ module Logs.Web (
|
||||||
Downloader(..),
|
Downloader(..),
|
||||||
getDownloader,
|
getDownloader,
|
||||||
setDownloader,
|
setDownloader,
|
||||||
|
setTempUrl,
|
||||||
|
removeTempUrl,
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import qualified Data.ByteString.Lazy.Char8 as L
|
import qualified Data.ByteString.Lazy.Char8 as L
|
||||||
|
import qualified Data.Map as M
|
||||||
import Data.Tuple.Utils
|
import Data.Tuple.Utils
|
||||||
|
|
||||||
import Common.Annex
|
import Common.Annex
|
||||||
|
import qualified Annex
|
||||||
import Logs
|
import Logs
|
||||||
import Logs.Presence
|
import Logs.Presence
|
||||||
import Logs.Location
|
import Logs.Location
|
||||||
|
@ -37,7 +41,10 @@ webUUID = UUID "00000000-0000-0000-0000-000000000001"
|
||||||
|
|
||||||
{- Gets all urls that a key might be available from. -}
|
{- Gets all urls that a key might be available from. -}
|
||||||
getUrls :: Key -> Annex [URLString]
|
getUrls :: Key -> Annex [URLString]
|
||||||
getUrls key = go $ urlLogFile key : oldurlLogs key
|
getUrls key = do
|
||||||
|
l <- go $ urlLogFile key : oldurlLogs key
|
||||||
|
tmpl <- Annex.getState (maybeToList . M.lookup key . Annex.tempurls)
|
||||||
|
return (tmpl ++ l)
|
||||||
where
|
where
|
||||||
go [] = return []
|
go [] = return []
|
||||||
go (l:ls) = do
|
go (l:ls) = do
|
||||||
|
@ -49,19 +56,18 @@ getUrls key = go $ urlLogFile key : oldurlLogs key
|
||||||
getUrlsWithPrefix :: Key -> String -> Annex [URLString]
|
getUrlsWithPrefix :: Key -> String -> Annex [URLString]
|
||||||
getUrlsWithPrefix key prefix = filter (prefix `isPrefixOf`) <$> getUrls key
|
getUrlsWithPrefix key prefix = filter (prefix `isPrefixOf`) <$> getUrls key
|
||||||
|
|
||||||
setUrlPresent :: Key -> URLString -> Annex ()
|
setUrlPresent :: UUID -> Key -> URLString -> Annex ()
|
||||||
setUrlPresent key url = do
|
setUrlPresent uuid key url = do
|
||||||
us <- getUrls key
|
us <- getUrls key
|
||||||
unless (url `elem` us) $ do
|
unless (url `elem` us) $ do
|
||||||
addLog (urlLogFile key) =<< logNow InfoPresent url
|
addLog (urlLogFile key) =<< logNow InfoPresent url
|
||||||
-- update location log to indicate that the web has the key
|
logChange key uuid InfoPresent
|
||||||
logChange key webUUID InfoPresent
|
|
||||||
|
|
||||||
setUrlMissing :: Key -> URLString -> Annex ()
|
setUrlMissing :: UUID -> Key -> URLString -> Annex ()
|
||||||
setUrlMissing key url = do
|
setUrlMissing uuid key url = do
|
||||||
addLog (urlLogFile key) =<< logNow InfoMissing url
|
addLog (urlLogFile key) =<< logNow InfoMissing url
|
||||||
whenM (null <$> getUrls key) $
|
whenM (null <$> getUrls key) $
|
||||||
logChange key webUUID InfoMissing
|
logChange key uuid InfoMissing
|
||||||
|
|
||||||
{- Finds all known urls. -}
|
{- Finds all known urls. -}
|
||||||
knownUrls :: Annex [URLString]
|
knownUrls :: Annex [URLString]
|
||||||
|
@ -81,18 +87,27 @@ knownUrls = do
|
||||||
geturls Nothing = return []
|
geturls Nothing = return []
|
||||||
geturls (Just logsha) = getLog . L.unpack <$> catObject logsha
|
geturls (Just logsha) = getLog . L.unpack <$> catObject logsha
|
||||||
|
|
||||||
data Downloader = DefaultDownloader | QuviDownloader
|
setTempUrl :: Key -> URLString -> Annex ()
|
||||||
|
setTempUrl key url = Annex.changeState $ \s ->
|
||||||
|
s { Annex.tempurls = M.insert key url (Annex.tempurls s) }
|
||||||
|
|
||||||
|
removeTempUrl :: Key -> Annex ()
|
||||||
|
removeTempUrl key = Annex.changeState $ \s ->
|
||||||
|
s { Annex.tempurls = M.delete key (Annex.tempurls s) }
|
||||||
|
|
||||||
|
data Downloader = WebDownloader | QuviDownloader | OtherDownloader
|
||||||
|
deriving (Eq)
|
||||||
|
|
||||||
|
{- To keep track of how an url is downloaded, it's mangled slightly in
|
||||||
|
- the log. For quvi, "quvi:" is prefixed. For urls that are handled by
|
||||||
|
- some other remote, ":" is prefixed. -}
|
||||||
|
setDownloader :: URLString -> Downloader -> String
|
||||||
|
setDownloader u WebDownloader = u
|
||||||
|
setDownloader u QuviDownloader = "quvi:" ++ u
|
||||||
|
setDownloader u OtherDownloader = ":" ++ u
|
||||||
|
|
||||||
{- Determines the downloader for an URL.
|
|
||||||
-
|
|
||||||
- Some URLs are not downloaded by normal means, and this is indicated
|
|
||||||
- by prefixing them with downloader: when they are recorded in the url
|
|
||||||
- logs. -}
|
|
||||||
getDownloader :: URLString -> (URLString, Downloader)
|
getDownloader :: URLString -> (URLString, Downloader)
|
||||||
getDownloader u = case separate (== ':') u of
|
getDownloader u = case separate (== ':') u of
|
||||||
("quvi", u') -> (u', QuviDownloader)
|
("quvi", u') -> (u', QuviDownloader)
|
||||||
_ -> (u, DefaultDownloader)
|
("", u') -> (u', OtherDownloader)
|
||||||
|
_ -> (u, WebDownloader)
|
||||||
setDownloader :: URLString -> Downloader -> URLString
|
|
||||||
setDownloader u DefaultDownloader = u
|
|
||||||
setDownloader u QuviDownloader = "quvi:" ++ u
|
|
||||||
|
|
13
Remote.hs
13
Remote.hs
|
@ -45,7 +45,8 @@ module Remote (
|
||||||
forceTrust,
|
forceTrust,
|
||||||
logStatus,
|
logStatus,
|
||||||
checkAvailable,
|
checkAvailable,
|
||||||
isXMPPRemote
|
isXMPPRemote,
|
||||||
|
claimingUrl,
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
|
@ -60,6 +61,7 @@ import Annex.UUID
|
||||||
import Logs.UUID
|
import Logs.UUID
|
||||||
import Logs.Trust
|
import Logs.Trust
|
||||||
import Logs.Location hiding (logStatus)
|
import Logs.Location hiding (logStatus)
|
||||||
|
import Logs.Web
|
||||||
import Remote.List
|
import Remote.List
|
||||||
import Config
|
import Config
|
||||||
import Git.Types (RemoteName)
|
import Git.Types (RemoteName)
|
||||||
|
@ -318,3 +320,12 @@ hasKey r k = either (Left . show) Right <$> tryNonAsync (checkPresent r k)
|
||||||
|
|
||||||
hasKeyCheap :: Remote -> Bool
|
hasKeyCheap :: Remote -> Bool
|
||||||
hasKeyCheap = checkPresentCheap
|
hasKeyCheap = checkPresentCheap
|
||||||
|
|
||||||
|
{- The web special remote claims urls by default. -}
|
||||||
|
claimingUrl :: URLString -> Annex Remote
|
||||||
|
claimingUrl url = do
|
||||||
|
rs <- remoteList
|
||||||
|
let web = Prelude.head $ filter (\r -> uuid r == webUUID) rs
|
||||||
|
fromMaybe web <$> firstM checkclaim rs
|
||||||
|
where
|
||||||
|
checkclaim = maybe (pure False) (flip id url) . claimUrl
|
||||||
|
|
|
@ -75,6 +75,7 @@ gen r u c gc = do
|
||||||
, mkUnavailable = return Nothing
|
, mkUnavailable = return Nothing
|
||||||
, getInfo = return [("repo", buprepo)]
|
, getInfo = return [("repo", buprepo)]
|
||||||
, claimUrl = Nothing
|
, claimUrl = Nothing
|
||||||
|
, checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
return $ Just $ specialRemote' specialcfg c
|
return $ Just $ specialRemote' specialcfg c
|
||||||
(simplyPrepare $ store this buprepo)
|
(simplyPrepare $ store this buprepo)
|
||||||
|
|
|
@ -72,6 +72,7 @@ gen r u c gc = do
|
||||||
, mkUnavailable = return Nothing
|
, mkUnavailable = return Nothing
|
||||||
, getInfo = return [("repo", ddarrepo)]
|
, getInfo = return [("repo", ddarrepo)]
|
||||||
, claimUrl = Nothing
|
, claimUrl = Nothing
|
||||||
|
, checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
ddarrepo = fromMaybe (error "missing ddarrepo") $ remoteAnnexDdarRepo gc
|
ddarrepo = fromMaybe (error "missing ddarrepo") $ remoteAnnexDdarRepo gc
|
||||||
specialcfg = (specialRemoteCfg c)
|
specialcfg = (specialRemoteCfg c)
|
||||||
|
|
|
@ -69,7 +69,8 @@ gen r u c gc = do
|
||||||
mkUnavailable = gen r u c $
|
mkUnavailable = gen r u c $
|
||||||
gc { remoteAnnexDirectory = Just "/dev/null" },
|
gc { remoteAnnexDirectory = Just "/dev/null" },
|
||||||
getInfo = return [("directory", dir)],
|
getInfo = return [("directory", dir)],
|
||||||
claimUrl = Nothing
|
claimUrl = Nothing,
|
||||||
|
checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
where
|
where
|
||||||
dir = fromMaybe (error "missing directory") $ remoteAnnexDirectory gc
|
dir = fromMaybe (error "missing directory") $ remoteAnnexDirectory gc
|
||||||
|
|
|
@ -70,7 +70,8 @@ gen r u c gc = do
|
||||||
mkUnavailable = gen r u c $
|
mkUnavailable = gen r u c $
|
||||||
gc { remoteAnnexExternalType = Just "!dne!" },
|
gc { remoteAnnexExternalType = Just "!dne!" },
|
||||||
getInfo = return [("externaltype", externaltype)],
|
getInfo = return [("externaltype", externaltype)],
|
||||||
claimUrl = Just (claimurl external)
|
claimUrl = Just (claimurl external),
|
||||||
|
checkUrl = checkurl external
|
||||||
}
|
}
|
||||||
where
|
where
|
||||||
externaltype = fromMaybe (error "missing externaltype") (remoteAnnexExternalType gc)
|
externaltype = fromMaybe (error "missing externaltype") (remoteAnnexExternalType gc)
|
||||||
|
@ -217,8 +218,10 @@ handleRequest' lck external req mp responsehandler
|
||||||
state <- fromMaybe ""
|
state <- fromMaybe ""
|
||||||
<$> getRemoteState (externalUUID external) key
|
<$> getRemoteState (externalUUID external) key
|
||||||
send $ VALUE state
|
send $ VALUE state
|
||||||
handleRemoteRequest (SETURLPRESENT key url) = setUrlPresent key url
|
handleRemoteRequest (SETURLPRESENT key url) =
|
||||||
handleRemoteRequest (SETURLMISSING key url) = setUrlMissing key url
|
setUrlPresent (externalUUID external) key url
|
||||||
|
handleRemoteRequest (SETURLMISSING key url) =
|
||||||
|
setUrlMissing (externalUUID external) key url
|
||||||
handleRemoteRequest (GETURLS key prefix) = do
|
handleRemoteRequest (GETURLS key prefix) = do
|
||||||
mapM_ (send . VALUE) =<< getUrlsWithPrefix key prefix
|
mapM_ (send . VALUE) =<< getUrlsWithPrefix key prefix
|
||||||
send (VALUE "") -- end of list
|
send (VALUE "") -- end of list
|
||||||
|
@ -425,3 +428,11 @@ claimurl external url =
|
||||||
UNSUPPORTED_REQUEST -> Just $ return False
|
UNSUPPORTED_REQUEST -> Just $ return False
|
||||||
_ -> Nothing
|
_ -> Nothing
|
||||||
|
|
||||||
|
checkurl :: External -> URLString -> Annex (Maybe Integer)
|
||||||
|
checkurl external url =
|
||||||
|
handleRequest external (CHECKURL url) Nothing $ \req -> case req of
|
||||||
|
CHECKURL_SIZE sz -> Just $ return $ Just sz
|
||||||
|
CHECKURL_SIZEUNKNOWN -> Just $ return Nothing
|
||||||
|
CHECKURL_FAILURE errmsg -> Just $ error errmsg
|
||||||
|
UNSUPPORTED_REQUEST -> error "CHECKURL not implemented by external special remote"
|
||||||
|
_ -> Nothing
|
||||||
|
|
13
Remote/External/Types.hs
vendored
13
Remote/External/Types.hs
vendored
|
@ -92,6 +92,7 @@ data Request
|
||||||
| GETCOST
|
| GETCOST
|
||||||
| GETAVAILABILITY
|
| GETAVAILABILITY
|
||||||
| CLAIMURL URLString
|
| CLAIMURL URLString
|
||||||
|
| CHECKURL URLString
|
||||||
| TRANSFER Direction Key FilePath
|
| TRANSFER Direction Key FilePath
|
||||||
| CHECKPRESENT Key
|
| CHECKPRESENT Key
|
||||||
| REMOVE Key
|
| REMOVE Key
|
||||||
|
@ -109,6 +110,7 @@ instance Proto.Sendable Request where
|
||||||
formatMessage GETCOST = ["GETCOST"]
|
formatMessage GETCOST = ["GETCOST"]
|
||||||
formatMessage GETAVAILABILITY = ["GETAVAILABILITY"]
|
formatMessage GETAVAILABILITY = ["GETAVAILABILITY"]
|
||||||
formatMessage (CLAIMURL url) = [ "CLAIMURL", Proto.serialize url ]
|
formatMessage (CLAIMURL url) = [ "CLAIMURL", Proto.serialize url ]
|
||||||
|
formatMessage (CHECKURL url) = [ "CHECKURL", Proto.serialize url ]
|
||||||
formatMessage (TRANSFER direction key file) =
|
formatMessage (TRANSFER direction key file) =
|
||||||
[ "TRANSFER"
|
[ "TRANSFER"
|
||||||
, Proto.serialize direction
|
, Proto.serialize direction
|
||||||
|
@ -135,6 +137,9 @@ data Response
|
||||||
| INITREMOTE_FAILURE ErrorMsg
|
| INITREMOTE_FAILURE ErrorMsg
|
||||||
| CLAIMURL_SUCCESS
|
| CLAIMURL_SUCCESS
|
||||||
| CLAIMURL_FAILURE
|
| CLAIMURL_FAILURE
|
||||||
|
| CHECKURL_SIZE Size
|
||||||
|
| CHECKURL_SIZEUNKNOWN
|
||||||
|
| CHECKURL_FAILURE ErrorMsg
|
||||||
| UNSUPPORTED_REQUEST
|
| UNSUPPORTED_REQUEST
|
||||||
deriving (Show)
|
deriving (Show)
|
||||||
|
|
||||||
|
@ -154,6 +159,9 @@ instance Proto.Receivable Response where
|
||||||
parseCommand "INITREMOTE-FAILURE" = Proto.parse1 INITREMOTE_FAILURE
|
parseCommand "INITREMOTE-FAILURE" = Proto.parse1 INITREMOTE_FAILURE
|
||||||
parseCommand "CLAIMURL-SUCCESS" = Proto.parse0 CLAIMURL_SUCCESS
|
parseCommand "CLAIMURL-SUCCESS" = Proto.parse0 CLAIMURL_SUCCESS
|
||||||
parseCommand "CLAIMURL-FAILURE" = Proto.parse0 CLAIMURL_FAILURE
|
parseCommand "CLAIMURL-FAILURE" = Proto.parse0 CLAIMURL_FAILURE
|
||||||
|
parseCommand "CHECKURL-SIZE" = Proto.parse1 CHECKURL_SIZE
|
||||||
|
parseCommand "CHECKURL-SIZEUNKNOWN" = Proto.parse0 CHECKURL_SIZEUNKNOWN
|
||||||
|
parseCommand "CHECKURL-FAILURE" = Proto.parse1 CHECKURL_FAILURE
|
||||||
parseCommand "UNSUPPORTED-REQUEST" = Proto.parse0 UNSUPPORTED_REQUEST
|
parseCommand "UNSUPPORTED-REQUEST" = Proto.parse0 UNSUPPORTED_REQUEST
|
||||||
parseCommand _ = Proto.parseFail
|
parseCommand _ = Proto.parseFail
|
||||||
|
|
||||||
|
@ -225,6 +233,7 @@ instance Proto.Receivable AsyncMessage where
|
||||||
type ErrorMsg = String
|
type ErrorMsg = String
|
||||||
type Setting = String
|
type Setting = String
|
||||||
type ProtocolVersion = Int
|
type ProtocolVersion = Int
|
||||||
|
type Size = Integer
|
||||||
|
|
||||||
supportedProtocolVersions :: [ProtocolVersion]
|
supportedProtocolVersions :: [ProtocolVersion]
|
||||||
supportedProtocolVersions = [1]
|
supportedProtocolVersions = [1]
|
||||||
|
@ -253,6 +262,10 @@ instance Proto.Serializable Cost where
|
||||||
serialize = show
|
serialize = show
|
||||||
deserialize = readish
|
deserialize = readish
|
||||||
|
|
||||||
|
instance Proto.Serializable Size where
|
||||||
|
serialize = show
|
||||||
|
deserialize = readish
|
||||||
|
|
||||||
instance Proto.Serializable Availability where
|
instance Proto.Serializable Availability where
|
||||||
serialize GloballyAvailable = "GLOBAL"
|
serialize GloballyAvailable = "GLOBAL"
|
||||||
serialize LocallyAvailable = "LOCAL"
|
serialize LocallyAvailable = "LOCAL"
|
||||||
|
|
|
@ -123,6 +123,7 @@ gen' r u c gc = do
|
||||||
, mkUnavailable = return Nothing
|
, mkUnavailable = return Nothing
|
||||||
, getInfo = return $ gitRepoInfo r
|
, getInfo = return $ gitRepoInfo r
|
||||||
, claimUrl = Nothing
|
, claimUrl = Nothing
|
||||||
|
, checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
return $ Just $ specialRemote' specialcfg c
|
return $ Just $ specialRemote' specialcfg c
|
||||||
(simplyPrepare $ store this rsyncopts)
|
(simplyPrepare $ store this rsyncopts)
|
||||||
|
|
|
@ -161,6 +161,7 @@ gen r u c gc
|
||||||
, mkUnavailable = unavailable r u c gc
|
, mkUnavailable = unavailable r u c gc
|
||||||
, getInfo = return $ gitRepoInfo r
|
, getInfo = return $ gitRepoInfo r
|
||||||
, claimUrl = Nothing
|
, claimUrl = Nothing
|
||||||
|
, checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
unavailable :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> Annex (Maybe Remote)
|
unavailable :: Git.Repo -> UUID -> RemoteConfig -> RemoteGitConfig -> Annex (Maybe Remote)
|
||||||
|
|
|
@ -69,7 +69,8 @@ gen r u c gc = new <$> remoteCost gc veryExpensiveRemoteCost
|
||||||
mkUnavailable = return Nothing,
|
mkUnavailable = return Nothing,
|
||||||
getInfo = includeCredsInfo c (AWS.creds u) $
|
getInfo = includeCredsInfo c (AWS.creds u) $
|
||||||
[ ("glacier vault", getVault c) ],
|
[ ("glacier vault", getVault c) ],
|
||||||
claimUrl = Nothing
|
claimUrl = Nothing,
|
||||||
|
checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
specialcfg = (specialRemoteCfg c)
|
specialcfg = (specialRemoteCfg c)
|
||||||
-- Disabled until jobList gets support for chunks.
|
-- Disabled until jobList gets support for chunks.
|
||||||
|
|
|
@ -62,7 +62,8 @@ gen r u c gc = do
|
||||||
mkUnavailable = gen r u c $
|
mkUnavailable = gen r u c $
|
||||||
gc { remoteAnnexHookType = Just "!dne!" },
|
gc { remoteAnnexHookType = Just "!dne!" },
|
||||||
getInfo = return [("hooktype", hooktype)],
|
getInfo = return [("hooktype", hooktype)],
|
||||||
claimUrl = Nothing
|
claimUrl = Nothing,
|
||||||
|
checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
where
|
where
|
||||||
hooktype = fromMaybe (error "missing hooktype") $ remoteAnnexHookType gc
|
hooktype = fromMaybe (error "missing hooktype") $ remoteAnnexHookType gc
|
||||||
|
|
|
@ -85,6 +85,7 @@ gen r u c gc = do
|
||||||
, mkUnavailable = return Nothing
|
, mkUnavailable = return Nothing
|
||||||
, getInfo = return [("url", url)]
|
, getInfo = return [("url", url)]
|
||||||
, claimUrl = Nothing
|
, claimUrl = Nothing
|
||||||
|
, checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
where
|
where
|
||||||
specialcfg = (specialRemoteCfg c)
|
specialcfg = (specialRemoteCfg c)
|
||||||
|
|
|
@ -93,7 +93,8 @@ gen r u c gc = do
|
||||||
else Nothing
|
else Nothing
|
||||||
, Just ("partsize", maybe "unlimited" (roughSize storageUnits False) (getPartSize c))
|
, Just ("partsize", maybe "unlimited" (roughSize storageUnits False) (getPartSize c))
|
||||||
],
|
],
|
||||||
claimUrl = Nothing
|
claimUrl = Nothing,
|
||||||
|
checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
s3Setup :: Maybe UUID -> Maybe CredPair -> RemoteConfig -> Annex (RemoteConfig, UUID)
|
s3Setup :: Maybe UUID -> Maybe CredPair -> RemoteConfig -> Annex (RemoteConfig, UUID)
|
||||||
|
@ -163,7 +164,7 @@ store r h = fileStorer $ \k f p -> do
|
||||||
_ -> singlepartupload k f p
|
_ -> singlepartupload k f p
|
||||||
-- Store public URL to item in Internet Archive.
|
-- Store public URL to item in Internet Archive.
|
||||||
when (isIA (hinfo h) && not (isChunkKey k)) $
|
when (isIA (hinfo h) && not (isChunkKey k)) $
|
||||||
setUrlPresent k (iaKeyUrl r k)
|
setUrlPresent webUUID k (iaKeyUrl r k)
|
||||||
return True
|
return True
|
||||||
where
|
where
|
||||||
singlepartupload k f p = do
|
singlepartupload k f p = do
|
||||||
|
|
|
@ -86,7 +86,8 @@ gen r u c gc = do
|
||||||
remotetype = remote,
|
remotetype = remote,
|
||||||
mkUnavailable = return Nothing,
|
mkUnavailable = return Nothing,
|
||||||
getInfo = return [],
|
getInfo = return [],
|
||||||
claimUrl = Nothing
|
claimUrl = Nothing,
|
||||||
|
checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
tahoeSetup :: Maybe UUID -> Maybe CredPair -> RemoteConfig -> Annex (RemoteConfig, UUID)
|
tahoeSetup :: Maybe UUID -> Maybe CredPair -> RemoteConfig -> Annex (RemoteConfig, UUID)
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
{- Web remotes.
|
{- Web remote.
|
||||||
-
|
-
|
||||||
- Copyright 2011 Joey Hess <joey@kitenet.net>
|
- Copyright 2011 Joey Hess <joey@kitenet.net>
|
||||||
-
|
-
|
||||||
|
@ -52,7 +52,7 @@ gen r _ c gc =
|
||||||
removeKey = dropKey,
|
removeKey = dropKey,
|
||||||
checkPresent = checkKey,
|
checkPresent = checkKey,
|
||||||
checkPresentCheap = False,
|
checkPresentCheap = False,
|
||||||
whereisKey = Just getUrls,
|
whereisKey = Just getWebUrls,
|
||||||
remoteFsck = Nothing,
|
remoteFsck = Nothing,
|
||||||
repairRepo = Nothing,
|
repairRepo = Nothing,
|
||||||
config = c,
|
config = c,
|
||||||
|
@ -64,11 +64,12 @@ gen r _ c gc =
|
||||||
remotetype = remote,
|
remotetype = remote,
|
||||||
mkUnavailable = return Nothing,
|
mkUnavailable = return Nothing,
|
||||||
getInfo = return [],
|
getInfo = return [],
|
||||||
claimUrl = Nothing -- implicitly claims all urls
|
claimUrl = Nothing, -- implicitly claims all urls
|
||||||
|
checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
downloadKey :: Key -> AssociatedFile -> FilePath -> MeterUpdate -> Annex Bool
|
downloadKey :: Key -> AssociatedFile -> FilePath -> MeterUpdate -> Annex Bool
|
||||||
downloadKey key _file dest _p = get =<< getUrls key
|
downloadKey key _file dest _p = get =<< getWebUrls key
|
||||||
where
|
where
|
||||||
get [] = do
|
get [] = do
|
||||||
warning "no known url"
|
warning "no known url"
|
||||||
|
@ -86,7 +87,7 @@ downloadKey key _file dest _p = get =<< getUrls key
|
||||||
warning "quvi support needed for this url"
|
warning "quvi support needed for this url"
|
||||||
return False
|
return False
|
||||||
#endif
|
#endif
|
||||||
DefaultDownloader -> downloadUrl [u'] dest
|
_ -> downloadUrl [u'] dest
|
||||||
|
|
||||||
downloadKeyCheap :: Key -> FilePath -> Annex Bool
|
downloadKeyCheap :: Key -> FilePath -> Annex Bool
|
||||||
downloadKeyCheap _ _ = return False
|
downloadKeyCheap _ _ = return False
|
||||||
|
@ -98,12 +99,12 @@ uploadKey _ _ _ = do
|
||||||
|
|
||||||
dropKey :: Key -> Annex Bool
|
dropKey :: Key -> Annex Bool
|
||||||
dropKey k = do
|
dropKey k = do
|
||||||
mapM_ (setUrlMissing k) =<< getUrls k
|
mapM_ (setUrlMissing webUUID k) =<< getWebUrls k
|
||||||
return True
|
return True
|
||||||
|
|
||||||
checkKey :: Key -> Annex Bool
|
checkKey :: Key -> Annex Bool
|
||||||
checkKey key = do
|
checkKey key = do
|
||||||
us <- getUrls key
|
us <- getWebUrls key
|
||||||
if null us
|
if null us
|
||||||
then return False
|
then return False
|
||||||
else either error return =<< checkKey' key us
|
else either error return =<< checkKey' key us
|
||||||
|
@ -118,7 +119,7 @@ checkKey' key us = firsthit us (Right False) $ \u -> do
|
||||||
#else
|
#else
|
||||||
return $ Left "quvi support needed for this url"
|
return $ Left "quvi support needed for this url"
|
||||||
#endif
|
#endif
|
||||||
DefaultDownloader -> do
|
_ -> do
|
||||||
Url.withUrlOptions $ catchMsgIO .
|
Url.withUrlOptions $ catchMsgIO .
|
||||||
Url.checkBoth u' (keySize key)
|
Url.checkBoth u' (keySize key)
|
||||||
where
|
where
|
||||||
|
@ -128,3 +129,9 @@ checkKey' key us = firsthit us (Right False) $ \u -> do
|
||||||
case r of
|
case r of
|
||||||
Right _ -> return r
|
Right _ -> return r
|
||||||
Left _ -> firsthit rest r a
|
Left _ -> firsthit rest r a
|
||||||
|
|
||||||
|
getWebUrls :: Key -> Annex [URLString]
|
||||||
|
getWebUrls key = filter supported <$> getUrls key
|
||||||
|
where
|
||||||
|
supported u = snd (getDownloader u)
|
||||||
|
`elem` [WebDownloader, QuviDownloader]
|
||||||
|
|
|
@ -74,7 +74,8 @@ gen r u c gc = new <$> remoteCost gc expensiveRemoteCost
|
||||||
mkUnavailable = gen r u (M.insert "url" "http://!dne!/" c) gc,
|
mkUnavailable = gen r u (M.insert "url" "http://!dne!/" c) gc,
|
||||||
getInfo = includeCredsInfo c (davCreds u) $
|
getInfo = includeCredsInfo c (davCreds u) $
|
||||||
[("url", fromMaybe "unknown" (M.lookup "url" c))],
|
[("url", fromMaybe "unknown" (M.lookup "url" c))],
|
||||||
claimUrl = Nothing
|
claimUrl = Nothing,
|
||||||
|
checkUrl = const $ return Nothing
|
||||||
}
|
}
|
||||||
chunkconfig = getChunkConfig c
|
chunkconfig = getChunkConfig c
|
||||||
|
|
||||||
|
|
|
@ -103,7 +103,12 @@ data RemoteA a = Remote {
|
||||||
-- Information about the remote, for git annex info to display.
|
-- Information about the remote, for git annex info to display.
|
||||||
getInfo :: a [(String, String)],
|
getInfo :: a [(String, String)],
|
||||||
-- Some remotes can download from an url (or uri).
|
-- Some remotes can download from an url (or uri).
|
||||||
claimUrl :: Maybe (URLString -> a Bool)
|
claimUrl :: Maybe (URLString -> a Bool),
|
||||||
|
-- Checks that the url is accessible, and gets the size of its
|
||||||
|
-- content. Returns Nothing if the url is accessible, but
|
||||||
|
-- its size cannot be determined inexpensively.
|
||||||
|
-- Throws an exception if the url is inaccessible.
|
||||||
|
checkUrl :: URLString -> a (Maybe Integer)
|
||||||
}
|
}
|
||||||
|
|
||||||
instance Show (RemoteA a) where
|
instance Show (RemoteA a) where
|
||||||
|
|
3
debian/changelog
vendored
3
debian/changelog
vendored
|
@ -4,6 +4,9 @@ git-annex (5.20141204) UNRELEASED; urgency=medium
|
||||||
Thanks, Jon Ander Peñalba.
|
Thanks, Jon Ander Peñalba.
|
||||||
* External special remote protocol now includes commands for setting
|
* External special remote protocol now includes commands for setting
|
||||||
and getting the urls associated with a key.
|
and getting the urls associated with a key.
|
||||||
|
* Urls can now be claimed by remotes. This will allow creating,
|
||||||
|
for example, a external special remote that handles magnet: and
|
||||||
|
*.torrent urls.
|
||||||
|
|
||||||
-- Joey Hess <id@joeyh.name> Fri, 05 Dec 2014 13:42:08 -0400
|
-- Joey Hess <id@joeyh.name> Fri, 05 Dec 2014 13:42:08 -0400
|
||||||
|
|
||||||
|
|
|
@ -125,10 +125,16 @@ replying with `UNSUPPORTED-REQUEST` is acceptable.
|
||||||
If the remote replies with `UNSUPPORTED-REQUEST`, its availability
|
If the remote replies with `UNSUPPORTED-REQUEST`, its availability
|
||||||
is assumed to be global. So, only remotes that are only reachable
|
is assumed to be global. So, only remotes that are only reachable
|
||||||
locally need to worry about implementing this.
|
locally need to worry about implementing this.
|
||||||
* `CLAIMURL Value`
|
* `CLAIMURL Url`
|
||||||
Asks the remote if it wishes to claim responsibility for downloading
|
Asks the remote if it wishes to claim responsibility for downloading
|
||||||
an url. If so, the remote should send back an `CLAIMURL-SUCCESS` reply.
|
an url. If so, the remote should send back an `CLAIMURL-SUCCESS` reply.
|
||||||
If not, it can send `CLAIMURL-FAILURE`.
|
If not, it can send `CLAIMURL-FAILURE`.
|
||||||
|
* `CHECKURL Url`
|
||||||
|
Asks the remote to check if the url's content can currently be downloaded
|
||||||
|
(without downloading it). If the url is not accessible, send
|
||||||
|
`CHECKURL-FAILURE`. If the url is accessible and the size is known,
|
||||||
|
send the size in `CHECKURL-SIZE`. If the url is accessible, but the size
|
||||||
|
is unknown, send `CHECKURL-SIZEUNOWN`.
|
||||||
|
|
||||||
More optional requests may be added, without changing the protocol version,
|
More optional requests may be added, without changing the protocol version,
|
||||||
so if an unknown request is seen, reply with `UNSUPPORTED-REQUEST`.
|
so if an unknown request is seen, reply with `UNSUPPORTED-REQUEST`.
|
||||||
|
@ -175,6 +181,14 @@ while it's handling a request.
|
||||||
Indicates that the CLAIMURL url will be handled by this remote.
|
Indicates that the CLAIMURL url will be handled by this remote.
|
||||||
* `CLAIMURL-FAILURE`
|
* `CLAIMURL-FAILURE`
|
||||||
Indicates that the CLAIMURL url wil not be handled by this remote.
|
Indicates that the CLAIMURL url wil not be handled by this remote.
|
||||||
|
* `CHECKURL-SIZE Size`
|
||||||
|
Indicates that the requested url has been verified to exist,
|
||||||
|
and its size is known. The size is in bytes.
|
||||||
|
* `CHECKURL-SIZEUNKNOWN`
|
||||||
|
Indicates that the requested url has been verified to exist,
|
||||||
|
but its size could not be determined.
|
||||||
|
* `CHECKURL-FAILURE`
|
||||||
|
Indicates that the requested url could not be accessed.
|
||||||
* `UNSUPPORTED-REQUEST`
|
* `UNSUPPORTED-REQUEST`
|
||||||
Indicates that the special remote does not know how to handle a request.
|
Indicates that the special remote does not know how to handle a request.
|
||||||
|
|
||||||
|
@ -255,14 +269,14 @@ in control.
|
||||||
* `GETSTATE Key`
|
* `GETSTATE Key`
|
||||||
Gets any state that has been stored for the key.
|
Gets any state that has been stored for the key.
|
||||||
(git-annex replies with VALUE followed by the state.)
|
(git-annex replies with VALUE followed by the state.)
|
||||||
* `SETURLPRESENT Key Value`
|
* `SETURLPRESENT Key Url`
|
||||||
Records an url (or uri) where the Key can be downloaded from.
|
Records an url (or uri) where the Key can be downloaded from.
|
||||||
* `SETURLMISSING Key Value`
|
* `SETURLMISSING Key Url`
|
||||||
Records that the key can no longer be downloaded from the specified
|
Records that the key can no longer be downloaded from the specified
|
||||||
url (or uri).
|
url (or uri).
|
||||||
* `GETURLS Key Value`
|
* `GETURLS Key Prefix`
|
||||||
Gets the recorded urls where a Key can be downloaded from.
|
Gets the recorded urls where a Key can be downloaded from.
|
||||||
Only urls that start with the Value will be returned. The Value
|
Only urls that start with the Prefix will be returned. The Prefix
|
||||||
may be empty to get all urls.
|
may be empty to get all urls.
|
||||||
(git-annex replies one or more times with VALUE for each url.
|
(git-annex replies one or more times with VALUE for each url.
|
||||||
The final VALUE has an empty value, indicating the end of the url list.)
|
The final VALUE has an empty value, indicating the end of the url list.)
|
||||||
|
|
14
doc/devblog/day_237__extending_addurl.mdwn
Normal file
14
doc/devblog/day_237__extending_addurl.mdwn
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
Worked on [[todo/extensible_addurl]] today. When `git annex addurl` is run,
|
||||||
|
remotes will be asked if they claim the url, and whichever remote does will
|
||||||
|
be used to download it, and location tracking will indicate that remote
|
||||||
|
contains the object. This is a masive 1000 line patch touching 30 files,
|
||||||
|
including follow-on changes in `rmurl` and `whereis` and even `rekey`.
|
||||||
|
|
||||||
|
It should now be possible to build an external special remote that handles
|
||||||
|
*.torrent and magnet: urls and passes them off to a bittorrent client for
|
||||||
|
download, for example.
|
||||||
|
|
||||||
|
Another use for this would be to make an external special remote that
|
||||||
|
uses youtube-dl or some other program than quvi for downloading web videos.
|
||||||
|
The builtin quvi support could probably be moved out of the web special
|
||||||
|
remote, to a separate remote. I haven't tried to do that yet.
|
|
@ -182,8 +182,9 @@ Example:
|
||||||
## `aaa/bbb/*.log.web`
|
## `aaa/bbb/*.log.web`
|
||||||
|
|
||||||
These log files record urls used by the
|
These log files record urls used by the
|
||||||
[[web_special_remote|special_remotes/web]]. Their format is similar
|
[[web_special_remote|special_remotes/web]] and sometimes by other remotes.
|
||||||
to the location tracking files, but with urls rather than UUIDs.
|
Their format is similar to the location tracking files, but with urls
|
||||||
|
rather than UUIDs.
|
||||||
|
|
||||||
## `aaa/bbb/*.log.rmt`
|
## `aaa/bbb/*.log.rmt`
|
||||||
|
|
||||||
|
|
|
@ -25,11 +25,40 @@ Solution: Add a new method to remotes:
|
||||||
claimUrl :: Maybe (URLString -> Annex Bool)
|
claimUrl :: Maybe (URLString -> Annex Bool)
|
||||||
|
|
||||||
Remotes that implement this method (including special remotes) will
|
Remotes that implement this method (including special remotes) will
|
||||||
be queried when such an uri is added, to see which claims it. Once the
|
be queried when such an uri is added, to see which claims it.
|
||||||
remote is known, addurl will record that the Key is present on that remote,
|
|
||||||
and record the uri in the url log.
|
|
||||||
|
|
||||||
Then retrieval of the Key works more or less as usual. The only
|
Once the remote is known, addurl --file will record that the Key is present
|
||||||
|
on that remote, and record the uri in the url log.
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
What about using addurl to add a new file? In this mode, the Key is not yet
|
||||||
|
known. addurl currently handles this by generating a dummy Key for the url
|
||||||
|
(hitting the url to get its size), and running a Transfer using the dummy
|
||||||
|
key that downloads from the web. Once the download is done, the dummy Key
|
||||||
|
is upgraded to the final Key.
|
||||||
|
|
||||||
|
Something similar could be done for other remotes, but the url log for the
|
||||||
|
dummy key would need to have the url added to it, for the remote to know
|
||||||
|
what to download, and then that could be removed after the download. Which
|
||||||
|
causes ugly churn in git, and would leave a mess if interrupted.
|
||||||
|
|
||||||
|
One option is to add another new method to remotes:
|
||||||
|
|
||||||
|
downloadUrl :: Maybe (URLString -> Annex FilePath)
|
||||||
|
|
||||||
|
Or, the url log could have support added for recording temporary key
|
||||||
|
urls in memory. (done)
|
||||||
|
|
||||||
|
Another problem is that the size of the Key isn't known. addurl
|
||||||
|
could always operate in relaxed mode, where it generates a size-less Key.
|
||||||
|
Or, yet another method could be added: (done)
|
||||||
|
|
||||||
|
sizeUrl :: URLString -> Annex (Maybe Integer)
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Retrieval of the Key works more or less as usual. The only
|
||||||
difference being that remotes that support this interface can look
|
difference being that remotes that support this interface can look
|
||||||
at the url log to find the one with the right "$downloader:" prefix,
|
at the url log to find the one with the right "$downloader:" prefix,
|
||||||
and so know where to download from. (Much as the web special remote already
|
and so know where to download from. (Much as the web special remote already
|
||||||
|
@ -55,3 +84,5 @@ This could be implemented in either the web special remote or even in an
|
||||||
external special remote.
|
external special remote.
|
||||||
|
|
||||||
Some other discussion at <https://github.com/datalad/datalad/issues/10>
|
Some other discussion at <https://github.com/datalad/datalad/issues/10>
|
||||||
|
|
||||||
|
> [[done]]! --[[Joey]]
|
||||||
|
|
Loading…
Reference in a new issue