1 {-| Module describing a node.
3 All updates are functional (copy-based) and return a new node with
9 Copyright (C) 2009 Google Inc.
11 This program is free software; you can redistribute it and/or modify
12 it under the terms of the GNU General Public License as published by
13 the Free Software Foundation; either version 2 of the License, or
14 (at your option) any later version.
16 This program is distributed in the hope that it will be useful, but
17 WITHOUT ANY WARRANTY; without even the implied warranty of
18 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
19 General Public License for more details.
21 You should have received a copy of the GNU General Public License
22 along with this program; if not, write to the Free Software
23 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
28 module Ganeti.HTools.Node
29 ( Node(failN1, name, idx, t_mem, n_mem, f_mem, r_mem,
32 p_mem, p_dsk, p_rem, p_cpu,
34 plist, slist, offline)
38 -- ** Finalization after data loading
50 -- * Instance (re)location
63 import Text.Printf (printf)
65 import qualified Ganeti.HTools.Container as Container
66 import qualified Ganeti.HTools.Instance as Instance
67 import qualified Ganeti.HTools.PeerMap as PeerMap
69 import qualified Ganeti.HTools.Types as T
71 -- * Type declarations
74 data Node = Node { name :: String -- ^ The node name
75 , t_mem :: Double -- ^ Total memory (MiB)
76 , n_mem :: Int -- ^ Node memory (MiB)
77 , f_mem :: Int -- ^ Free memory (MiB)
78 , x_mem :: Int -- ^ Unaccounted memory (MiB)
79 , t_dsk :: Double -- ^ Total disk space (MiB)
80 , f_dsk :: Int -- ^ Free disk space (MiB)
81 , t_cpu :: Double -- ^ Total CPU count
82 , u_cpu :: Int -- ^ Used VCPU count
83 , plist :: [T.Idx]-- ^ List of primary instance indices
84 , slist :: [T.Idx]-- ^ List of secondary instance indices
85 , idx :: T.Ndx -- ^ Internal index for book-keeping
86 , peers :: PeerMap.PeerMap -- ^ Pnode to instance mapping
87 , failN1:: Bool -- ^ Whether the node has failed n1
88 , r_mem :: Int -- ^ Maximum memory needed for
89 -- failover by primaries of this node
90 , p_mem :: Double -- ^ Percent of free memory
91 , p_dsk :: Double -- ^ Percent of free disk
92 , p_rem :: Double -- ^ Percent of reserved memory
93 , p_cpu :: Double -- ^ Ratio of virtual to physical CPUs
94 , m_dsk :: Double -- ^ Minimum free disk ratio
95 , m_cpu :: Double -- ^ Max ratio of virt-to-phys CPUs
96 , offline :: Bool -- ^ Whether the node should not be used
97 -- for allocations and skipped from
101 instance T.Element Node where
107 -- | A simple name for the int, node association list.
108 type AssocList = [(T.Ndx, Node)]
110 -- | A simple name for a node map.
111 type List = Container.Container Node
113 -- | Constant node index for a non-moveable instance.
121 -- * Initialization functions
123 -- | Create a new node.
125 -- The index and the peers maps are empty, and will be need to be
126 -- update later via the 'setIdx' and 'buildPeers' functions.
127 create :: String -> Double -> Int -> Int -> Double
128 -> Int -> Double -> Bool -> Node
129 create name_init mem_t_init mem_n_init mem_f_init
130 dsk_t_init dsk_f_init cpu_t_init offline_init =
145 peers = PeerMap.empty,
147 p_mem = (fromIntegral mem_f_init) / mem_t_init,
148 p_dsk = (fromIntegral dsk_f_init) / dsk_t_init,
151 offline = offline_init,
157 -- | Changes the index.
159 -- This is used only during the building of the data structures.
160 setIdx :: Node -> T.Ndx -> Node
161 setIdx t i = t {idx = i}
163 -- | Changes the name.
165 -- This is used only during the building of the data structures.
166 setName :: Node -> String -> Node
167 setName t s = t {name = s}
169 -- | Sets the offline attribute.
170 setOffline :: Node -> Bool -> Node
171 setOffline t val = t { offline = val }
173 -- | Sets the unnaccounted memory.
174 setXmem :: Node -> Int -> Node
175 setXmem t val = t { x_mem = val }
177 -- | Sets the max disk usage ratio
178 setMdsk :: Node -> Double -> Node
179 setMdsk t val = t { m_dsk = val }
181 -- | Sets the max cpu usage ratio
182 setMcpu :: Node -> Double -> Node
183 setMcpu t val = t { m_cpu = val }
185 -- | Computes the maximum reserved memory for peers from a peer map.
186 computeMaxRes :: PeerMap.PeerMap -> PeerMap.Elem
187 computeMaxRes new_peers = PeerMap.maxElem new_peers
189 -- | Builds the peer map for a given node.
190 buildPeers :: Node -> Instance.List -> Node
193 (\i_idx -> let inst = Container.find i_idx il
194 in (Instance.pnode inst, Instance.mem inst))
196 pmap = PeerMap.accumArray (+) mdata
197 new_rmem = computeMaxRes pmap
198 new_failN1 = computeFailN1 new_rmem (f_mem t) (f_dsk t)
199 new_prem = (fromIntegral new_rmem) / (t_mem t)
200 in t {peers=pmap, failN1 = new_failN1, r_mem = new_rmem, p_rem = new_prem}
202 -- | Assigns an instance to a node as primary without other updates.
203 setPri :: Node -> T.Idx -> Node
204 setPri t idx = t { plist = idx:(plist t) }
206 -- | Assigns an instance to a node as secondary without other updates.
207 setSec :: Node -> T.Idx -> Node
208 setSec t idx = t { slist = idx:(slist t) }
210 -- | Add primary cpus to a node
211 addCpus :: Node -> Int -> Node
213 let new_count = (u_cpu t) + count
214 in t { u_cpu = new_count, p_cpu = (fromIntegral new_count) / (t_cpu t) }
216 -- * Update functions
218 -- | Sets the free memory.
219 setFmem :: Node -> Int -> Node
221 let new_n1 = computeFailN1 (r_mem t) new_mem (f_dsk t)
222 new_mp = (fromIntegral new_mem) / (t_mem t)
224 t { f_mem = new_mem, failN1 = new_n1, p_mem = new_mp }
226 -- | Given the rmem, free memory and disk, computes the failn1 status.
227 computeFailN1 :: Int -> Int -> Int -> Bool
228 computeFailN1 new_rmem new_mem new_dsk =
229 new_mem <= new_rmem || new_dsk <= 0
231 -- | Given the new free memory and disk, fail if any of them is below zero.
232 failHealth :: Int -> Int -> Bool
233 failHealth new_mem new_dsk = new_mem <= 0 || new_dsk <= 0
235 -- | Given new limits, check if any of them are overtaken
236 failLimits :: Node -> Double -> Double -> Bool
237 failLimits t new_dsk new_cpu =
240 in (l_dsk > new_dsk) || (l_cpu >= 0 && l_cpu < new_cpu)
242 -- | Removes a primary instance.
243 removePri :: Node -> Instance.Instance -> Node
245 let iname = Instance.idx inst
246 new_plist = delete iname (plist t)
247 new_mem = f_mem t + Instance.mem inst
248 new_dsk = f_dsk t + Instance.dsk inst
249 new_mp = (fromIntegral new_mem) / (t_mem t)
250 new_dp = (fromIntegral new_dsk) / (t_dsk t)
251 new_failn1 = computeFailN1 (r_mem t) new_mem new_dsk
252 new_ucpu = (u_cpu t) - (Instance.vcpus inst)
253 new_rcpu = (fromIntegral new_ucpu) / (t_cpu t)
254 in t {plist = new_plist, f_mem = new_mem, f_dsk = new_dsk,
255 failN1 = new_failn1, p_mem = new_mp, p_dsk = new_dp,
256 u_cpu = new_ucpu, p_cpu = new_rcpu}
258 -- | Removes a secondary instance.
259 removeSec :: Node -> Instance.Instance -> Node
261 let iname = Instance.idx inst
262 pnode = Instance.pnode inst
263 new_slist = delete iname (slist t)
264 new_dsk = f_dsk t + Instance.dsk inst
266 old_peem = PeerMap.find pnode old_peers
267 new_peem = old_peem - (Instance.mem inst)
268 new_peers = PeerMap.add pnode new_peem old_peers
270 new_rmem = if old_peem < old_rmem then
273 computeMaxRes new_peers
274 new_prem = (fromIntegral new_rmem) / (t_mem t)
275 new_failn1 = computeFailN1 new_rmem (f_mem t) new_dsk
276 new_dp = (fromIntegral new_dsk) / (t_dsk t)
277 in t {slist = new_slist, f_dsk = new_dsk, peers = new_peers,
278 failN1 = new_failn1, r_mem = new_rmem, p_dsk = new_dp,
281 -- | Adds a primary instance.
282 addPri :: Node -> Instance.Instance -> Maybe Node
284 let iname = Instance.idx inst
285 new_mem = f_mem t - Instance.mem inst
286 new_dsk = f_dsk t - Instance.dsk inst
287 new_failn1 = computeFailN1 (r_mem t) new_mem new_dsk
288 new_ucpu = (u_cpu t) + (Instance.vcpus inst)
289 new_pcpu = (fromIntegral new_ucpu) / (t_cpu t)
290 new_dp = (fromIntegral new_dsk) / (t_dsk t)
292 if (failHealth new_mem new_dsk) || (new_failn1 && not (failN1 t)) ||
293 (failLimits t new_dp new_pcpu)
297 let new_plist = iname:(plist t)
298 new_mp = (fromIntegral new_mem) / (t_mem t)
300 Just t {plist = new_plist, f_mem = new_mem, f_dsk = new_dsk,
301 failN1 = new_failn1, p_mem = new_mp, p_dsk = new_dp,
302 u_cpu = new_ucpu, p_cpu = new_pcpu}
304 -- | Adds a secondary instance.
305 addSec :: Node -> Instance.Instance -> T.Ndx -> Maybe Node
307 let iname = Instance.idx inst
310 new_dsk = f_dsk t - Instance.dsk inst
311 new_peem = PeerMap.find pdx old_peers + Instance.mem inst
312 new_peers = PeerMap.add pdx new_peem old_peers
313 new_rmem = max (r_mem t) new_peem
314 new_prem = (fromIntegral new_rmem) / (t_mem t)
315 new_failn1 = computeFailN1 new_rmem old_mem new_dsk
316 new_dp = (fromIntegral new_dsk) / (t_dsk t)
317 in if (failHealth old_mem new_dsk) || (new_failn1 && not (failN1 t)) ||
318 (failLimits t new_dp noLimit)
322 let new_slist = iname:(slist t)
324 Just t {slist = new_slist, f_dsk = new_dsk,
325 peers = new_peers, failN1 = new_failn1,
326 r_mem = new_rmem, p_dsk = new_dp,
329 -- * Display functions
331 -- | String converter for the node list functionality.
332 list :: Int -> Node -> String
334 let pl = length $ plist t
335 sl = length $ slist t
345 imem = (truncate tmem) - nmem - xmem - fmem
348 then printf " - %-*s %57s %3d %3d"
349 mname (name t) "" pl sl
351 printf " %c %-*s %5.0f %5d %5d %5d %5d %5d %5.0f %5d\
352 \ %4.0f %4d %3d %3d %6.4f %6.4f %5.2f"
353 (if off then '-' else if fn then '*' else ' ')
354 mname (name t) tmem nmem imem xmem fmem (r_mem t)
355 ((t_dsk t) / 1024) ((f_dsk t) `div` 1024)