73 lines
		
	
	
	
		
			2.3 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
			
		
		
	
	
			73 lines
		
	
	
	
		
			2.3 KiB
			
		
	
	
	
		
			Haskell
		
	
	
	
	
	
{- git-annex environment
 | 
						|
 -
 | 
						|
 - Copyright 2012-2023 Joey Hess <id@joeyh.name>
 | 
						|
 -
 | 
						|
 - Licensed under the GNU AGPL version 3 or higher.
 | 
						|
 -}
 | 
						|
 | 
						|
{-# LANGUAGE OverloadedStrings #-}
 | 
						|
 | 
						|
module Annex.Environment (
 | 
						|
	checkEnvironment,
 | 
						|
	checkEnvironmentIO,
 | 
						|
	ensureCommit,
 | 
						|
) where
 | 
						|
 | 
						|
import Annex.Common
 | 
						|
import Utility.UserInfo
 | 
						|
import qualified Git.Config
 | 
						|
import Config
 | 
						|
import Utility.Env.Set
 | 
						|
 | 
						|
import Control.Exception
 | 
						|
 | 
						|
{- Checks that the system's environment allows git to function.
 | 
						|
 - Git requires a GECOS username, or suitable git configuration, or
 | 
						|
 - environment variables. When none of those are set, this will set the
 | 
						|
 - environment variables.
 | 
						|
 -
 | 
						|
 - Git also requires the system have a hostname containing a dot.
 | 
						|
 - Otherwise, it tries various methods to find a FQDN, and will fail if it
 | 
						|
 - does not. To avoid replicating that code here, which would break if its
 | 
						|
 - methods change, this function does not check the hostname is valid.
 | 
						|
 - Instead, git-annex init calls ensureCommit, which makes sure that git
 | 
						|
 - gets set up to allow committing.
 | 
						|
 -}
 | 
						|
checkEnvironment :: Annex ()
 | 
						|
checkEnvironment = do
 | 
						|
	gitusername <- fromRepo $ Git.Config.getMaybe "user.name"
 | 
						|
	when (isNothing gitusername || gitusername == Just "") $
 | 
						|
		unlessM userConfigOnly $
 | 
						|
			liftIO checkEnvironmentIO
 | 
						|
 | 
						|
checkEnvironmentIO :: IO ()
 | 
						|
checkEnvironmentIO = whenM (isNothing <$> myUserGecos) $ do
 | 
						|
	username <- either (const "unknown") id <$> myUserName
 | 
						|
	ensureEnv "GIT_AUTHOR_NAME" username
 | 
						|
	ensureEnv "GIT_COMMITTER_NAME" username
 | 
						|
  where
 | 
						|
	-- existing environment is not overwritten
 | 
						|
	ensureEnv var val = setEnv var val False
 | 
						|
 | 
						|
{- Runs an action that commits to the repository, and if it fails, 
 | 
						|
 - sets user.email and user.name to a dummy value and tries the action again.
 | 
						|
 -
 | 
						|
 - Note that user.email and user.name are left set afterwards, so this only
 | 
						|
 - needs to be used once to make sure that future commits will succeed.
 | 
						|
 -}
 | 
						|
ensureCommit :: Annex a -> Annex a
 | 
						|
ensureCommit a = either retry return =<< tryNonAsync a 
 | 
						|
  where
 | 
						|
	retry e = ifM userConfigOnly
 | 
						|
		( liftIO (throwIO e)
 | 
						|
		, do
 | 
						|
			name <- liftIO $ either (const "unknown") id <$> myUserName
 | 
						|
			setConfig "user.name" name
 | 
						|
			setConfig "user.email" name
 | 
						|
			a
 | 
						|
		)
 | 
						|
 | 
						|
userConfigOnly :: Annex Bool
 | 
						|
userConfigOnly = do
 | 
						|
	v <- fromRepo $ Git.Config.getMaybe "user.useconfigonly"
 | 
						|
	return (fromMaybe False (Git.Config.isTrueFalse' =<< v))
 |