propellor/src/Propellor/Property/OS.hs

189 lines
6.0 KiB
Haskell
Raw Normal View History

2014-11-23 19:42:22 +00:00
module Propellor.Property.OS (
cleanInstallOnce,
2014-12-04 21:11:15 +00:00
Confirmation(..),
2014-11-24 04:52:46 +00:00
preserveNetworkInterfaces,
preserveRootSshAuthorized,
2014-11-23 23:49:53 +00:00
grubBoots,
2014-12-04 21:30:40 +00:00
GrubDev,
rebootForced,
2014-11-23 23:49:53 +00:00
kernelInstalled,
2014-11-23 19:42:22 +00:00
oldOSRemoved,
) where
import Propellor
import qualified Propellor.Property.Debootstrap as Debootstrap
2014-11-24 04:51:36 +00:00
import qualified Propellor.Property.Ssh as Ssh
2014-12-04 21:11:15 +00:00
import qualified Propellor.Property.User as User
import Propellor.Property.Mount
2014-12-04 21:30:40 +00:00
import Propellor.Property.Chroot.Util (stdPATH)
2014-12-04 21:11:15 +00:00
import System.Posix.Files (rename, fileExist)
2014-11-24 04:40:53 +00:00
2014-11-23 19:42:22 +00:00
-- | Replaces whatever OS was installed before with a clean installation
-- of the OS that the Host is configured to have.
--
-- This can replace one Linux distribution with different one.
-- But, it can also fail and leave the system in an unbootable state.
--
-- To avoid this property being accidentially used, you have to provide
2014-12-04 21:30:40 +00:00
-- a Confirmation containing the name of the host that you intend to apply
-- the property to.
--
2014-11-23 19:42:22 +00:00
-- This property only runs once. The cleanly installed system will have
-- a file /etc/propellor-cleaninstall, which indicates it was cleanly
-- installed.
2014-12-04 21:30:40 +00:00
--
-- The files from the old os will be left in /old-os
2014-11-23 19:42:22 +00:00
--
-- You will typically want to run some more properties after the clean
2014-12-04 21:30:40 +00:00
-- install succeeds, to bootstrap from the cleanly installed system to
-- a fully working system. For example:
2014-11-23 19:42:22 +00:00
--
-- > & os (System (Debian Unstable) "amd64")
2014-12-04 21:11:15 +00:00
-- > & cleanInstall (Confirmed "foo.example.com")
2014-11-23 19:42:22 +00:00
-- > `onChange` propertyList "fixing up after clean install"
2014-11-24 04:52:46 +00:00
-- > [ preserveNetworkInterfaces
-- > , preserverRootSshAuthorized
2014-11-23 23:49:53 +00:00
-- > -- , kernelInstalled
-- > -- , grubBoots "hd0"
2014-12-04 21:30:40 +00:00
-- > -- , oldOsRemoved (Confirmed "foo.example.com")
-- > -- , rebootForced
2014-11-23 19:42:22 +00:00
-- > ]
-- > & Apt.installed ["ssh"]
-- > & User.hasSomePassword "root"
-- > & User.accountFor "joey"
-- > & User.hasSomePassword "joey"
2014-11-23 19:42:22 +00:00
-- > -- rest of system properties here
2014-12-04 21:11:15 +00:00
cleanInstallOnce :: Confirmation -> Property
cleanInstallOnce confirmation = check (not <$> doesFileExist flagfile) $
go `requires` confirmed "clean install confirmed" confirmation
where
go =
finalized
`requires`
propellorbootstrapped
`requires`
User.shadowConfig True
`requires`
flipped
`requires`
umountall
`requires`
osbootstrapped
osbootstrapped = withOS (newOSDir ++ " bootstrapped") $ \o -> case o of
(Just d@(System (Debian _) _)) -> debootstrap d
(Just u@(System (Ubuntu _) _)) -> debootstrap u
_ -> error "os is not declared to be Debian or Ubuntu"
debootstrap targetos = ensureProperty $ toProp $
Debootstrap.built newOSDir targetos Debootstrap.DefaultConfig
umountall = property "mount points unmounted" $ liftIO $ do
mnts <- filter (`notElem` ("/": trickydirs)) <$> mountPoints
-- reverse so that deeper mount points come first
forM_ (reverse mnts) umountLazy
return $ if null mnts then NoChange else MadeChange
flipped = property (newOSDir ++ " moved into place") $ liftIO $ do
createDirectoryIfMissing True oldOSDir
rootcontents <- dirContents "/"
forM_ rootcontents $ \d ->
when (d `notElem` (oldOSDir:newOSDir:trickydirs)) $
rename d (oldOSDir ++ d)
newrootcontents <- dirContents newOSDir
forM_ newrootcontents $ \d -> do
let dest = "/" ++ takeFileName d
whenM (not <$> fileExist dest) $
rename d dest
removeDirectoryRecursive newOSDir
2014-12-04 21:30:40 +00:00
-- Prepare environment for running additional properties.
liftIO $ writeFile flagfile ""
void $ setEnv "PATH" stdPATH True
2014-12-04 21:11:15 +00:00
return MadeChange
propellorbootstrapped = property "propellor re-debootstrapped in new os" $
return NoChange
2014-11-23 19:42:22 +00:00
-- re-bootstrap propellor in /usr/local/propellor,
-- (using git repo bundle, privdata file, and possibly
-- git repo url, which all need to be arranged to
-- be present in /old-os's /usr/local/propellor)
2014-12-04 21:30:40 +00:00
-- TODO
2014-12-04 21:11:15 +00:00
2014-12-04 21:30:40 +00:00
-- Ensure that MadeChange is returned by the overall property,
-- so that anything hooking in onChange will run afterwards.
finalized = property "clean OS installed" $ return MadeChange
2014-12-04 21:11:15 +00:00
2014-11-23 19:42:22 +00:00
flagfile = "/etc/propellor-cleaninstall"
2014-12-04 21:30:40 +00:00
trickydirs =
-- /tmp can contain X's sockets, which prevent moving it
-- so it's left as-is.
[ "/tmp"
-- /proc is left mounted
, "/proc"
]
2014-11-23 19:42:22 +00:00
2014-12-04 21:11:15 +00:00
data Confirmation = Confirmed HostName
2014-11-24 04:40:53 +00:00
2014-12-04 21:11:15 +00:00
confirmed :: Desc -> Confirmation -> Property
confirmed desc (Confirmed c) = property desc $ do
2014-11-24 04:40:53 +00:00
hostname <- asks hostName
2014-12-04 21:11:15 +00:00
if hostname /= c
then do
warningMessage "Run with a bad confirmation, not matching hostname."
return FailedChange
else return NoChange
2014-11-23 19:42:22 +00:00
2014-12-04 21:30:40 +00:00
-- | /etc/network/interfaces is configured to bring up all interfaces that
2014-11-23 19:42:22 +00:00
-- are currently up, using the same IP addresses.
2014-11-24 04:52:46 +00:00
preserveNetworkInterfaces :: Property
preserveNetworkInterfaces = undefined
2014-11-23 19:42:22 +00:00
2014-12-04 21:30:40 +00:00
-- | Root's .ssh/authorized_keys has added to it any ssh keys that
2014-11-24 04:40:53 +00:00
-- were authorized in the old OS. Any other contents of the file are
-- retained.
2014-11-24 04:52:46 +00:00
preserveRootSshAuthorized :: Property
preserveRootSshAuthorized = check (doesDirectoryExist oldloc) $
2014-11-24 04:40:53 +00:00
property (newloc ++ " copied from old OS") $ do
ks <- liftIO $ lines <$> readFile oldloc
2014-11-24 04:51:36 +00:00
ensureProperties (map (Ssh.authorizedKey "root") ks)
2014-11-24 04:40:53 +00:00
where
newloc = "/root/.ssh/authorized_keys"
2014-12-04 21:11:15 +00:00
oldloc = oldOSDir ++ newloc
2014-11-23 19:42:22 +00:00
2014-12-04 21:30:40 +00:00
-- | Installs an appropriate kernel from the OS distribution.
2014-11-23 23:49:53 +00:00
kernelInstalled :: Property
kernelInstalled = undefined
2014-11-23 19:42:22 +00:00
2014-12-04 21:30:40 +00:00
-- | Installs grub onto a device to boot the system.
2014-11-24 04:40:53 +00:00
--
-- You may want to install grub to multiple devices; eg for a system
-- that uses software RAID.
2014-11-23 23:49:53 +00:00
grubBoots :: GrubDev -> Property
grubBoots = undefined
2014-11-23 19:42:22 +00:00
2014-11-24 04:40:53 +00:00
type GrubDev = String
2014-12-04 21:30:40 +00:00
-- | Forces an immediate reboot, without contacting the init system.
--
-- Can be used after cleanInstallOnce.
rebootForced :: Property
rebootForced = cmdProperty "reboot" [ "--force" ]
2014-11-23 19:42:22 +00:00
-- Removes the old OS's backup from /old-os
2014-12-04 21:11:15 +00:00
oldOSRemoved :: Confirmation -> Property
oldOSRemoved confirmation = check (doesDirectoryExist oldOSDir) $
go `requires` confirmed "old OS backup removal confirmed" confirmation
where
go = property "old OS backup removed" $ do
liftIO $ removeDirectoryRecursive oldOSDir
2014-11-23 19:42:22 +00:00
return MadeChange
2014-12-04 21:11:15 +00:00
oldOSDir :: FilePath
oldOSDir = "/old-os"
newOSDir :: FilePath
newOSDir = "/new-os"