remove toSimpleProp

It didn't do what I thought it did with a RevertableProperty; it always
returned Nothing because even if the input properties to <!> are NoInfo, it
casts them to HasInfo.

Even if it had worked, it lost type safety. Better to export the
Property NoInfo that is used in a RevertableProperty, so it can be used
directly.
This commit is contained in:
Joey Hess 2015-01-25 14:45:14 -04:00
parent 334abae312
commit e9d5d9aff1
6 changed files with 59 additions and 35 deletions

View File

@ -0,0 +1,18 @@
Currently, a RevertableProperty's Properties always both HasInfo. This
means that if a Property NoInfo is updated to be a RevertableProperty, and
someplace called ensureProperty on it, that will refuse to compile.
The workaround is generally to export the original NoInfo property under
a different name, so it can still be used with ensureProperty.
This could be fixed:
data RevertableProperty i1 i2 where
RProp :: Property i1 -> Property i2 -> RevertableProperty i1 i2
However, needing to write "RevertableProperty HasInfo NoInfo" is quite
a mouthful!
Since only 2 places in the propellor source code currently need to deal
with this, it doesn't currently seem worth making the change, unless a less
intrusive way can be found.

View File

@ -266,17 +266,24 @@ data AptKey = AptKey
} }
trustsKey :: AptKey -> RevertableProperty trustsKey :: AptKey -> RevertableProperty
trustsKey k = trust <!> untrust trustsKey k = trustsKey' k <!> untrustKey k
trustsKey' :: AptKey -> Property NoInfo
trustsKey' k = check (not <$> doesFileExist f) $ property desc $ makeChange $ do
withHandle StdinHandle createProcessSuccess
(proc "gpg" ["--no-default-keyring", "--keyring", f, "--import", "-"]) $ \h -> do
hPutStr h (pubkey k)
hClose h
nukeFile $ f ++ "~" -- gpg dropping
where where
desc = "apt trusts key " ++ keyname k desc = "apt trusts key " ++ keyname k
f = "/etc/apt/trusted.gpg.d" </> keyname k ++ ".gpg" f = aptKeyFile k
untrust = File.notPresent f
trust = check (not <$> doesFileExist f) $ property desc $ makeChange $ do untrustKey :: AptKey -> Property NoInfo
withHandle StdinHandle createProcessSuccess untrustKey = File.notPresent . aptKeyFile
(proc "gpg" ["--no-default-keyring", "--keyring", f, "--import", "-"]) $ \h -> do
hPutStr h (pubkey k) aptKeyFile :: AptKey -> FilePath
hClose h aptKeyFile k = "/etc/apt/trusted.gpg.d" </> keyname k ++ ".gpg"
nukeFile $ f ++ "~" -- gpg dropping
-- | Cleans apt's cache of downloaded packages to avoid using up disk -- | Cleans apt's cache of downloaded packages to avoid using up disk
-- space. -- space.

View File

@ -1,3 +1,5 @@
{-# LANGUAGE FlexibleContexts #-}
module Propellor.Property.Debootstrap ( module Propellor.Property.Debootstrap (
Url, Url,
DebootstrapConfig(..), DebootstrapConfig(..),
@ -56,18 +58,18 @@ toParams (c1 :+ c2) = toParams c1 <> toParams c2
-- Note that reverting this property does not stop any processes -- Note that reverting this property does not stop any processes
-- currently running in the chroot. -- currently running in the chroot.
built :: FilePath -> System -> DebootstrapConfig -> RevertableProperty built :: FilePath -> System -> DebootstrapConfig -> RevertableProperty
built = built' (toProp installed) built target system config = built' (toProp installed) target system config <!> teardown
built' :: Property HasInfo -> FilePath -> System -> DebootstrapConfig -> RevertableProperty
built' installprop target system@(System _ arch) config = setup <!> teardown
where where
setup = check (unpopulated target <||> ispartial) setupprop
`requires` installprop
teardown = check (not <$> unpopulated target) teardownprop teardown = check (not <$> unpopulated target) teardownprop
unpopulated d = null <$> catchDefaultIO [] (dirContents d) teardownprop = property ("removed debootstrapped " ++ target) $
makeChange (removetarget target)
built' :: (Combines (Property NoInfo) (Property i)) => Property i -> FilePath -> System -> DebootstrapConfig -> Property (CInfo NoInfo i)
built' installprop target system@(System _ arch) config =
check (unpopulated target <||> ispartial) setupprop
`requires` installprop
where
setupprop = property ("debootstrapped " ++ target) $ liftIO $ do setupprop = property ("debootstrapped " ++ target) $ liftIO $ do
createDirectoryIfMissing True target createDirectoryIfMissing True target
-- Don't allow non-root users to see inside the chroot, -- Don't allow non-root users to see inside the chroot,
@ -92,25 +94,26 @@ built' installprop target system@(System _ arch) config = setup <!> teardown
, return FailedChange , return FailedChange
) )
teardownprop = property ("removed debootstrapped " ++ target) $
makeChange removetarget
removetarget = do
submnts <- filter (\p -> simplifyPath p /= simplifyPath target)
. filter (dirContains target)
<$> mountPoints
forM_ submnts umountLazy
removeDirectoryRecursive target
-- A failed debootstrap run will leave a debootstrap directory; -- A failed debootstrap run will leave a debootstrap directory;
-- recover by deleting it and trying again. -- recover by deleting it and trying again.
ispartial = ifM (doesDirectoryExist (target </> "debootstrap")) ispartial = ifM (doesDirectoryExist (target </> "debootstrap"))
( do ( do
removetarget removetarget target
return True return True
, return False , return False
) )
unpopulated :: FilePath -> IO Bool
unpopulated d = null <$> catchDefaultIO [] (dirContents d)
removetarget :: FilePath -> IO ()
removetarget target = do
submnts <- filter (\p -> simplifyPath p /= simplifyPath target)
. filter (dirContains target)
<$> mountPoints
forM_ submnts umountLazy
removeDirectoryRecursive target
extractSuite :: System -> Maybe String extractSuite :: System -> Maybe String
extractSuite (System (Debian s) _) = Just $ Apt.showSuite s extractSuite (System (Debian s) _) = Just $ Apt.showSuite s
extractSuite (System (Ubuntu r) _) = Just r extractSuite (System (Ubuntu r) _) = Just r

View File

@ -89,10 +89,10 @@ cleanInstallOnce confirmation = check (not <$> doesFileExist flagfile) $
(Just u@(System (Ubuntu _) _)) -> debootstrap u (Just u@(System (Ubuntu _) _)) -> debootstrap u
_ -> error "os is not declared to be Debian or Ubuntu" _ -> error "os is not declared to be Debian or Ubuntu"
debootstrap targetos = ensureProperty $ fromJust $ toSimpleProp $ debootstrap targetos = ensureProperty $
-- Ignore the os setting, and install debootstrap from -- Ignore the os setting, and install debootstrap from
-- source, since we don't know what OS we're running in yet. -- source, since we don't know what OS we're running in yet.
Debootstrap.built' (toProp Debootstrap.sourceInstall) Debootstrap.built' Debootstrap.sourceInstall
newOSDir targetos Debootstrap.DefaultConfig newOSDir targetos Debootstrap.DefaultConfig
-- debootstrap, I wish it was faster.. -- debootstrap, I wish it was faster..
-- TODO eatmydata to speed it up -- TODO eatmydata to speed it up

View File

@ -118,7 +118,7 @@ latestVersion :: Property NoInfo
latestVersion = withOS "obnam latest version" $ \o -> case o of latestVersion = withOS "obnam latest version" $ \o -> case o of
(Just (System (Debian suite) _)) | isStable suite -> ensureProperty $ (Just (System (Debian suite) _)) | isStable suite -> ensureProperty $
Apt.setSourcesListD (stablesources suite) "obnam" Apt.setSourcesListD (stablesources suite) "obnam"
`requires` (fromJust (toSimpleProp (Apt.trustsKey key))) `requires` Apt.trustsKey' key
_ -> noChange _ -> noChange
where where
stablesources suite = stablesources suite =

View File

@ -179,7 +179,6 @@ class IsProp p where
-- | Sets description. -- | Sets description.
describe :: p -> Desc -> p describe :: p -> Desc -> p
toProp :: p -> Property HasInfo toProp :: p -> Property HasInfo
toSimpleProp :: p -> Maybe (Property NoInfo)
getDesc :: p -> Desc getDesc :: p -> Desc
-- | Gets the info of the property, combined with all info -- | Gets the info of the property, combined with all info
-- of all children properties. -- of all children properties.
@ -188,7 +187,6 @@ class IsProp p where
instance IsProp (Property HasInfo) where instance IsProp (Property HasInfo) where
describe (IProperty _ a i cs) d = IProperty d a i cs describe (IProperty _ a i cs) d = IProperty d a i cs
toProp = id toProp = id
toSimpleProp _ = Nothing
getDesc = propertyDesc getDesc = propertyDesc
getInfoRecursive (IProperty _ _ i cs) = getInfoRecursive (IProperty _ _ i cs) =
i <> mconcat (map getInfoRecursive cs) i <> mconcat (map getInfoRecursive cs)
@ -196,7 +194,6 @@ instance IsProp (Property NoInfo) where
describe (SProperty _ a cs) d = SProperty d a cs describe (SProperty _ a cs) d = SProperty d a cs
toProp = toIProperty toProp = toIProperty
getDesc = propertyDesc getDesc = propertyDesc
toSimpleProp = Just
getInfoRecursive _ = mempty getInfoRecursive _ = mempty
instance IsProp RevertableProperty where instance IsProp RevertableProperty where
@ -205,7 +202,6 @@ instance IsProp RevertableProperty where
RevertableProperty (describe p1 d) (describe p2 ("not " ++ d)) RevertableProperty (describe p1 d) (describe p2 ("not " ++ d))
getDesc (RevertableProperty p1 _) = getDesc p1 getDesc (RevertableProperty p1 _) = getDesc p1
toProp (RevertableProperty p1 _) = p1 toProp (RevertableProperty p1 _) = p1
toSimpleProp = toSimpleProp . toProp
-- | Return the Info of the currently active side. -- | Return the Info of the currently active side.
getInfoRecursive (RevertableProperty p1 _p2) = getInfoRecursive p1 getInfoRecursive (RevertableProperty p1 _p2) = getInfoRecursive p1