From d8e7c45e2e7d3a64f8d9d7e0d7ae621ba21af6be Mon Sep 17 00:00:00 2001 From: Iustin Pop <iustin@google.com> Date: Thu, 8 Nov 2012 13:51:11 +0100 Subject: [PATCH] Convert tag objects to a safer type Currently, we keep information about the "target" of a tag operation in a data type similar to (TagKind, Maybe String). This is unsafe, as nothing (at the type level) prevents us from accidentally having (TagCluster, Just "instance1.example.com"), or (TagInstance, Nothing). To fix this problem, we rename the current TagObject type to TagType (an internal utility type), and create TagObject as a better/safer data type (see the definition), which doesn't allow such possibilities in the future. The downside is that, since at encoding level (both opcode and luxi) this is done in an ugly way (type elements spread at the same level as level as other value), we have to add custom encoders/decoders. The encoder is shared between the OpCode and Luxi usage, the decoder is different however as Luxi uses custom decoding. This also fixes the recent breakage in confd w.r.t. QueryTags. Signed-off-by: Iustin Pop <iustin@google.com> Reviewed-by: Michael Hanselmann <hansmi@google.com> --- htest/Test/Ganeti/Luxi.hs | 2 +- htest/Test/Ganeti/OpCodes.hs | 11 ++++-- htools/Ganeti/Luxi.hs | 7 ++-- htools/Ganeti/OpCodes.hs | 74 +++++++++++++++++++++++++++++------ htools/Ganeti/Query/Server.hs | 10 ++--- 5 files changed, 81 insertions(+), 23 deletions(-) diff --git a/htest/Test/Ganeti/Luxi.hs b/htest/Test/Ganeti/Luxi.hs index 8e01bcc9f..a788e59cf 100644 --- a/htest/Test/Ganeti/Luxi.hs +++ b/htest/Test/Ganeti/Luxi.hs @@ -69,7 +69,7 @@ instance Arbitrary Luxi.LuxiOp where listOf getFQDN <*> arbitrary Luxi.ReqQueryConfigValues -> Luxi.QueryConfigValues <$> getFields Luxi.ReqQueryClusterInfo -> pure Luxi.QueryClusterInfo - Luxi.ReqQueryTags -> Luxi.QueryTags <$> arbitrary <*> getFQDN + Luxi.ReqQueryTags -> Luxi.QueryTags <$> arbitrary Luxi.ReqSubmitJob -> Luxi.SubmitJob <$> resize maxOpCodes arbitrary Luxi.ReqSubmitManyJobs -> Luxi.SubmitManyJobs <$> resize maxOpCodes arbitrary diff --git a/htest/Test/Ganeti/OpCodes.hs b/htest/Test/Ganeti/OpCodes.hs index e2bc8a468..728fa76dc 100644 --- a/htest/Test/Ganeti/OpCodes.hs +++ b/htest/Test/Ganeti/OpCodes.hs @@ -48,7 +48,12 @@ import qualified Ganeti.OpCodes as OpCodes -- * Arbitrary instances -$(genArbitrary ''OpCodes.TagObject) +instance Arbitrary OpCodes.TagObject where + arbitrary = oneof [ OpCodes.TagInstance <$> getFQDN + , OpCodes.TagNode <$> getFQDN + , OpCodes.TagGroup <$> getFQDN + , pure OpCodes.TagCluster + ] $(genArbitrary ''OpCodes.ReplaceDisksMode) @@ -72,9 +77,9 @@ instance Arbitrary OpCodes.OpCode where OpCodes.OpInstanceMigrate <$> getFQDN <*> arbitrary <*> arbitrary <*> arbitrary <*> getMaybe getFQDN "OP_TAGS_SET" -> - OpCodes.OpTagsSet <$> arbitrary <*> genTags <*> getMaybe getFQDN + OpCodes.OpTagsSet <$> arbitrary <*> genTags "OP_TAGS_DEL" -> - OpCodes.OpTagsSet <$> arbitrary <*> genTags <*> getMaybe getFQDN + OpCodes.OpTagsSet <$> arbitrary <*> genTags _ -> fail "Wrong opcode" -- * Test cases diff --git a/htools/Ganeti/Luxi.hs b/htools/Ganeti/Luxi.hs index 622eceb6c..8ad7f32b7 100644 --- a/htools/Ganeti/Luxi.hs +++ b/htools/Ganeti/Luxi.hs @@ -137,8 +137,8 @@ $(genLuxiOp "LuxiOp" ) , (luxiReqQueryClusterInfo, []) , (luxiReqQueryTags, - [ simpleField "kind" [t| TagObject |] - , simpleField "name" [t| String |] + [ customField 'decodeTagObject 'encodeTagObject $ + simpleField "kind" [t| TagObject |] ]) , (luxiReqSubmitJob, [ simpleField "job" [t| [OpCode] |] ] @@ -386,7 +386,8 @@ decodeCall (LuxiCall call args) = return $ QueryConfigValues fields ReqQueryTags -> do (kind, name) <- fromJVal args - return $ QueryTags kind name + item <- tagObjectFrom kind name + return $ QueryTags item ReqCancelJob -> do [job] <- fromJVal args rid <- parseJobId job diff --git a/htools/Ganeti/OpCodes.hs b/htools/Ganeti/OpCodes.hs index 789d84a04..e2c91810f 100644 --- a/htools/Ganeti/OpCodes.hs +++ b/htools/Ganeti/OpCodes.hs @@ -28,6 +28,9 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA module Ganeti.OpCodes ( OpCode(..) , TagObject(..) + , tagObjectFrom + , encodeTagObject + , decodeTagObject , ReplaceDisksMode(..) , DiskIndex , mkDiskIndex @@ -36,7 +39,8 @@ module Ganeti.OpCodes , allOpIDs ) where -import Text.JSON (readJSON, showJSON, makeObj, JSON) +import Text.JSON (readJSON, showJSON, makeObj, JSON, JSValue(..), fromJSString) +import Text.JSON.Pretty (pp_value) import qualified Ganeti.Constants as C import Ganeti.THH @@ -44,13 +48,61 @@ import Ganeti.THH import Ganeti.JSON -- | Data type representing what items do the tag operations apply to. -$(declareSADT "TagObject" - [ ("TagInstance", 'C.tagInstance) - , ("TagNode", 'C.tagNode) - , ("TagGroup", 'C.tagNodegroup) - , ("TagCluster", 'C.tagCluster) +$(declareSADT "TagType" + [ ("TagTypeInstance", 'C.tagInstance) + , ("TagTypeNode", 'C.tagNode) + , ("TagTypeGroup", 'C.tagNodegroup) + , ("TagTypeCluster", 'C.tagCluster) ]) -$(makeJSONInstance ''TagObject) +$(makeJSONInstance ''TagType) + +-- | Data type holding a tag object (type and object name). +data TagObject = TagInstance String + | TagNode String + | TagGroup String + | TagCluster + deriving (Show, Read, Eq) + +-- | Tag type for a given tag object. +tagTypeOf :: TagObject -> TagType +tagTypeOf (TagInstance {}) = TagTypeInstance +tagTypeOf (TagNode {}) = TagTypeNode +tagTypeOf (TagGroup {}) = TagTypeGroup +tagTypeOf (TagCluster {}) = TagTypeCluster + +-- | Gets the potential tag object name. +tagNameOf :: TagObject -> Maybe String +tagNameOf (TagInstance s) = Just s +tagNameOf (TagNode s) = Just s +tagNameOf (TagGroup s) = Just s +tagNameOf TagCluster = Nothing + +-- | Builds a 'TagObject' from a tag type and name. +tagObjectFrom :: (Monad m) => TagType -> JSValue -> m TagObject +tagObjectFrom TagTypeInstance (JSString s) = + return . TagInstance $ fromJSString s +tagObjectFrom TagTypeNode (JSString s) = return . TagNode $ fromJSString s +tagObjectFrom TagTypeGroup (JSString s) = return . TagGroup $ fromJSString s +tagObjectFrom TagTypeCluster JSNull = return TagCluster +tagObjectFrom t v = + fail $ "Invalid tag type/name combination: " ++ show t ++ "/" ++ + show (pp_value v) + +-- | Name of the tag \"name\" field. +tagNameField :: String +tagNameField = "name" + +-- | Custom encoder for 'TagObject' as represented in an opcode. +encodeTagObject :: TagObject -> (JSValue, [(String, JSValue)]) +encodeTagObject t = ( showJSON (tagTypeOf t) + , [(tagNameField, maybe JSNull showJSON (tagNameOf t))] ) + +-- | Custom decoder for 'TagObject' as represented in an opcode. +decodeTagObject :: (Monad m) => [(String, JSValue)] -> JSValue -> m TagObject +decodeTagObject obj kind = do + ttype <- fromJVal kind + tname <- fromObj obj tagNameField + tagObjectFrom ttype tname -- | Replace disks type. $(declareSADT "ReplaceDisksMode" @@ -107,14 +159,14 @@ $(genOpCode "OpCode" , optionalField $ simpleField "target_node" [t| String |] ]) , ("OpTagsSet", - [ simpleField "kind" [t| TagObject |] + [ customField 'decodeTagObject 'encodeTagObject $ + simpleField "kind" [t| TagObject |] , simpleField "tags" [t| [String] |] - , optionalNullSerField $ simpleField "name" [t| String |] ]) , ("OpTagsDel", - [ simpleField "kind" [t| TagObject |] + [ customField 'decodeTagObject 'encodeTagObject $ + simpleField "kind" [t| TagObject |] , simpleField "tags" [t| [String] |] - , optionalNullSerField $ simpleField "name" [t| String |] ]) ]) diff --git a/htools/Ganeti/Query/Server.hs b/htools/Ganeti/Query/Server.hs index 91aa745ca..6b33a9ea9 100644 --- a/htools/Ganeti/Query/Server.hs +++ b/htools/Ganeti/Query/Server.hs @@ -138,12 +138,12 @@ handleCall cdata QueryClusterInfo = in return . Ok . J.makeObj $ obj -handleCall cfg (QueryTags kind name) = +handleCall cfg (QueryTags kind) = let tags = case kind of - TagCluster -> Ok . clusterTags $ configCluster cfg - TagGroup -> groupTags <$> Config.getGroup cfg name - TagNode -> nodeTags <$> Config.getNode cfg name - TagInstance -> instTags <$> Config.getInstance cfg name + TagCluster -> Ok . clusterTags $ configCluster cfg + TagGroup name -> groupTags <$> Config.getGroup cfg name + TagNode name -> nodeTags <$> Config.getNode cfg name + TagInstance name -> instTags <$> Config.getInstance cfg name in return (J.showJSON <$> tags) handleCall cfg (Query qkind qfields qfilter) = do -- GitLab