From cd67e3376f62b8ab278d4d7905ad65208070130b Mon Sep 17 00:00:00 2001 From: Iustin Pop <iustin@google.com> Date: Thu, 4 Oct 2012 02:22:14 +0100 Subject: [PATCH] Add support for classic queries This patch adds support for classic-style queries (before query2) to the query socket server. The patch is rather trivial, since as in Python we just piggy-back on the query2 implementation. Signed-off-by: Iustin Pop <iustin@google.com> Reviewed-by: Agata Murawska <agatamurawska@google.com> --- htools/Ganeti/Query/Query.hs | 14 +++++++++++++- htools/Ganeti/Query/Server.hs | 20 ++++++++++++++++++++ 2 files changed, 33 insertions(+), 1 deletion(-) diff --git a/htools/Ganeti/Query/Query.hs b/htools/Ganeti/Query/Query.hs index b0090faa3..2d10c3db0 100644 --- a/htools/Ganeti/Query/Query.hs +++ b/htools/Ganeti/Query/Query.hs @@ -45,16 +45,19 @@ too. -} module Ganeti.Query.Query - ( query , queryFields + , queryCompat , getRequestedNames + , nameField ) where import Control.Monad (filterM) import Control.Monad.Trans (lift) +import Data.List (intercalate) import Data.Maybe (fromMaybe) import qualified Data.Map as Map +import qualified Text.JSON as J import Ganeti.BasicTypes import Ganeti.Config @@ -209,3 +212,12 @@ queryFields (QueryFields QRGroup fields) = queryFields (QueryFields qkind _) = Bad $ "QueryFields '" ++ show qkind ++ "' not supported" + +-- | Classic query converter. It gets a standard query result on input +-- and computes the classic style results. +queryCompat :: QueryResult -> Result [[J.JSValue]] +queryCompat (QueryResult fields qrdata) = + case map fdefName $ filter ((== QFTUnknown) . fdefKind) fields of + [] -> Ok $ map (map (maybe J.JSNull J.showJSON . rentryValue)) qrdata + unknown -> Bad $ "Unknown output fields selected: " ++ + intercalate ", " unknown diff --git a/htools/Ganeti/Query/Server.hs b/htools/Ganeti/Query/Server.hs index ca4409c20..97ece0b6a 100644 --- a/htools/Ganeti/Query/Server.hs +++ b/htools/Ganeti/Query/Server.hs @@ -51,11 +51,25 @@ import Ganeti.Logging import Ganeti.Luxi import qualified Ganeti.Query.Language as Qlang import Ganeti.Query.Query +import Ganeti.Query.Filter (makeSimpleFilter) -- | A type for functions that can return the configuration when -- executed. type ConfigReader = IO (Result ConfigData) +-- | Helper for classic queries. +handleClassicQuery :: ConfigData -- ^ Cluster config + -> Qlang.ItemType -- ^ Query type + -> [String] -- ^ Requested names (empty means all) + -> [String] -- ^ Requested fields + -> Bool -- ^ Whether to do sync queries or not + -> IO (Result JSValue) +handleClassicQuery _ _ _ _ True = return . Bad $ "Sync queries are not allowed" +handleClassicQuery cfg qkind names fields _ = do + let flt = makeSimpleFilter (nameField qkind) names + qr <- query cfg True (Qlang.Query qkind fields flt) + return $ showJSON <$> (qr >>= queryCompat) + -- | Minimal wrapper to handle the missing config case. handleCallWrapper :: Result ConfigData -> LuxiOp -> IO (Result JSValue) handleCallWrapper (Bad msg) _ = @@ -136,6 +150,12 @@ handleCall _ (QueryFields qkind qfields) = do let result = queryFields (Qlang.QueryFields qkind qfields) return $ J.showJSON <$> result +handleCall cfg (QueryNodes names fields lock) = + handleClassicQuery cfg Qlang.QRNode names fields lock + +handleCall cfg (QueryGroups names fields lock) = + handleClassicQuery cfg Qlang.QRGroup names fields lock + handleCall _ op = return . Bad $ "Luxi call '" ++ strOfOp op ++ "' not implemented" -- GitLab