root / Ganeti / HTools / IAlloc.hs @ 0ca66853
History | View | Annotate | Download (6.4 kB)
1 | 43643696 | Iustin Pop | {-| Implementation of the iallocator interface. |
---|---|---|---|
2 | 43643696 | Iustin Pop | |
3 | 43643696 | Iustin Pop | -} |
4 | 43643696 | Iustin Pop | |
5 | e2fa2baf | Iustin Pop | {- |
6 | e2fa2baf | Iustin Pop | |
7 | e2fa2baf | Iustin Pop | Copyright (C) 2009 Google Inc. |
8 | e2fa2baf | Iustin Pop | |
9 | e2fa2baf | Iustin Pop | This program is free software; you can redistribute it and/or modify |
10 | e2fa2baf | Iustin Pop | it under the terms of the GNU General Public License as published by |
11 | e2fa2baf | Iustin Pop | the Free Software Foundation; either version 2 of the License, or |
12 | e2fa2baf | Iustin Pop | (at your option) any later version. |
13 | e2fa2baf | Iustin Pop | |
14 | e2fa2baf | Iustin Pop | This program is distributed in the hope that it will be useful, but |
15 | e2fa2baf | Iustin Pop | WITHOUT ANY WARRANTY; without even the implied warranty of |
16 | e2fa2baf | Iustin Pop | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
17 | e2fa2baf | Iustin Pop | General Public License for more details. |
18 | e2fa2baf | Iustin Pop | |
19 | e2fa2baf | Iustin Pop | You should have received a copy of the GNU General Public License |
20 | e2fa2baf | Iustin Pop | along with this program; if not, write to the Free Software |
21 | e2fa2baf | Iustin Pop | Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA |
22 | e2fa2baf | Iustin Pop | 02110-1301, USA. |
23 | e2fa2baf | Iustin Pop | |
24 | e2fa2baf | Iustin Pop | -} |
25 | e2fa2baf | Iustin Pop | |
26 | 43643696 | Iustin Pop | module Ganeti.HTools.IAlloc |
27 | 19f38ee8 | Iustin Pop | ( parseData |
28 | 43643696 | Iustin Pop | , formatResponse |
29 | 43643696 | Iustin Pop | ) where |
30 | 43643696 | Iustin Pop | |
31 | 43643696 | Iustin Pop | import Data.Either () |
32 | 43643696 | Iustin Pop | import Control.Monad |
33 | 942403e6 | Iustin Pop | import Text.JSON (JSObject, JSValue(JSBool, JSString, JSArray), |
34 | 942403e6 | Iustin Pop | makeObj, encodeStrict, decodeStrict, |
35 | 942403e6 | Iustin Pop | fromJSObject, toJSString) |
36 | 262a08a2 | Iustin Pop | import qualified Ganeti.HTools.Container as Container |
37 | 942403e6 | Iustin Pop | import qualified Ganeti.HTools.Node as Node |
38 | 942403e6 | Iustin Pop | import qualified Ganeti.HTools.Instance as Instance |
39 | e4c5beaf | Iustin Pop | import Ganeti.HTools.Loader |
40 | e4c5beaf | Iustin Pop | import Ganeti.HTools.Utils |
41 | e4c5beaf | Iustin Pop | import Ganeti.HTools.Types |
42 | 43643696 | Iustin Pop | |
43 | 9188aeef | Iustin Pop | -- | Parse the basic specifications of an instance. |
44 | 9188aeef | Iustin Pop | -- |
45 | 9188aeef | Iustin Pop | -- Instances in the cluster instance list and the instance in an |
46 | 9188aeef | Iustin Pop | -- 'Allocate' request share some common properties, which are read by |
47 | 9188aeef | Iustin Pop | -- this function. |
48 | e4c5beaf | Iustin Pop | parseBaseInstance :: String |
49 | 262f3e6c | Iustin Pop | -> [(String, JSValue)] |
50 | e4c5beaf | Iustin Pop | -> Result (String, Instance.Instance) |
51 | e4c5beaf | Iustin Pop | parseBaseInstance n a = do |
52 | 734b1ff1 | Iustin Pop | disk <- fromObj "disk_space_total" a |
53 | e4c5beaf | Iustin Pop | mem <- fromObj "memory" a |
54 | d752eb39 | Iustin Pop | vcpus <- fromObj "vcpus" a |
55 | 17e7af2b | Iustin Pop | tags <- fromObj "tags" a |
56 | e4c5beaf | Iustin Pop | let running = "running" |
57 | 17e7af2b | Iustin Pop | return (n, Instance.create n mem disk vcpus running tags 0 0) |
58 | 585d4420 | Iustin Pop | |
59 | 262f3e6c | Iustin Pop | -- | Parses an instance as found in the cluster instance listg. |
60 | 9188aeef | Iustin Pop | parseInstance :: NameAssoc -- ^ The node name-to-index association list |
61 | 9188aeef | Iustin Pop | -> String -- ^ The name of the instance |
62 | 262f3e6c | Iustin Pop | -> [(String, JSValue)] -- ^ The JSON object |
63 | e4c5beaf | Iustin Pop | -> Result (String, Instance.Instance) |
64 | e4c5beaf | Iustin Pop | parseInstance ktn n a = do |
65 | 262f3e6c | Iustin Pop | base <- parseBaseInstance n a |
66 | 262f3e6c | Iustin Pop | nodes <- fromObj "nodes" a |
67 | e41f4ba0 | Iustin Pop | pnode <- if null nodes |
68 | e41f4ba0 | Iustin Pop | then Bad $ "empty node list for instance " ++ n |
69 | e41f4ba0 | Iustin Pop | else readEitherString $ head nodes |
70 | 262f3e6c | Iustin Pop | pidx <- lookupNode ktn n pnode |
71 | 262f3e6c | Iustin Pop | let snodes = tail nodes |
72 | 262f3e6c | Iustin Pop | sidx <- (if null snodes then return Node.noSecondary |
73 | 262f3e6c | Iustin Pop | else readEitherString (head snodes) >>= lookupNode ktn n) |
74 | 262f3e6c | Iustin Pop | return (n, Instance.setBoth (snd base) pidx sidx) |
75 | 585d4420 | Iustin Pop | |
76 | 9188aeef | Iustin Pop | -- | Parses a node as found in the cluster node list. |
77 | 9188aeef | Iustin Pop | parseNode :: String -- ^ The node's name |
78 | 262f3e6c | Iustin Pop | -> [(String, JSValue)] -- ^ The JSON object |
79 | 9188aeef | Iustin Pop | -> Result (String, Node.Node) |
80 | e4c5beaf | Iustin Pop | parseNode n a = do |
81 | 262f3e6c | Iustin Pop | offline <- fromObj "offline" a |
82 | 262f3e6c | Iustin Pop | drained <- fromObj "drained" a |
83 | 262f3e6c | Iustin Pop | node <- (if offline || drained |
84 | 262f3e6c | Iustin Pop | then return $ Node.create n 0 0 0 0 0 0 True |
85 | 262f3e6c | Iustin Pop | else do |
86 | 262f3e6c | Iustin Pop | mtotal <- fromObj "total_memory" a |
87 | 262f3e6c | Iustin Pop | mnode <- fromObj "reserved_memory" a |
88 | 262f3e6c | Iustin Pop | mfree <- fromObj "free_memory" a |
89 | 262f3e6c | Iustin Pop | dtotal <- fromObj "total_disk" a |
90 | 262f3e6c | Iustin Pop | dfree <- fromObj "free_disk" a |
91 | 262f3e6c | Iustin Pop | ctotal <- fromObj "total_cpus" a |
92 | 262f3e6c | Iustin Pop | return $ Node.create n mtotal mnode mfree |
93 | 262f3e6c | Iustin Pop | dtotal dfree ctotal False) |
94 | 262f3e6c | Iustin Pop | return (n, node) |
95 | 144f190b | Iustin Pop | |
96 | 9188aeef | Iustin Pop | -- | Top-level parser. |
97 | 9188aeef | Iustin Pop | parseData :: String -- ^ The JSON message as received from Ganeti |
98 | 9188aeef | Iustin Pop | -> Result Request -- ^ A (possible valid) request |
99 | e4c5beaf | Iustin Pop | parseData body = do |
100 | c96d44df | Iustin Pop | decoded <- fromJResult "Parsing input IAllocator message" (decodeStrict body) |
101 | 262f3e6c | Iustin Pop | let obj = fromJSObject decoded |
102 | e4c5beaf | Iustin Pop | -- request parser |
103 | 262f3e6c | Iustin Pop | request <- liftM fromJSObject (fromObj "request" obj) |
104 | e4c5beaf | Iustin Pop | -- existing node parsing |
105 | 262f3e6c | Iustin Pop | nlist <- liftM fromJSObject (fromObj "nodes" obj) |
106 | 262f3e6c | Iustin Pop | nobj <- mapM (\(x,y) -> asJSObject y >>= parseNode x . fromJSObject) nlist |
107 | 497e30a1 | Iustin Pop | let (ktn, nl) = assignIndices nobj |
108 | e4c5beaf | Iustin Pop | -- existing instance parsing |
109 | e4c5beaf | Iustin Pop | ilist <- fromObj "instances" obj |
110 | e4c5beaf | Iustin Pop | let idata = fromJSObject ilist |
111 | 262f3e6c | Iustin Pop | iobj <- mapM (\(x,y) -> |
112 | 262f3e6c | Iustin Pop | asJSObject y >>= parseInstance ktn x . fromJSObject) idata |
113 | 497e30a1 | Iustin Pop | let (kti, il) = assignIndices iobj |
114 | 669ea132 | Iustin Pop | -- cluster tags |
115 | 669ea132 | Iustin Pop | ctags <- fromObj "cluster_tags" obj |
116 | 3e4480e0 | Iustin Pop | (map_n, map_i, ptags) <- mergeData [] [] [] (nl, il, ctags) |
117 | e4c5beaf | Iustin Pop | optype <- fromObj "type" request |
118 | e4c5beaf | Iustin Pop | rqtype <- |
119 | e4c5beaf | Iustin Pop | case optype of |
120 | e4c5beaf | Iustin Pop | "allocate" -> |
121 | e4c5beaf | Iustin Pop | do |
122 | 20c891d0 | Iustin Pop | rname <- fromObj "name" request |
123 | 20c891d0 | Iustin Pop | req_nodes <- fromObj "required_nodes" request |
124 | e4c5beaf | Iustin Pop | inew <- parseBaseInstance rname request |
125 | ed41c179 | Iustin Pop | let io = snd inew |
126 | ed41c179 | Iustin Pop | return $ Allocate io req_nodes |
127 | e4c5beaf | Iustin Pop | "relocate" -> |
128 | e4c5beaf | Iustin Pop | do |
129 | 20c891d0 | Iustin Pop | rname <- fromObj "name" request |
130 | 5a1edeb6 | Iustin Pop | ridx <- lookupInstance kti rname |
131 | 20c891d0 | Iustin Pop | req_nodes <- fromObj "required_nodes" request |
132 | ed41c179 | Iustin Pop | ex_nodes <- fromObj "relocate_from" request |
133 | 3e4480e0 | Iustin Pop | ex_idex <- mapM (Container.findByName map_n) ex_nodes |
134 | e7724ccc | Iustin Pop | return $ Relocate ridx req_nodes (map Node.idx ex_idex) |
135 | 54365762 | Iustin Pop | "multi-evacuate" -> |
136 | 54365762 | Iustin Pop | do |
137 | 54365762 | Iustin Pop | ex_names <- fromObj "evac_nodes" request |
138 | 3e4480e0 | Iustin Pop | ex_nodes <- mapM (Container.findByName map_n) ex_names |
139 | 54365762 | Iustin Pop | let ex_ndx = map Node.idx ex_nodes |
140 | 54365762 | Iustin Pop | return $ Evacuate ex_ndx |
141 | 9f6dcdea | Iustin Pop | other -> fail ("Invalid request type '" ++ other ++ "'") |
142 | 3e4480e0 | Iustin Pop | return $ Request rqtype map_n map_i ptags |
143 | 942403e6 | Iustin Pop | |
144 | e41f4ba0 | Iustin Pop | -- | Format the result |
145 | 3e4480e0 | Iustin Pop | formatRVal :: RqType -> [Node.AllocElement] -> JSValue |
146 | 3e4480e0 | Iustin Pop | formatRVal _ [] = JSArray [] |
147 | e41f4ba0 | Iustin Pop | |
148 | 3e4480e0 | Iustin Pop | formatRVal (Evacuate _) elems = |
149 | 3e4480e0 | Iustin Pop | let sols = map (\(_, inst, nl) -> Instance.name inst : map Node.name nl) |
150 | 3e4480e0 | Iustin Pop | elems |
151 | 54365762 | Iustin Pop | jsols = map (JSArray . map (JSString . toJSString)) sols |
152 | 54365762 | Iustin Pop | in JSArray jsols |
153 | 54365762 | Iustin Pop | |
154 | 3e4480e0 | Iustin Pop | formatRVal _ elems = |
155 | 54365762 | Iustin Pop | let (_, _, nodes) = head elems |
156 | 3e4480e0 | Iustin Pop | nodes' = map Node.name nodes |
157 | 54365762 | Iustin Pop | in JSArray $ map (JSString . toJSString) nodes' |
158 | 54365762 | Iustin Pop | |
159 | 9188aeef | Iustin Pop | -- | Formats the response into a valid IAllocator response message. |
160 | 9188aeef | Iustin Pop | formatResponse :: Bool -- ^ Whether the request was successful |
161 | 9188aeef | Iustin Pop | -> String -- ^ Information text |
162 | 54365762 | Iustin Pop | -> RqType -- ^ Request type |
163 | 54365762 | Iustin Pop | -> [Node.AllocElement] -- ^ The resulting allocations |
164 | 9188aeef | Iustin Pop | -> String -- ^ The JSON-formatted message |
165 | 3e4480e0 | Iustin Pop | formatResponse success info rq elems = |
166 | 43643696 | Iustin Pop | let |
167 | 43643696 | Iustin Pop | e_success = ("success", JSBool success) |
168 | 43643696 | Iustin Pop | e_info = ("info", JSString . toJSString $ info) |
169 | 3e4480e0 | Iustin Pop | e_nodes = ("nodes", formatRVal rq elems) |
170 | 43643696 | Iustin Pop | in encodeStrict $ makeObj [e_success, e_info, e_nodes] |