2014-03-31 03:37:54 +00:00
|
|
|
module Propellor.CmdLine where
|
2014-03-30 23:10:32 +00:00
|
|
|
|
|
|
|
import System.Environment
|
|
|
|
import Data.List
|
|
|
|
import System.Exit
|
|
|
|
|
2014-03-31 03:55:59 +00:00
|
|
|
import Propellor
|
2014-03-30 23:10:32 +00:00
|
|
|
import Utility.FileMode
|
2014-03-31 03:55:59 +00:00
|
|
|
import Utility.SafeCommand
|
2014-03-30 23:10:32 +00:00
|
|
|
|
|
|
|
data CmdLine
|
|
|
|
= Run HostName
|
|
|
|
| Spin HostName
|
|
|
|
| Boot HostName
|
2014-03-31 01:01:18 +00:00
|
|
|
| Set HostName PrivDataField
|
2014-03-31 16:06:04 +00:00
|
|
|
| AddKey String
|
2014-03-31 20:37:19 +00:00
|
|
|
| Continue CmdLine
|
|
|
|
deriving (Read, Show, Eq)
|
|
|
|
|
|
|
|
usage :: IO a
|
|
|
|
usage = do
|
|
|
|
putStrLn $ unlines
|
|
|
|
[ "Usage:"
|
|
|
|
, " propellor"
|
|
|
|
, " propellor hostname"
|
|
|
|
, " propellor --spin hostname"
|
|
|
|
, " propellor --set hostname field"
|
|
|
|
, " propellor --add-key keyid"
|
|
|
|
]
|
|
|
|
exitFailure
|
2014-03-30 23:10:32 +00:00
|
|
|
|
|
|
|
processCmdLine :: IO CmdLine
|
|
|
|
processCmdLine = go =<< getArgs
|
|
|
|
where
|
|
|
|
go ("--help":_) = usage
|
|
|
|
go ("--spin":h:[]) = return $ Spin h
|
|
|
|
go ("--boot":h:[]) = return $ Boot h
|
2014-03-31 16:06:04 +00:00
|
|
|
go ("--add-key":k:[]) = return $ AddKey k
|
2014-03-31 01:01:18 +00:00
|
|
|
go ("--set":h:f:[]) = case readish f of
|
|
|
|
Just pf -> return $ Set h pf
|
2014-03-30 23:10:32 +00:00
|
|
|
Nothing -> error $ "Unknown privdata field " ++ f
|
2014-03-31 20:37:19 +00:00
|
|
|
go ("--continue":s:[]) =case readish s of
|
|
|
|
Just cmdline -> return $ Continue cmdline
|
|
|
|
Nothing -> error "--continue serialization failure"
|
2014-03-30 23:10:32 +00:00
|
|
|
go (h:[]) = return $ Run h
|
|
|
|
go [] = do
|
|
|
|
s <- takeWhile (/= '\n') <$> readProcess "hostname" ["-f"]
|
|
|
|
if null s
|
|
|
|
then error "Cannot determine hostname! Pass it on the command line."
|
|
|
|
else return $ Run s
|
|
|
|
go _ = usage
|
|
|
|
|
2014-03-31 03:02:10 +00:00
|
|
|
defaultMain :: (HostName -> Maybe [Property]) -> IO ()
|
2014-03-31 20:37:19 +00:00
|
|
|
defaultMain getprops = go True =<< processCmdLine
|
2014-03-30 23:10:32 +00:00
|
|
|
where
|
2014-03-31 20:37:19 +00:00
|
|
|
go _ (Continue cmdline) = go False cmdline
|
|
|
|
go _ (Set host field) = setPrivData host field
|
|
|
|
go _ (AddKey keyid) = addKey keyid
|
|
|
|
go _ (Spin host) = withprops host $ const $ spin host
|
|
|
|
go True cmdline = pullFirst cmdline $ go False cmdline
|
|
|
|
go _ (Run host) = withprops host $ ensureProperties
|
|
|
|
go _ (Boot host) = withprops host $ boot
|
|
|
|
|
2014-03-31 18:59:06 +00:00
|
|
|
withprops host a = maybe (unknownhost host) a (getprops host)
|
2014-03-30 23:10:32 +00:00
|
|
|
|
2014-03-31 03:02:10 +00:00
|
|
|
unknownhost :: HostName -> IO a
|
|
|
|
unknownhost h = error $ unwords
|
|
|
|
[ "Unknown host:", h
|
|
|
|
, "(perhaps you should specify the real hostname on the command line?)"
|
|
|
|
]
|
|
|
|
|
2014-03-31 20:37:19 +00:00
|
|
|
pullFirst :: CmdLine -> IO () -> IO ()
|
|
|
|
pullFirst cmdline next = do
|
2014-03-31 20:20:38 +00:00
|
|
|
branchref <- takeWhile (/= '\n')
|
|
|
|
<$> readProcess "git" ["symbolic-ref", "HEAD"]
|
|
|
|
let originbranch = "origin" </> takeFileName branchref
|
|
|
|
void $ boolSystem "git" [Param "fetch"]
|
|
|
|
|
|
|
|
whenM (doesFileExist keyring) $ do
|
|
|
|
{- To verify origin/master commit's signature, have to
|
|
|
|
- convince gpg to use our keyring. While running git log.
|
|
|
|
- Which has no way to pass options to gpg.
|
|
|
|
- Argh! -}
|
|
|
|
let gpgconf = privDataDir </> "gpg.conf"
|
|
|
|
writeFile gpgconf $ unlines
|
|
|
|
[ " keyring " ++ keyring
|
|
|
|
, "no-auto-check-trustdb"
|
|
|
|
]
|
|
|
|
-- gpg is picky about perms
|
|
|
|
modifyFileMode privDataDir (removeModes otherGroupModes)
|
|
|
|
s <- readProcessEnv "git" ["log", "-n", "1", "--format=%G?", originbranch]
|
|
|
|
(Just [("GNUPGHOME", privDataDir)])
|
|
|
|
nukeFile $ privDataDir </> "trustring.gpg"
|
|
|
|
nukeFile $ privDataDir </> "gpg.conf"
|
2014-03-31 20:37:19 +00:00
|
|
|
if s == "U\n" || s == "G\n"
|
2014-03-31 20:42:25 +00:00
|
|
|
then do
|
|
|
|
putStrLn $ "git branch " ++ originbranch ++ " gpg signature verified; merging"
|
|
|
|
hFlush stdout
|
2014-03-31 20:37:19 +00:00
|
|
|
else error $ "git branch " ++ originbranch ++ " is not signed with a trusted gpg key; refusing to deploy it!"
|
2014-03-31 20:20:38 +00:00
|
|
|
|
2014-03-31 20:40:03 +00:00
|
|
|
oldsha <- getCurrentGitSha1 branchref
|
2014-03-31 20:20:38 +00:00
|
|
|
void $ boolSystem "git" [Param "merge", Param originbranch]
|
2014-03-31 20:40:03 +00:00
|
|
|
newsha <- getCurrentGitSha1 branchref
|
2014-03-31 20:37:19 +00:00
|
|
|
|
|
|
|
if oldsha == newsha
|
|
|
|
then next
|
|
|
|
else do
|
2014-03-31 20:42:25 +00:00
|
|
|
putStrLn "Rebuilding propeller.."
|
2014-03-31 20:41:10 +00:00
|
|
|
ifM (boolSystem "make" [Param "build"])
|
|
|
|
( void $ boolSystem "./propellor" [Param "--continue", Param (show cmdline)]
|
|
|
|
, error "Propellor build failed!"
|
|
|
|
)
|
2014-03-31 20:20:38 +00:00
|
|
|
|
2014-03-31 20:40:03 +00:00
|
|
|
getCurrentGitSha1 :: String -> IO String
|
|
|
|
getCurrentGitSha1 branchref = readProcess "git" ["show-ref", "--hash", branchref]
|
2014-03-31 20:20:38 +00:00
|
|
|
|
2014-03-30 23:10:32 +00:00
|
|
|
spin :: HostName -> IO ()
|
|
|
|
spin host = do
|
|
|
|
url <- getUrl
|
2014-03-31 16:06:04 +00:00
|
|
|
void $ gitCommit [Param "--allow-empty", Param "-a", Param "-m", Param "propellor spin"]
|
2014-03-30 23:19:29 +00:00
|
|
|
void $ boolSystem "git" [Param "push"]
|
2014-03-31 19:40:16 +00:00
|
|
|
go url =<< gpgDecrypt (privDataFile host)
|
|
|
|
where
|
|
|
|
go url privdata = withBothHandles createProcessSuccess (proc "ssh" [user, bootstrapcmd]) $ \(toh, fromh) -> do
|
|
|
|
let finish = do
|
|
|
|
senddata toh (privDataFile host) privDataMarker privdata
|
|
|
|
hClose toh
|
|
|
|
|
|
|
|
-- Display remaining output.
|
|
|
|
void $ tryIO $ forever $
|
|
|
|
showremote =<< hGetLine fromh
|
|
|
|
hClose fromh
|
2014-03-31 16:28:40 +00:00
|
|
|
status <- getstatus fromh `catchIO` error "protocol error"
|
2014-03-31 16:06:04 +00:00
|
|
|
case status of
|
2014-03-31 19:52:40 +00:00
|
|
|
Ready -> finish
|
2014-03-31 19:40:16 +00:00
|
|
|
NeedGitClone -> do
|
|
|
|
hClose toh
|
|
|
|
hClose fromh
|
|
|
|
sendGitClone host url
|
|
|
|
go url privdata
|
|
|
|
|
2014-03-31 16:06:04 +00:00
|
|
|
user = "root@"++host
|
2014-03-31 19:40:16 +00:00
|
|
|
|
|
|
|
bootstrapcmd = shellWrap $ intercalate " && "
|
2014-03-30 23:10:32 +00:00
|
|
|
[ intercalate " ; "
|
|
|
|
[ "if [ ! -d " ++ localdir ++ " ]"
|
2014-03-30 23:13:26 +00:00
|
|
|
, "then " ++ intercalate " && "
|
2014-03-30 23:10:32 +00:00
|
|
|
[ "apt-get -y install git"
|
2014-03-31 19:40:16 +00:00
|
|
|
, "echo " ++ toMarked statusMarker (show NeedGitClone)
|
2014-03-30 23:10:32 +00:00
|
|
|
]
|
|
|
|
, "fi"
|
|
|
|
]
|
|
|
|
, "cd " ++ localdir
|
2014-03-31 19:40:16 +00:00
|
|
|
, "make build"
|
2014-03-30 23:10:32 +00:00
|
|
|
, "./propellor --boot " ++ host
|
|
|
|
]
|
2014-03-31 19:40:16 +00:00
|
|
|
|
2014-03-31 16:28:40 +00:00
|
|
|
getstatus :: Handle -> IO BootStrapStatus
|
2014-03-31 18:22:48 +00:00
|
|
|
getstatus h = do
|
|
|
|
l <- hGetLine h
|
|
|
|
case readish =<< fromMarked statusMarker l of
|
|
|
|
Nothing -> do
|
2014-03-31 18:24:15 +00:00
|
|
|
showremote l
|
2014-03-31 18:22:48 +00:00
|
|
|
getstatus h
|
|
|
|
Just status -> return status
|
2014-03-31 19:40:16 +00:00
|
|
|
|
2014-03-31 18:44:38 +00:00
|
|
|
showremote s = putStrLn s
|
2014-03-31 19:05:13 +00:00
|
|
|
senddata toh f marker s = do
|
|
|
|
putStr $ "Sending " ++ f ++ " (" ++ show (length s) ++ " bytes) to " ++ host ++ "..."
|
|
|
|
hFlush stdout
|
|
|
|
hPutStrLn toh $ toMarked marker s
|
|
|
|
hFlush toh
|
|
|
|
putStrLn "done"
|
2014-03-30 23:10:32 +00:00
|
|
|
|
2014-03-31 19:40:16 +00:00
|
|
|
sendGitClone :: HostName -> String -> IO ()
|
|
|
|
sendGitClone host url = do
|
|
|
|
putStrLn $ "Pushing git repository to " ++ host
|
|
|
|
withTmpFile "gitbundle" $ \tmp _ -> do
|
|
|
|
-- TODO: ssh connection caching, or better push method
|
|
|
|
-- with less connections.
|
|
|
|
void $ boolSystem "git" [Param "bundle", Param "create", File tmp, Param "HEAD"]
|
|
|
|
void $ boolSystem "scp" [File tmp, Param ("root@"++host++":"++remotebundle)]
|
|
|
|
void $ boolSystem "ssh" [Param ("root@"++host), Param unpackcmd]
|
|
|
|
where
|
|
|
|
remotebundle = "/usr/local/propellor.git"
|
|
|
|
unpackcmd = shellWrap $ intercalate " && "
|
|
|
|
[ "git clone " ++ remotebundle ++ " " ++ localdir
|
|
|
|
, "cd " ++ localdir
|
|
|
|
, "git checkout -b master"
|
|
|
|
, "git remote rm origin"
|
|
|
|
, "git remote add origin " ++ url
|
|
|
|
, "rm -f " ++ remotebundle
|
|
|
|
]
|
|
|
|
|
2014-03-31 19:52:40 +00:00
|
|
|
data BootStrapStatus = Ready | NeedGitClone
|
2014-03-31 16:06:04 +00:00
|
|
|
deriving (Read, Show, Eq)
|
|
|
|
|
|
|
|
type Marker = String
|
|
|
|
type Marked = String
|
|
|
|
|
|
|
|
statusMarker :: Marker
|
|
|
|
statusMarker = "STATUS"
|
|
|
|
|
|
|
|
privDataMarker :: String
|
|
|
|
privDataMarker = "PRIVDATA "
|
|
|
|
|
|
|
|
toMarked :: Marker -> String -> String
|
2014-03-31 19:43:24 +00:00
|
|
|
toMarked marker = intercalate "\n" . map (marker ++) . lines
|
2014-03-31 16:06:04 +00:00
|
|
|
|
2014-03-31 18:21:14 +00:00
|
|
|
fromMarked :: Marker -> Marked -> Maybe String
|
|
|
|
fromMarked marker s
|
|
|
|
| null matches = Nothing
|
2014-03-31 19:43:24 +00:00
|
|
|
| otherwise = Just $ intercalate "\n" $
|
|
|
|
map (drop len) matches
|
2014-03-31 16:06:04 +00:00
|
|
|
where
|
|
|
|
len = length marker
|
2014-03-31 18:21:14 +00:00
|
|
|
matches = filter (marker `isPrefixOf`) $ lines s
|
2014-03-31 16:06:04 +00:00
|
|
|
|
2014-03-30 23:10:32 +00:00
|
|
|
boot :: [Property] -> IO ()
|
|
|
|
boot props = do
|
2014-03-31 19:52:40 +00:00
|
|
|
putStrLn $ toMarked statusMarker $ show Ready
|
2014-03-31 16:28:40 +00:00
|
|
|
hFlush stdout
|
2014-03-31 18:26:56 +00:00
|
|
|
reply <- hGetContentsStrict stdin
|
2014-03-31 18:15:12 +00:00
|
|
|
|
2014-03-30 23:19:29 +00:00
|
|
|
makePrivDataDir
|
2014-03-31 18:21:14 +00:00
|
|
|
maybe noop (writeFileProtected privDataLocal) $
|
|
|
|
fromMarked privDataMarker reply
|
2014-03-30 23:10:32 +00:00
|
|
|
ensureProperties props
|
|
|
|
|
2014-03-31 16:06:04 +00:00
|
|
|
addKey :: String -> IO ()
|
|
|
|
addKey keyid = exitBool =<< allM id [ gpg, gitadd, gitcommit ]
|
|
|
|
where
|
|
|
|
gpg = boolSystem "sh"
|
|
|
|
[ Param "-c"
|
|
|
|
, Param $ "gpg --export " ++ keyid ++ " | gpg " ++
|
|
|
|
unwords (gpgopts ++ ["--import"])
|
|
|
|
]
|
|
|
|
gitadd = boolSystem "git"
|
|
|
|
[ Param "add"
|
|
|
|
, File keyring
|
|
|
|
]
|
|
|
|
gitcommit = gitCommit
|
|
|
|
[ File keyring
|
|
|
|
, Param "-m"
|
|
|
|
, Param "propellor addkey"
|
|
|
|
]
|
|
|
|
|
|
|
|
{- Automatically sign the commit if there'a a keyring. -}
|
|
|
|
gitCommit :: [CommandParam] -> IO Bool
|
|
|
|
gitCommit ps = do
|
|
|
|
k <- doesFileExist keyring
|
|
|
|
boolSystem "git" $ catMaybes $
|
|
|
|
[ Just (Param "commit")
|
|
|
|
, if k then Just (Param "--gpg-sign") else Nothing
|
|
|
|
] ++ map Just ps
|
|
|
|
|
|
|
|
keyring :: FilePath
|
|
|
|
keyring = privDataDir </> "keyring.gpg"
|
|
|
|
|
|
|
|
gpgopts :: [String]
|
|
|
|
gpgopts = ["--options", "/dev/null", "--no-default-keyring", "--keyring", keyring]
|
|
|
|
|
2014-03-30 23:10:32 +00:00
|
|
|
localdir :: FilePath
|
|
|
|
localdir = "/usr/local/propellor"
|
|
|
|
|
|
|
|
getUrl :: IO String
|
|
|
|
getUrl = fromMaybe nourl <$> getM get urls
|
|
|
|
where
|
|
|
|
urls = ["remote.deploy.url", "remote.origin.url"]
|
|
|
|
nourl = error $ "Cannot find deploy url in " ++ show urls
|
|
|
|
get u = do
|
|
|
|
v <- catchMaybeIO $
|
|
|
|
takeWhile (/= '\n')
|
|
|
|
<$> readProcess "git" ["config", u]
|
|
|
|
return $ case v of
|
|
|
|
Just url | not (null url) -> Just url
|
|
|
|
_ -> Nothing
|