169 lines
		
	
	
	
		
			5.5 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
			
		
		
	
	
			169 lines
		
	
	
	
		
			5.5 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
{- git-annex-shell main program
 | 
						|
 -
 | 
						|
 - Copyright 2010-2021 Joey Hess <id@joeyh.name>
 | 
						|
 -
 | 
						|
 - Licensed under the GNU AGPL version 3 or higher.
 | 
						|
 -}
 | 
						|
 | 
						|
module CmdLine.GitAnnexShell where
 | 
						|
 | 
						|
import Annex.Common
 | 
						|
import qualified Git.Construct
 | 
						|
import qualified Git.Config
 | 
						|
import CmdLine
 | 
						|
import CmdLine.AnnexSetter
 | 
						|
import Command
 | 
						|
import Annex.UUID
 | 
						|
import CmdLine.GitAnnexShell.Checks
 | 
						|
import CmdLine.GitAnnexShell.Fields
 | 
						|
import Remote.GCrypt (getGCryptUUID)
 | 
						|
import P2P.Protocol (ServerMode(..))
 | 
						|
 | 
						|
import qualified Command.ConfigList
 | 
						|
import qualified Command.NotifyChanges
 | 
						|
import qualified Command.GCryptSetup
 | 
						|
import qualified Command.P2PStdIO
 | 
						|
import qualified Command.InAnnex
 | 
						|
import qualified Command.RecvKey
 | 
						|
import qualified Command.SendKey
 | 
						|
import qualified Command.DropKey
 | 
						|
 | 
						|
import qualified Data.Map as M
 | 
						|
 | 
						|
cmdsMap :: M.Map ServerMode [Command]
 | 
						|
cmdsMap = M.fromList $ map mk
 | 
						|
	[ (ServeReadOnly, readonlycmds)
 | 
						|
	, (ServeAppendOnly, appendcmds)
 | 
						|
	, (ServeReadWrite, allcmds)
 | 
						|
	]
 | 
						|
  where
 | 
						|
	readonlycmds = map addAnnexOptions
 | 
						|
		[ Command.ConfigList.cmd
 | 
						|
		, gitAnnexShellCheck Command.NotifyChanges.cmd
 | 
						|
		-- p2pstdio checks the environment variables to
 | 
						|
		-- determine the security policy to use
 | 
						|
		, gitAnnexShellCheck Command.P2PStdIO.cmd
 | 
						|
		, gitAnnexShellCheck Command.InAnnex.cmd
 | 
						|
		, gitAnnexShellCheck Command.SendKey.cmd
 | 
						|
		]
 | 
						|
	appendcmds = readonlycmds ++ map addAnnexOptions
 | 
						|
		[ gitAnnexShellCheck Command.RecvKey.cmd
 | 
						|
		]
 | 
						|
	allcmds = appendcmds ++ map addAnnexOptions
 | 
						|
		[ gitAnnexShellCheck Command.DropKey.cmd
 | 
						|
		, Command.GCryptSetup.cmd
 | 
						|
		]
 | 
						|
 | 
						|
	mk (s, l) = (s, map (adddirparam . noMessages) l)
 | 
						|
	adddirparam c = c { cmdparamdesc = "DIRECTORY " ++ cmdparamdesc c }
 | 
						|
 | 
						|
cmdsFor :: ServerMode -> [Command]
 | 
						|
cmdsFor = fromMaybe [] . flip M.lookup cmdsMap
 | 
						|
 | 
						|
cmdsList :: [Command]
 | 
						|
cmdsList = nub $ concat $ M.elems cmdsMap
 | 
						|
 | 
						|
addAnnexOptions :: Command -> Command
 | 
						|
addAnnexOptions c = c { cmdannexoptions = commonShellOptions ++ cmdannexoptions c }
 | 
						|
 | 
						|
commonShellOptions :: [AnnexOption]
 | 
						|
commonShellOptions = 
 | 
						|
	annexOption (setAnnexState . checkUUID) (strOption
 | 
						|
		( long "uuid" <> metavar paramUUID
 | 
						|
		<> help "local repository uuid"
 | 
						|
		))
 | 
						|
	: commonOptions
 | 
						|
  where
 | 
						|
	checkUUID expected = getUUID >>= check
 | 
						|
	  where
 | 
						|
		check u | u == toUUID expected = noop
 | 
						|
		check NoUUID = checkGCryptUUID expected
 | 
						|
		check u = unexpectedUUID expected u
 | 
						|
	checkGCryptUUID expected = check =<< getGCryptUUID True =<< gitRepo
 | 
						|
	  where
 | 
						|
		check (Just u) | u == toUUID expected = noop
 | 
						|
		check Nothing = unexpected expected "uninitialized repository"
 | 
						|
		check (Just u) = unexpectedUUID expected u
 | 
						|
	unexpectedUUID expected u = unexpected expected $ "UUID " ++ fromUUID u
 | 
						|
	unexpected expected s = giveup $
 | 
						|
		"expected repository UUID " ++ expected ++ " but found " ++ s
 | 
						|
 | 
						|
run :: [String] -> IO ()
 | 
						|
run [] = failure
 | 
						|
-- skip leading -c options, passed by eg, ssh
 | 
						|
run ("-c":p) = run p
 | 
						|
-- a command can be either a builtin or something to pass to git-shell
 | 
						|
run c@(cmd:dir:params)
 | 
						|
	| cmd `elem` builtins = builtin cmd dir params
 | 
						|
	| otherwise = external c
 | 
						|
run c@(cmd:_)
 | 
						|
	-- Handle the case of being the user's login shell. It will be passed
 | 
						|
	-- a single string containing all the real parameters.
 | 
						|
	| "git-annex-shell " `isPrefixOf` cmd = run $ drop 1 $ shellUnEscape cmd
 | 
						|
	| cmd `elem` builtins = failure
 | 
						|
	| otherwise = external c
 | 
						|
 | 
						|
builtins :: [String]
 | 
						|
builtins = map cmdname cmdsList
 | 
						|
 | 
						|
builtin :: String -> String -> [String] -> IO ()
 | 
						|
builtin cmd dir params = do
 | 
						|
	unless (cmd `elem` map cmdname (cmdsFor ServeReadOnly))
 | 
						|
		checkNotReadOnly
 | 
						|
	unless (cmd `elem` map cmdname (cmdsFor ServeAppendOnly))
 | 
						|
		checkNotAppendOnly
 | 
						|
	checkDirectory $ Just dir
 | 
						|
	let (params', fieldparams, opts) = partitionParams params
 | 
						|
	    rsyncopts = ("RsyncOptions", unwords opts)
 | 
						|
	    fields = rsyncopts : filter checkField (parseFields fieldparams)
 | 
						|
	dispatch False False (cmd : params') cmdsList fields mkrepo
 | 
						|
		"git-annex-shell"
 | 
						|
		"Restricted login shell for git-annex only SSH access"
 | 
						|
  where
 | 
						|
	mkrepo = do
 | 
						|
		r <- Git.Construct.repoAbsPath (toRawFilePath dir)
 | 
						|
			>>= Git.Construct.fromAbsPath
 | 
						|
		Git.Config.read r
 | 
						|
			`catchIO` \_ -> do
 | 
						|
				hn <- fromMaybe "unknown" <$> getHostname
 | 
						|
				giveup $ "failed to read git config of git repository in " ++ hn ++ " on " ++ dir ++ "; perhaps this repository is not set up correctly or has moved"
 | 
						|
 | 
						|
external :: [String] -> IO ()
 | 
						|
external params = do
 | 
						|
	{- Normal git-shell commands all have the directory as their last
 | 
						|
	 - parameter. -}
 | 
						|
	let lastparam = lastMaybe =<< shellUnEscape <$> lastMaybe params
 | 
						|
	    (params', _, _) = partitionParams params
 | 
						|
	checkDirectory lastparam
 | 
						|
	checkNotLimited
 | 
						|
	unlessM (boolSystem "git-shell" $ map Param $ "-c":params') $
 | 
						|
		giveup "git-shell failed"
 | 
						|
 | 
						|
{- Split the input list into 3 groups separated with a double dash --.
 | 
						|
 - Parameters between two -- markers are field settings, in the form:
 | 
						|
 - field=value field=value
 | 
						|
 -
 | 
						|
 - Parameters after the last -- are the command itself and its arguments e.g.,
 | 
						|
 - rsync --bandwidth=100.
 | 
						|
 -}
 | 
						|
partitionParams :: [String] -> ([String], [String], [String])
 | 
						|
partitionParams ps = case segment (== "--") ps of
 | 
						|
	params:fieldparams:rest -> ( params, fieldparams, intercalate ["--"] rest )
 | 
						|
	[params] -> (params, [], [])
 | 
						|
	_ -> ([], [], [])
 | 
						|
 | 
						|
parseFields :: [String] -> [(String, String)]
 | 
						|
parseFields = map (separate (== '='))
 | 
						|
 | 
						|
{- Only allow known fields to be set, ignore others.
 | 
						|
 - Make sure that field values make sense. -}
 | 
						|
checkField :: (String, String) -> Bool
 | 
						|
checkField (field, val)
 | 
						|
	| field == fieldName remoteUUID = fieldCheck remoteUUID val
 | 
						|
	| field == fieldName autoInit = fieldCheck autoInit val
 | 
						|
	| otherwise = False
 | 
						|
 | 
						|
failure :: IO ()
 | 
						|
failure = giveup $ "bad parameters\n\n" ++ usage h cmdsList
 | 
						|
  where
 | 
						|
	h = "git-annex-shell [-c] command [parameters ...] [option ...]"
 |