2010-10-27 20:53:54 +00:00
|
|
|
{- git-annex remote repositories
|
|
|
|
-
|
|
|
|
- Copyright 2010 Joey Hess <joey@kitenet.net>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
2010-10-13 19:55:18 +00:00
|
|
|
|
|
|
|
module Remotes (
|
2010-10-14 06:41:54 +00:00
|
|
|
list,
|
2011-02-03 22:55:12 +00:00
|
|
|
tryGitConfigRead,
|
2011-01-04 21:45:27 +00:00
|
|
|
readConfigs,
|
2010-10-23 17:18:47 +00:00
|
|
|
keyPossibilities,
|
2010-10-23 18:14:36 +00:00
|
|
|
inAnnex,
|
2010-11-01 03:38:07 +00:00
|
|
|
same,
|
2010-12-28 21:17:02 +00:00
|
|
|
byName,
|
2010-10-23 18:58:14 +00:00
|
|
|
copyFromRemote,
|
2010-10-25 21:17:03 +00:00
|
|
|
copyToRemote,
|
2011-03-05 19:31:46 +00:00
|
|
|
onRemote
|
2010-10-13 19:55:18 +00:00
|
|
|
) where
|
|
|
|
|
2010-11-22 21:51:55 +00:00
|
|
|
import Control.Exception.Extensible
|
2010-10-14 01:28:47 +00:00
|
|
|
import Control.Monad.State (liftIO)
|
2010-10-14 02:59:43 +00:00
|
|
|
import qualified Data.Map as Map
|
2010-10-14 03:18:58 +00:00
|
|
|
import Data.String.Utils
|
2010-12-31 19:46:33 +00:00
|
|
|
import System.Cmd.Utils
|
2010-12-29 20:21:38 +00:00
|
|
|
import Data.List (intersect, sortBy)
|
2010-11-22 21:51:55 +00:00
|
|
|
import Control.Monad (when, unless, filterM)
|
2010-10-16 20:20:49 +00:00
|
|
|
|
2010-10-14 07:18:11 +00:00
|
|
|
import Types
|
2010-10-14 06:36:41 +00:00
|
|
|
import qualified GitRepo as Git
|
2010-10-14 07:18:11 +00:00
|
|
|
import qualified Annex
|
2010-10-13 19:55:18 +00:00
|
|
|
import LocationLog
|
2010-10-14 03:18:58 +00:00
|
|
|
import Locations
|
2010-10-13 19:55:18 +00:00
|
|
|
import UUID
|
2011-01-26 19:37:16 +00:00
|
|
|
import Trust
|
2010-10-23 18:14:36 +00:00
|
|
|
import Utility
|
2011-01-16 20:05:05 +00:00
|
|
|
import qualified Content
|
2010-11-08 19:15:21 +00:00
|
|
|
import Messages
|
2010-11-18 17:48:28 +00:00
|
|
|
import CopyFile
|
2010-12-31 23:09:17 +00:00
|
|
|
import RsyncFile
|
2011-03-05 19:47:00 +00:00
|
|
|
import Ssh
|
2010-10-13 19:55:18 +00:00
|
|
|
|
|
|
|
{- Human visible list of remotes. -}
|
2010-10-14 06:41:54 +00:00
|
|
|
list :: [Git.Repo] -> String
|
2010-10-22 19:21:23 +00:00
|
|
|
list remotes = join ", " $ map Git.repoDescribe remotes
|
2010-10-13 19:55:18 +00:00
|
|
|
|
2011-01-04 21:20:35 +00:00
|
|
|
{- The git configs for the git repo's remotes is not read on startup
|
|
|
|
- because reading it may be expensive. This function tries to read the
|
|
|
|
- config for a specified remote, and updates state. If successful, it
|
|
|
|
- returns the updated git repo. -}
|
|
|
|
tryGitConfigRead :: Git.Repo -> Annex (Either Git.Repo Git.Repo)
|
|
|
|
tryGitConfigRead r
|
|
|
|
| not $ Map.null $ Git.configMap r = return $ Right r -- already read
|
|
|
|
| Git.repoIsSsh r = store $ onRemote r (pipedconfig, r) "configlist" []
|
|
|
|
| Git.repoIsUrl r = return $ Left r
|
|
|
|
| otherwise = store $ safely $ Git.configRead r
|
|
|
|
where
|
|
|
|
-- Reading config can fail due to IO error or
|
|
|
|
-- for other reasons; catch all possible exceptions.
|
|
|
|
safely a = do
|
|
|
|
result <- liftIO (try (a)::IO (Either SomeException Git.Repo))
|
|
|
|
case result of
|
|
|
|
Left _ -> return r
|
|
|
|
Right r' -> return r'
|
|
|
|
pipedconfig cmd params = safely $
|
2011-02-28 20:25:31 +00:00
|
|
|
pOpen ReadFromPipe cmd (toCommand params) $
|
2011-01-04 21:20:35 +00:00
|
|
|
Git.hConfigRead r
|
|
|
|
store a = do
|
|
|
|
r' <- a
|
|
|
|
g <- Annex.gitRepo
|
|
|
|
let l = Git.remotes g
|
|
|
|
let g' = Git.remotesAdd g $ exchange l r'
|
2011-01-26 04:17:38 +00:00
|
|
|
Annex.changeState $ \s -> s { Annex.repo = g' }
|
2011-01-04 21:20:35 +00:00
|
|
|
return $ Right r'
|
|
|
|
exchange [] _ = []
|
|
|
|
exchange (old:ls) new =
|
|
|
|
if Git.repoRemoteName old == Git.repoRemoteName new
|
|
|
|
then new : exchange ls new
|
|
|
|
else old : exchange ls new
|
|
|
|
|
|
|
|
{- Reads the configs of all remotes.
|
2011-01-04 21:15:39 +00:00
|
|
|
-
|
|
|
|
- This has to be called before things that rely on eg, the UUID of
|
|
|
|
- remotes. Most such things will take care of running this themselves.
|
|
|
|
-
|
|
|
|
- As reading the config of remotes can be expensive, this
|
|
|
|
- function will only read configs once per git-annex run. It's
|
|
|
|
- assumed to be cheap to read the config of non-URL remotes,
|
|
|
|
- so this is done each time git-annex is run. Conversely,
|
|
|
|
- the config of an URL remote is only read when there is no
|
|
|
|
- cached UUID value.
|
|
|
|
- -}
|
2011-01-04 21:20:35 +00:00
|
|
|
readConfigs :: Annex ()
|
|
|
|
readConfigs = do
|
2011-01-26 04:17:38 +00:00
|
|
|
remotesread <- Annex.getState Annex.remotesread
|
2011-01-04 21:15:39 +00:00
|
|
|
unless remotesread $ do
|
2011-02-03 22:55:12 +00:00
|
|
|
g <- Annex.gitRepo
|
2011-01-05 02:14:24 +00:00
|
|
|
allremotes <- filterM repoNotIgnored $ Git.remotes g
|
2011-01-04 21:15:39 +00:00
|
|
|
let cheap = filter (not . Git.repoIsUrl) allremotes
|
|
|
|
let expensive = filter Git.repoIsUrl allremotes
|
|
|
|
doexpensive <- filterM cachedUUID expensive
|
|
|
|
unless (null doexpensive) $
|
|
|
|
showNote $ "getting UUID for " ++
|
|
|
|
list doexpensive ++ "..."
|
|
|
|
let todo = cheap ++ doexpensive
|
|
|
|
unless (null todo) $ do
|
2011-01-31 17:52:11 +00:00
|
|
|
mapM_ tryGitConfigRead todo
|
2011-01-26 04:17:38 +00:00
|
|
|
Annex.changeState $ \s -> s { Annex.remotesread = True }
|
2011-01-04 21:15:39 +00:00
|
|
|
where
|
|
|
|
cachedUUID r = do
|
|
|
|
u <- getUUID r
|
|
|
|
return $ null u
|
|
|
|
|
2010-12-29 20:21:38 +00:00
|
|
|
{- Cost ordered lists of remotes that the LocationLog indicate may have a key.
|
2010-12-29 20:58:44 +00:00
|
|
|
-
|
2010-12-29 23:09:02 +00:00
|
|
|
- Also returns a list of UUIDs that are trusted to have the key
|
2010-12-29 20:58:44 +00:00
|
|
|
- (some may not have configured remotes).
|
|
|
|
-}
|
2011-01-26 20:44:14 +00:00
|
|
|
keyPossibilities :: Key -> Annex ([Git.Repo], [UUID])
|
2010-10-23 17:18:47 +00:00
|
|
|
keyPossibilities key = do
|
2011-01-04 21:20:35 +00:00
|
|
|
readConfigs
|
2011-01-04 21:15:39 +00:00
|
|
|
|
2010-10-14 03:18:58 +00:00
|
|
|
allremotes <- remotesByCost
|
2011-01-04 21:15:39 +00:00
|
|
|
g <- Annex.gitRepo
|
|
|
|
u <- getUUID g
|
2011-01-26 19:37:16 +00:00
|
|
|
trusted <- trustGet Trusted
|
2011-01-04 21:15:39 +00:00
|
|
|
|
2011-01-26 20:44:14 +00:00
|
|
|
-- get uuids of all repositories that are recorded to have the key
|
2011-01-04 21:15:39 +00:00
|
|
|
uuids <- liftIO $ keyLocations g key
|
|
|
|
let validuuids = filter (/= u) uuids
|
|
|
|
|
|
|
|
-- note that validuuids is assumed to not have dups
|
|
|
|
let validtrusteduuids = intersect validuuids trusted
|
|
|
|
|
|
|
|
-- remotes that match uuids that have the key
|
|
|
|
validremotes <- reposByUUID allremotes validuuids
|
|
|
|
|
2011-01-26 20:44:14 +00:00
|
|
|
return (validremotes, validtrusteduuids)
|
2010-10-13 19:55:18 +00:00
|
|
|
|
2010-10-23 17:18:47 +00:00
|
|
|
{- Checks if a given remote has the content for a key inAnnex.
|
|
|
|
- If the remote cannot be accessed, returns a Left error.
|
|
|
|
-}
|
|
|
|
inAnnex :: Git.Repo -> Key -> Annex (Either IOException Bool)
|
2010-11-22 21:51:55 +00:00
|
|
|
inAnnex r key = if Git.repoIsUrl r
|
|
|
|
then checkremote
|
|
|
|
else liftIO (try checklocal ::IO (Either IOException Bool))
|
2010-10-23 17:18:47 +00:00
|
|
|
where
|
2010-11-01 03:50:58 +00:00
|
|
|
checklocal = do
|
2010-12-31 19:46:33 +00:00
|
|
|
-- run a local check inexpensively,
|
|
|
|
-- by making an Annex monad using the remote
|
2010-11-01 03:21:16 +00:00
|
|
|
a <- Annex.new r []
|
2011-01-16 20:05:05 +00:00
|
|
|
Annex.eval a (Content.inAnnex key)
|
2010-11-01 03:50:58 +00:00
|
|
|
checkremote = do
|
2010-11-08 19:15:21 +00:00
|
|
|
showNote ("checking " ++ Git.repoDescribe r ++ "...")
|
2010-12-31 19:52:59 +00:00
|
|
|
inannex <- onRemote r (boolSystem, False) "inannex"
|
2011-03-16 02:53:14 +00:00
|
|
|
[Param (show key)]
|
2010-11-01 03:50:58 +00:00
|
|
|
return $ Right inannex
|
2010-10-23 17:18:47 +00:00
|
|
|
|
2010-10-13 19:55:18 +00:00
|
|
|
{- Cost Ordered list of remotes. -}
|
2010-10-14 06:36:41 +00:00
|
|
|
remotesByCost :: Annex [Git.Repo]
|
2010-10-14 01:28:47 +00:00
|
|
|
remotesByCost = do
|
2010-10-14 07:18:11 +00:00
|
|
|
g <- Annex.gitRepo
|
2010-10-14 06:36:41 +00:00
|
|
|
reposByCost $ Git.remotes g
|
2010-10-13 19:55:18 +00:00
|
|
|
|
2010-10-23 00:35:39 +00:00
|
|
|
{- Orders a list of git repos by cost. Throws out ignored ones. -}
|
2010-10-14 06:36:41 +00:00
|
|
|
reposByCost :: [Git.Repo] -> Annex [Git.Repo]
|
2010-10-14 01:28:47 +00:00
|
|
|
reposByCost l = do
|
2010-10-22 18:28:47 +00:00
|
|
|
notignored <- filterM repoNotIgnored l
|
|
|
|
costpairs <- mapM costpair notignored
|
2010-11-22 21:51:55 +00:00
|
|
|
return $ fst $ unzip $ sortBy cmpcost costpairs
|
2010-10-13 19:55:18 +00:00
|
|
|
where
|
2010-10-14 01:28:47 +00:00
|
|
|
costpair r = do
|
|
|
|
cost <- repoCost r
|
|
|
|
return (r, cost)
|
2010-11-22 21:51:55 +00:00
|
|
|
cmpcost (_, c1) (_, c2) = compare c1 c2
|
2010-10-13 19:55:18 +00:00
|
|
|
|
|
|
|
{- Calculates cost for a repo.
|
|
|
|
-
|
|
|
|
- The default cost is 100 for local repositories, and 200 for remote
|
|
|
|
- repositories; it can also be configured by remote.<name>.annex-cost
|
|
|
|
-}
|
2010-10-14 06:36:41 +00:00
|
|
|
repoCost :: Git.Repo -> Annex Int
|
2010-10-14 01:28:47 +00:00
|
|
|
repoCost r = do
|
2011-03-05 19:31:46 +00:00
|
|
|
cost <- Annex.repoConfig r "cost" ""
|
2010-11-22 21:51:55 +00:00
|
|
|
if not $ null cost
|
2010-11-01 02:56:56 +00:00
|
|
|
then return $ read cost
|
2010-11-22 21:51:55 +00:00
|
|
|
else if Git.repoIsUrl r
|
2010-10-22 18:28:47 +00:00
|
|
|
then return 200
|
|
|
|
else return 100
|
2010-10-14 02:59:43 +00:00
|
|
|
|
2010-10-23 00:35:39 +00:00
|
|
|
{- Checks if a repo should be ignored, based either on annex-ignore
|
|
|
|
- setting, or on command-line options. Allows command-line to override
|
|
|
|
- annex-ignore. -}
|
2010-10-22 18:28:47 +00:00
|
|
|
repoNotIgnored :: Git.Repo -> Annex Bool
|
|
|
|
repoNotIgnored r = do
|
2011-03-05 19:31:46 +00:00
|
|
|
ignored <- Annex.repoConfig r "ignore" "false"
|
2011-01-26 04:17:38 +00:00
|
|
|
to <- match Annex.toremote
|
|
|
|
from <- match Annex.fromremote
|
|
|
|
if to || from
|
|
|
|
then return True
|
2010-11-22 21:51:55 +00:00
|
|
|
else return $ not $ Git.configTrue ignored
|
2010-10-22 18:28:47 +00:00
|
|
|
where
|
2011-01-26 04:17:38 +00:00
|
|
|
match a = do
|
|
|
|
name <- Annex.getState a
|
|
|
|
case name of
|
|
|
|
Nothing -> return False
|
2011-02-04 02:20:55 +00:00
|
|
|
n -> return $ n == Git.repoRemoteName r
|
2010-10-22 18:28:47 +00:00
|
|
|
|
2010-11-01 03:38:07 +00:00
|
|
|
{- Checks if two repos are the same, by comparing their remote names. -}
|
|
|
|
same :: Git.Repo -> Git.Repo -> Bool
|
|
|
|
same a b = Git.repoRemoteName a == Git.repoRemoteName b
|
|
|
|
|
2011-03-03 21:21:00 +00:00
|
|
|
{- Looks up a remote by name. (Or by UUID.) -}
|
2010-12-28 21:17:02 +00:00
|
|
|
byName :: String -> Annex Git.Repo
|
2011-01-26 20:20:28 +00:00
|
|
|
byName "." = Annex.gitRepo -- special case to refer to current repository
|
2010-12-28 21:17:02 +00:00
|
|
|
byName name = do
|
2010-10-28 18:20:02 +00:00
|
|
|
when (null name) $ error "no remote specified"
|
|
|
|
g <- Annex.gitRepo
|
2011-03-03 21:21:00 +00:00
|
|
|
match <- filterM matching $ Git.remotes g
|
2010-10-28 18:20:02 +00:00
|
|
|
when (null match) $ error $
|
|
|
|
"there is no git remote named \"" ++ name ++ "\""
|
2010-11-22 21:51:55 +00:00
|
|
|
return $ head match
|
2011-03-03 21:21:00 +00:00
|
|
|
where
|
|
|
|
matching r = do
|
|
|
|
if Just name == Git.repoRemoteName r
|
|
|
|
then return True
|
|
|
|
else do
|
|
|
|
u <- getUUID r
|
|
|
|
return $ (name == u)
|
2010-10-23 18:14:36 +00:00
|
|
|
|
2010-12-31 23:09:17 +00:00
|
|
|
{- Tries to copy a key's content from a remote's annex to a file. -}
|
2010-10-23 18:14:36 +00:00
|
|
|
copyFromRemote :: Git.Repo -> Key -> FilePath -> Annex Bool
|
2010-11-22 21:51:55 +00:00
|
|
|
copyFromRemote r key file
|
2011-01-27 21:00:32 +00:00
|
|
|
| not $ Git.repoIsUrl r = liftIO $ copyFile (gitAnnexLocation r key) file
|
2010-12-31 23:09:17 +00:00
|
|
|
| Git.repoIsSsh r = rsynchelper r True key file
|
2010-11-22 21:51:55 +00:00
|
|
|
| otherwise = error "copying from non-ssh repo not supported"
|
2010-10-23 18:58:14 +00:00
|
|
|
|
2010-12-31 23:09:17 +00:00
|
|
|
{- Tries to copy a key's content to a remote's annex. -}
|
|
|
|
copyToRemote :: Git.Repo -> Key -> Annex Bool
|
|
|
|
copyToRemote r key
|
|
|
|
| not $ Git.repoIsUrl r = do
|
|
|
|
g <- Annex.gitRepo
|
2011-01-27 21:00:32 +00:00
|
|
|
let keysrc = gitAnnexLocation g key
|
2011-01-07 05:14:27 +00:00
|
|
|
-- run copy from perspective of remote
|
|
|
|
liftIO $ do
|
|
|
|
a <- Annex.new r []
|
2011-01-11 20:06:19 +00:00
|
|
|
Annex.eval a $ do
|
2011-01-16 20:05:05 +00:00
|
|
|
ok <- Content.getViaTmp key $
|
2011-01-11 20:06:19 +00:00
|
|
|
\f -> liftIO $ copyFile keysrc f
|
|
|
|
Annex.queueRun
|
|
|
|
return ok
|
2010-12-31 23:09:17 +00:00
|
|
|
| Git.repoIsSsh r = do
|
|
|
|
g <- Annex.gitRepo
|
2011-01-27 21:00:32 +00:00
|
|
|
let keysrc = gitAnnexLocation g key
|
2010-12-31 23:09:17 +00:00
|
|
|
rsynchelper r False key keysrc
|
|
|
|
| otherwise = error "copying to non-ssh repo not supported"
|
2010-10-25 21:17:03 +00:00
|
|
|
|
2010-12-31 23:09:17 +00:00
|
|
|
rsynchelper :: Git.Repo -> Bool -> Key -> FilePath -> Annex (Bool)
|
|
|
|
rsynchelper r sending key file = do
|
2010-12-02 20:55:21 +00:00
|
|
|
showProgress -- make way for progress bar
|
2010-12-31 23:09:17 +00:00
|
|
|
p <- rsyncParams r sending key file
|
|
|
|
res <- liftIO $ boolSystem "rsync" p
|
2010-12-02 21:45:28 +00:00
|
|
|
if res
|
|
|
|
then return res
|
|
|
|
else do
|
2010-12-31 00:31:52 +00:00
|
|
|
showLongNote "rsync failed -- run git annex again to resume file transfer"
|
2010-12-02 21:45:28 +00:00
|
|
|
return res
|
2010-12-31 23:09:17 +00:00
|
|
|
|
|
|
|
{- Generates rsync parameters that ssh to the remote and asks it
|
|
|
|
- to either receive or send the key's content. -}
|
2011-02-28 20:25:31 +00:00
|
|
|
rsyncParams :: Git.Repo -> Bool -> Key -> FilePath -> Annex [CommandParam]
|
2010-12-31 23:09:17 +00:00
|
|
|
rsyncParams r sending key file = do
|
2011-02-28 20:10:16 +00:00
|
|
|
Just (shellcmd, shellparams) <- git_annex_shell r
|
2010-12-31 23:09:17 +00:00
|
|
|
(if sending then "sendkey" else "recvkey")
|
2011-03-16 02:53:14 +00:00
|
|
|
[ Param $ show key
|
2011-02-28 20:10:16 +00:00
|
|
|
-- Command is terminated with "--", because
|
|
|
|
-- rsync will tack on its own options afterwards,
|
|
|
|
-- and they need to be ignored.
|
|
|
|
, Param "--"
|
|
|
|
]
|
2010-12-31 23:09:17 +00:00
|
|
|
-- Convert the ssh command into rsync command line.
|
2011-02-28 20:10:16 +00:00
|
|
|
let eparam = rsyncShell (Param shellcmd:shellparams)
|
2011-03-05 19:31:46 +00:00
|
|
|
o <- Annex.repoConfig r "rsync-options" ""
|
2011-02-28 20:10:16 +00:00
|
|
|
let base = options ++ map Param (words o) ++ eparam
|
2010-12-31 23:09:17 +00:00
|
|
|
if sending
|
2011-02-28 20:10:16 +00:00
|
|
|
then return $ base ++ [dummy, File file]
|
|
|
|
else return $ base ++ [File file, dummy]
|
2010-12-02 21:45:28 +00:00
|
|
|
where
|
2010-12-31 00:31:52 +00:00
|
|
|
-- inplace makes rsync resume partial files
|
2011-02-28 20:10:16 +00:00
|
|
|
options = [Params "-p --progress --inplace"]
|
2010-12-31 23:09:17 +00:00
|
|
|
-- the rsync shell parameter controls where rsync
|
2011-02-25 05:13:01 +00:00
|
|
|
-- goes, so the source/dest parameter can be a dummy value,
|
2010-12-31 23:09:17 +00:00
|
|
|
-- that just enables remote rsync mode.
|
2011-02-28 20:10:16 +00:00
|
|
|
dummy = Param ":"
|
2010-12-31 00:31:52 +00:00
|
|
|
|
2010-12-31 23:09:17 +00:00
|
|
|
{- Uses a supplied function to run a git-annex-shell command on a remote.
|
|
|
|
-
|
|
|
|
- Or, if the remote does not support running remote commands, returns
|
|
|
|
- a specified error value. -}
|
2010-12-31 19:46:33 +00:00
|
|
|
onRemote
|
|
|
|
:: Git.Repo
|
2011-02-28 20:25:31 +00:00
|
|
|
-> (FilePath -> [CommandParam] -> IO a, a)
|
2010-12-31 19:46:33 +00:00
|
|
|
-> String
|
2011-02-28 20:25:31 +00:00
|
|
|
-> [CommandParam]
|
2010-12-31 19:46:33 +00:00
|
|
|
-> Annex a
|
2010-12-31 23:09:17 +00:00
|
|
|
onRemote r (with, errorval) command params = do
|
|
|
|
s <- git_annex_shell r command params
|
|
|
|
case s of
|
2011-02-28 20:10:16 +00:00
|
|
|
Just (c, ps) -> liftIO $ with c ps
|
2010-12-31 23:09:17 +00:00
|
|
|
Nothing -> return errorval
|
|
|
|
|
|
|
|
{- Generates parameters to run a git-annex-shell command on a remote. -}
|
2011-02-28 20:25:31 +00:00
|
|
|
git_annex_shell :: Git.Repo -> String -> [CommandParam] -> Annex (Maybe (FilePath, [CommandParam]))
|
2010-12-31 23:09:17 +00:00
|
|
|
git_annex_shell r command params
|
2011-02-28 20:10:16 +00:00
|
|
|
| not $ Git.repoIsUrl r = return $ Just (shellcmd, shellopts)
|
2010-12-31 19:46:33 +00:00
|
|
|
| Git.repoIsSsh r = do
|
2011-03-05 19:47:00 +00:00
|
|
|
sshparams <- sshToRepo r [Param sshcmd]
|
|
|
|
return $ Just ("ssh", sshparams)
|
2010-12-31 23:09:17 +00:00
|
|
|
| otherwise = return Nothing
|
2010-12-31 00:31:52 +00:00
|
|
|
where
|
|
|
|
dir = Git.workTree r
|
2010-12-31 19:46:33 +00:00
|
|
|
shellcmd = "git-annex-shell"
|
2011-02-28 20:10:16 +00:00
|
|
|
shellopts = (Param command):(File dir):params
|
|
|
|
sshcmd = shellcmd ++ " " ++
|
2011-02-28 20:25:31 +00:00
|
|
|
unwords (map shellEscape $ toCommand shellopts)
|