Statistics
| Branch: | Tag: | Revision:

root / scripts / gnt-instance @ 312ac745

History | View | Annotate | Download (23.2 kB)

1 a8083063 Iustin Pop
#!/usr/bin/python
2 a8083063 Iustin Pop
#
3 a8083063 Iustin Pop
4 a8083063 Iustin Pop
# Copyright (C) 2006, 2007 Google Inc.
5 a8083063 Iustin Pop
#
6 a8083063 Iustin Pop
# This program is free software; you can redistribute it and/or modify
7 a8083063 Iustin Pop
# it under the terms of the GNU General Public License as published by
8 a8083063 Iustin Pop
# the Free Software Foundation; either version 2 of the License, or
9 a8083063 Iustin Pop
# (at your option) any later version.
10 a8083063 Iustin Pop
#
11 a8083063 Iustin Pop
# This program is distributed in the hope that it will be useful, but
12 a8083063 Iustin Pop
# WITHOUT ANY WARRANTY; without even the implied warranty of
13 a8083063 Iustin Pop
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14 a8083063 Iustin Pop
# General Public License for more details.
15 a8083063 Iustin Pop
#
16 a8083063 Iustin Pop
# You should have received a copy of the GNU General Public License
17 a8083063 Iustin Pop
# along with this program; if not, write to the Free Software
18 a8083063 Iustin Pop
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
19 a8083063 Iustin Pop
# 02110-1301, USA.
20 a8083063 Iustin Pop
21 a8083063 Iustin Pop
22 a8083063 Iustin Pop
import sys
23 a8083063 Iustin Pop
import os
24 312ac745 Iustin Pop
import itertools
25 a8083063 Iustin Pop
from optparse import make_option
26 a8083063 Iustin Pop
from cStringIO import StringIO
27 a8083063 Iustin Pop
28 a8083063 Iustin Pop
from ganeti.cli import *
29 a8083063 Iustin Pop
from ganeti import opcodes
30 a8083063 Iustin Pop
from ganeti import logger
31 a8083063 Iustin Pop
from ganeti import constants
32 a8083063 Iustin Pop
from ganeti import utils
33 312ac745 Iustin Pop
from ganeti import errors
34 312ac745 Iustin Pop
35 312ac745 Iustin Pop
36 312ac745 Iustin Pop
_SHUTDOWN_CLUSTER = "cluster"
37 312ac745 Iustin Pop
_SHUTDOWN_NODES_BOTH = "nodes"
38 312ac745 Iustin Pop
_SHUTDOWN_NODES_PRI = "nodes-pri"
39 312ac745 Iustin Pop
_SHUTDOWN_NODES_SEC = "nodes-sec"
40 312ac745 Iustin Pop
_SHUTDOWN_INSTANCES = "instances"
41 312ac745 Iustin Pop
42 312ac745 Iustin Pop
def _ExpandMultiNames(mode, names):
43 312ac745 Iustin Pop
  """Expand the given names using the passed mode.
44 312ac745 Iustin Pop
45 312ac745 Iustin Pop
  Args:
46 312ac745 Iustin Pop
    - mode, which can be one of _SHUTDOWN_CLUSTER, _SHUTDOWN_NODES_BOTH,
47 312ac745 Iustin Pop
      _SHUTDOWN_NODES_PRI, _SHUTDOWN_NODES_SEC or _SHUTDOWN_INSTANCES
48 312ac745 Iustin Pop
    - names, which is a list of names; for cluster, it must be empty,
49 312ac745 Iustin Pop
      and for node and instance it must be a list of valid item
50 312ac745 Iustin Pop
      names (short names are valid as usual, e.g. node1 instead of
51 312ac745 Iustin Pop
      node1.example.com)
52 312ac745 Iustin Pop
53 312ac745 Iustin Pop
  For _SHUTDOWN_CLUSTER, all instances will be returned. For
54 312ac745 Iustin Pop
  _SHUTDOWN_NODES_PRI/SEC, all instances having those nodes as
55 312ac745 Iustin Pop
  primary/secondary will be shutdown. For _SHUTDOWN_NODES_BOTH, all
56 312ac745 Iustin Pop
  instances having those nodes as either primary or secondary will be
57 312ac745 Iustin Pop
  returned. For _SHUTDOWN_INSTANCES, the given instances will be
58 312ac745 Iustin Pop
  returned.
59 312ac745 Iustin Pop
60 312ac745 Iustin Pop
  """
61 312ac745 Iustin Pop
  if mode == _SHUTDOWN_CLUSTER:
62 312ac745 Iustin Pop
    if names:
63 312ac745 Iustin Pop
      raise errors.OpPrereqError("Cluster filter mode takes no arguments")
64 312ac745 Iustin Pop
    op = opcodes.OpQueryInstances(output_fields=["name"], names=[])
65 312ac745 Iustin Pop
    idata = SubmitOpCode(op)
66 312ac745 Iustin Pop
    inames = [row[0] for row in idata]
67 312ac745 Iustin Pop
68 312ac745 Iustin Pop
  elif mode in (_SHUTDOWN_NODES_BOTH,
69 312ac745 Iustin Pop
                _SHUTDOWN_NODES_PRI,
70 312ac745 Iustin Pop
                _SHUTDOWN_NODES_SEC):
71 312ac745 Iustin Pop
    if not names:
72 312ac745 Iustin Pop
      raise errors.OpPrereqError("No node names passed")
73 312ac745 Iustin Pop
    op = opcodes.OpQueryNodes(output_fields=["name", "pinst_list",
74 312ac745 Iustin Pop
                                             "sinst_list"], names=names)
75 312ac745 Iustin Pop
    ndata = SubmitOpCode(op)
76 312ac745 Iustin Pop
    ipri = [row[1] for row in ndata]
77 312ac745 Iustin Pop
    pri_names = list(itertools.chain(*ipri))
78 312ac745 Iustin Pop
    isec = [row[2] for row in ndata]
79 312ac745 Iustin Pop
    sec_names = list(itertools.chain(*isec))
80 312ac745 Iustin Pop
    if mode == _SHUTDOWN_NODES_BOTH:
81 312ac745 Iustin Pop
      inames = pri_names + sec_names
82 312ac745 Iustin Pop
    elif mode == _SHUTDOWN_NODES_PRI:
83 312ac745 Iustin Pop
      inames = pri_names
84 312ac745 Iustin Pop
    elif mode == _SHUTDOWN_NODES_SEC:
85 312ac745 Iustin Pop
      inames = sec_names
86 312ac745 Iustin Pop
    else:
87 312ac745 Iustin Pop
      raise errors.ProgrammerError("Unhandled shutdown type")
88 312ac745 Iustin Pop
89 312ac745 Iustin Pop
  elif mode == _SHUTDOWN_INSTANCES:
90 312ac745 Iustin Pop
    if not names:
91 312ac745 Iustin Pop
      raise errors.OpPrereqError("No instance names passed")
92 312ac745 Iustin Pop
    op = opcodes.OpQueryInstances(output_fields=["name"], names=names)
93 312ac745 Iustin Pop
    idata = SubmitOpCode(op)
94 312ac745 Iustin Pop
    inames = [row[0] for row in idata]
95 312ac745 Iustin Pop
96 312ac745 Iustin Pop
  else:
97 312ac745 Iustin Pop
    raise errors.OpPrereqError("Unknown mode '%s'" % mode)
98 312ac745 Iustin Pop
99 312ac745 Iustin Pop
  return inames
100 a8083063 Iustin Pop
101 a8083063 Iustin Pop
102 a8083063 Iustin Pop
def ListInstances(opts, args):
103 a8083063 Iustin Pop
  """List nodes and their properties.
104 a8083063 Iustin Pop
105 a8083063 Iustin Pop
  """
106 a8083063 Iustin Pop
  if opts.output is None:
107 a8083063 Iustin Pop
    selected_fields = ["name", "os", "pnode", "admin_state",
108 a8083063 Iustin Pop
                       "oper_state", "oper_ram"]
109 a8083063 Iustin Pop
  else:
110 a8083063 Iustin Pop
    selected_fields = opts.output.split(",")
111 a8083063 Iustin Pop
112 069dcc86 Iustin Pop
  op = opcodes.OpQueryInstances(output_fields=selected_fields, names=[])
113 a8083063 Iustin Pop
  output = SubmitOpCode(op)
114 a8083063 Iustin Pop
115 a8083063 Iustin Pop
  if not opts.no_headers:
116 137161c9 Michael Hanselmann
    headers = {"name": "Instance", "os": "OS", "pnode": "Primary_node",
117 137161c9 Michael Hanselmann
               "snodes": "Secondary_Nodes", "admin_state": "Autostart",
118 137161c9 Michael Hanselmann
               "oper_state": "Status", "admin_ram": "Configured_memory",
119 137161c9 Michael Hanselmann
               "oper_ram": "Memory", "disk_template": "Disk_template",
120 137161c9 Michael Hanselmann
               "ip": "IP Address", "mac": "MAC Address",
121 644eeef9 Iustin Pop
               "bridge": "Bridge",
122 644eeef9 Iustin Pop
               "sda_size": "Disk/0", "sdb_size": "Disk/1"}
123 137161c9 Michael Hanselmann
  else:
124 137161c9 Michael Hanselmann
    headers = None
125 137161c9 Michael Hanselmann
126 137161c9 Michael Hanselmann
  if opts.human_readable:
127 644eeef9 Iustin Pop
    unitfields = ["admin_ram", "oper_ram", "sda_size", "sdb_size"]
128 137161c9 Michael Hanselmann
  else:
129 137161c9 Michael Hanselmann
    unitfields = None
130 137161c9 Michael Hanselmann
131 644eeef9 Iustin Pop
  numfields = ["admin_ram", "oper_ram", "sda_size", "sdb_size"]
132 137161c9 Michael Hanselmann
133 8a23d2d3 Iustin Pop
  # change raw values to nicer strings
134 8a23d2d3 Iustin Pop
  for row in output:
135 8a23d2d3 Iustin Pop
    for idx, field in enumerate(selected_fields):
136 8a23d2d3 Iustin Pop
      val = row[idx]
137 8a23d2d3 Iustin Pop
      if field == "snodes":
138 8a23d2d3 Iustin Pop
        val = ",".join(val) or "-"
139 8a23d2d3 Iustin Pop
      elif field == "admin_state":
140 8a23d2d3 Iustin Pop
        if val:
141 8a23d2d3 Iustin Pop
          val = "yes"
142 8a23d2d3 Iustin Pop
        else:
143 8a23d2d3 Iustin Pop
          val = "no"
144 8a23d2d3 Iustin Pop
      elif field == "oper_state":
145 8a23d2d3 Iustin Pop
        if val is None:
146 8a23d2d3 Iustin Pop
          val = "(node down)"
147 8a23d2d3 Iustin Pop
        elif val: # True
148 8a23d2d3 Iustin Pop
          val = "running"
149 8a23d2d3 Iustin Pop
        else:
150 8a23d2d3 Iustin Pop
          val = "stopped"
151 8a23d2d3 Iustin Pop
      elif field == "oper_ram":
152 8a23d2d3 Iustin Pop
        if val is None:
153 8a23d2d3 Iustin Pop
          val = "(node down)"
154 8a23d2d3 Iustin Pop
      elif field == "sda_size" or field == "sdb_size":
155 8a23d2d3 Iustin Pop
        if val is None:
156 8a23d2d3 Iustin Pop
          val = "N/A"
157 8a23d2d3 Iustin Pop
      row[idx] = str(val)
158 8a23d2d3 Iustin Pop
159 16be8703 Iustin Pop
  data = GenerateTable(separator=opts.separator, headers=headers,
160 16be8703 Iustin Pop
                       fields=selected_fields, unitfields=unitfields,
161 16be8703 Iustin Pop
                       numfields=numfields, data=output)
162 16be8703 Iustin Pop
163 16be8703 Iustin Pop
  for line in data:
164 16be8703 Iustin Pop
    logger.ToStdout(line)
165 a8083063 Iustin Pop
166 a8083063 Iustin Pop
  return 0
167 a8083063 Iustin Pop
168 a8083063 Iustin Pop
169 a8083063 Iustin Pop
def AddInstance(opts, args):
170 a8083063 Iustin Pop
  """Add an instance to the cluster.
171 a8083063 Iustin Pop
172 a8083063 Iustin Pop
  Args:
173 a8083063 Iustin Pop
    opts - class with options as members
174 a8083063 Iustin Pop
    args - list with a single element, the instance name
175 a8083063 Iustin Pop
  Opts used:
176 a8083063 Iustin Pop
    mem - amount of memory to allocate to instance (MiB)
177 a8083063 Iustin Pop
    size - amount of disk space to allocate to instance (MiB)
178 a8083063 Iustin Pop
    os - which OS to run on instance
179 a8083063 Iustin Pop
    node - node to run new instance on
180 a8083063 Iustin Pop
181 a8083063 Iustin Pop
  """
182 a8083063 Iustin Pop
  instance = args[0]
183 a8083063 Iustin Pop
184 a8083063 Iustin Pop
  op = opcodes.OpCreateInstance(instance_name=instance, mem_size=opts.mem,
185 a8083063 Iustin Pop
                                disk_size=opts.size, swap_size=opts.swap,
186 a8083063 Iustin Pop
                                disk_template=opts.disk_template,
187 a8083063 Iustin Pop
                                mode=constants.INSTANCE_CREATE,
188 a8083063 Iustin Pop
                                os_type=opts.os, pnode=opts.node,
189 a8083063 Iustin Pop
                                snode=opts.snode, vcpus=opts.vcpus,
190 a8083063 Iustin Pop
                                ip=opts.ip, bridge=opts.bridge, start=True,
191 a8083063 Iustin Pop
                                wait_for_sync=opts.wait_for_sync)
192 a8083063 Iustin Pop
  SubmitOpCode(op)
193 a8083063 Iustin Pop
  return 0
194 a8083063 Iustin Pop
195 a8083063 Iustin Pop
196 fe7b0351 Michael Hanselmann
def ReinstallInstance(opts, args):
197 fe7b0351 Michael Hanselmann
  """Reinstall an instance.
198 fe7b0351 Michael Hanselmann
199 fe7b0351 Michael Hanselmann
  Args:
200 fe7b0351 Michael Hanselmann
    opts - class with options as members
201 fe7b0351 Michael Hanselmann
    args - list containing a single element, the instance name
202 fe7b0351 Michael Hanselmann
203 fe7b0351 Michael Hanselmann
  """
204 fe7b0351 Michael Hanselmann
  instance_name = args[0]
205 fe7b0351 Michael Hanselmann
206 fe7b0351 Michael Hanselmann
  if not opts.force:
207 fe7b0351 Michael Hanselmann
    usertext = ("This will reinstall the instance %s and remove "
208 fe7b0351 Michael Hanselmann
                "all data. Continue?") % instance_name
209 fe7b0351 Michael Hanselmann
    if not opts._ask_user(usertext):
210 fe7b0351 Michael Hanselmann
      return 1
211 fe7b0351 Michael Hanselmann
212 d0834de3 Michael Hanselmann
  op = opcodes.OpReinstallInstance(instance_name=instance_name,
213 d0834de3 Michael Hanselmann
                                   os_type=opts.os)
214 fe7b0351 Michael Hanselmann
  SubmitOpCode(op)
215 fe7b0351 Michael Hanselmann
216 fe7b0351 Michael Hanselmann
  return 0
217 fe7b0351 Michael Hanselmann
218 fe7b0351 Michael Hanselmann
219 a8083063 Iustin Pop
def RemoveInstance(opts, args):
220 a8083063 Iustin Pop
  """Remove an instance.
221 a8083063 Iustin Pop
222 a8083063 Iustin Pop
  Args:
223 a8083063 Iustin Pop
    opts - class with options as members
224 a8083063 Iustin Pop
    args - list containing a single element, the instance name
225 a8083063 Iustin Pop
226 a8083063 Iustin Pop
  """
227 a8083063 Iustin Pop
  instance_name = args[0]
228 a8083063 Iustin Pop
  force = opts.force
229 a8083063 Iustin Pop
230 a8083063 Iustin Pop
  if not force:
231 a8083063 Iustin Pop
    usertext = ("This will remove the volumes of the instance %s"
232 a8083063 Iustin Pop
                " (including mirrors), thus removing all the data"
233 a8083063 Iustin Pop
                " of the instance. Continue?") % instance_name
234 a8083063 Iustin Pop
    if not opts._ask_user(usertext):
235 a8083063 Iustin Pop
      return 1
236 a8083063 Iustin Pop
237 a8083063 Iustin Pop
  op = opcodes.OpRemoveInstance(instance_name=instance_name)
238 a8083063 Iustin Pop
  SubmitOpCode(op)
239 a8083063 Iustin Pop
  return 0
240 a8083063 Iustin Pop
241 a8083063 Iustin Pop
242 a8083063 Iustin Pop
def ActivateDisks(opts, args):
243 a8083063 Iustin Pop
  """Activate an instance's disks.
244 a8083063 Iustin Pop
245 a8083063 Iustin Pop
  This serves two purposes:
246 a8083063 Iustin Pop
    - it allows one (as long as the instance is not running) to mount
247 a8083063 Iustin Pop
    the disks and modify them from the node
248 a8083063 Iustin Pop
    - it repairs inactive secondary drbds
249 a8083063 Iustin Pop
250 a8083063 Iustin Pop
  """
251 a8083063 Iustin Pop
  instance_name = args[0]
252 a8083063 Iustin Pop
  op = opcodes.OpActivateInstanceDisks(instance_name=instance_name)
253 a8083063 Iustin Pop
  disks_info = SubmitOpCode(op)
254 a8083063 Iustin Pop
  for host, iname, nname in disks_info:
255 a8083063 Iustin Pop
    print "%s:%s:%s" % (host, iname, nname)
256 a8083063 Iustin Pop
  return 0
257 a8083063 Iustin Pop
258 a8083063 Iustin Pop
259 a8083063 Iustin Pop
def DeactivateDisks(opts, args):
260 a8083063 Iustin Pop
  """Command-line interface for _ShutdownInstanceBlockDevices.
261 a8083063 Iustin Pop
262 a8083063 Iustin Pop
  This function takes the instance name, looks for its primary node
263 a8083063 Iustin Pop
  and the tries to shutdown its block devices on that node.
264 a8083063 Iustin Pop
265 a8083063 Iustin Pop
  """
266 a8083063 Iustin Pop
  instance_name = args[0]
267 a8083063 Iustin Pop
  op = opcodes.OpDeactivateInstanceDisks(instance_name=instance_name)
268 a8083063 Iustin Pop
  SubmitOpCode(op)
269 a8083063 Iustin Pop
  return 0
270 a8083063 Iustin Pop
271 a8083063 Iustin Pop
272 a8083063 Iustin Pop
def StartupInstance(opts, args):
273 a8083063 Iustin Pop
  """Shutdown an instance.
274 a8083063 Iustin Pop
275 a8083063 Iustin Pop
  Args:
276 a8083063 Iustin Pop
    opts - class with options as members
277 a8083063 Iustin Pop
    args - list containing a single element, the instance name
278 a8083063 Iustin Pop
279 a8083063 Iustin Pop
  """
280 312ac745 Iustin Pop
  if opts.multi_mode is None:
281 312ac745 Iustin Pop
    opts.multi_mode = _SHUTDOWN_INSTANCES
282 312ac745 Iustin Pop
  inames = _ExpandMultiNames(opts.multi_mode, args)
283 312ac745 Iustin Pop
  multi_on = len(inames) > 1
284 312ac745 Iustin Pop
  for name in inames:
285 312ac745 Iustin Pop
    op = opcodes.OpStartupInstance(instance_name=name,
286 312ac745 Iustin Pop
                                   force=opts.force,
287 312ac745 Iustin Pop
                                   extra_args=opts.extra_args)
288 312ac745 Iustin Pop
    if multi_on:
289 312ac745 Iustin Pop
      logger.ToStdout("Starting up %s" % name)
290 312ac745 Iustin Pop
    SubmitOpCode(op)
291 a8083063 Iustin Pop
  return 0
292 a8083063 Iustin Pop
293 a8083063 Iustin Pop
294 a8083063 Iustin Pop
def ShutdownInstance(opts, args):
295 a8083063 Iustin Pop
  """Shutdown an instance.
296 a8083063 Iustin Pop
297 a8083063 Iustin Pop
  Args:
298 a8083063 Iustin Pop
    opts - class with options as members
299 a8083063 Iustin Pop
    args - list containing a single element, the instance name
300 a8083063 Iustin Pop
301 a8083063 Iustin Pop
  """
302 312ac745 Iustin Pop
  if opts.multi_mode is None:
303 312ac745 Iustin Pop
    opts.multi_mode = _SHUTDOWN_INSTANCES
304 312ac745 Iustin Pop
  inames = _ExpandMultiNames(opts.multi_mode, args)
305 312ac745 Iustin Pop
  multi_on = len(inames) > 1
306 312ac745 Iustin Pop
  for name in inames:
307 312ac745 Iustin Pop
    op = opcodes.OpShutdownInstance(instance_name=name)
308 312ac745 Iustin Pop
    if multi_on:
309 312ac745 Iustin Pop
      logger.ToStdout("Shutting down %s" % name)
310 312ac745 Iustin Pop
    SubmitOpCode(op)
311 a8083063 Iustin Pop
  return 0
312 a8083063 Iustin Pop
313 a8083063 Iustin Pop
314 a8083063 Iustin Pop
def AddMDDRBDComponent(opts, args):
315 a8083063 Iustin Pop
  """Add a new component to a remote_raid1 disk.
316 a8083063 Iustin Pop
317 a8083063 Iustin Pop
  Args:
318 a8083063 Iustin Pop
    opts - class with options as members
319 a8083063 Iustin Pop
    args - list with a single element, the instance name
320 a8083063 Iustin Pop
321 a8083063 Iustin Pop
  """
322 a8083063 Iustin Pop
  op = opcodes.OpAddMDDRBDComponent(instance_name=args[0],
323 a8083063 Iustin Pop
                                    disk_name=opts.disk,
324 a8083063 Iustin Pop
                                    remote_node=opts.node)
325 a8083063 Iustin Pop
  SubmitOpCode(op)
326 a8083063 Iustin Pop
  return 0
327 a8083063 Iustin Pop
328 a8083063 Iustin Pop
329 a8083063 Iustin Pop
def RemoveMDDRBDComponent(opts, args):
330 bdb8c997 Guido Trotter
  """Remove a component from a remote_raid1 disk.
331 a8083063 Iustin Pop
332 a8083063 Iustin Pop
  Args:
333 a8083063 Iustin Pop
    opts - class with options as members
334 a8083063 Iustin Pop
    args - list with a single element, the instance name
335 a8083063 Iustin Pop
336 a8083063 Iustin Pop
  """
337 a8083063 Iustin Pop
  op = opcodes.OpRemoveMDDRBDComponent(instance_name=args[0],
338 a8083063 Iustin Pop
                                       disk_name=opts.disk,
339 a8083063 Iustin Pop
                                       disk_id=opts.port)
340 a8083063 Iustin Pop
  SubmitOpCode(op)
341 a8083063 Iustin Pop
  return 0
342 a8083063 Iustin Pop
343 a8083063 Iustin Pop
344 a8083063 Iustin Pop
def ReplaceDisks(opts, args):
345 a8083063 Iustin Pop
  """Replace the disks of an instance
346 a8083063 Iustin Pop
347 a8083063 Iustin Pop
  Args:
348 a8083063 Iustin Pop
    opts - class with options as members
349 a8083063 Iustin Pop
    args - list with a single element, the instance name
350 a8083063 Iustin Pop
351 a8083063 Iustin Pop
  """
352 a8083063 Iustin Pop
  instance_name = args[0]
353 a8083063 Iustin Pop
  new_secondary = opts.new_secondary
354 a8083063 Iustin Pop
  op = opcodes.OpReplaceDisks(instance_name=args[0],
355 a8083063 Iustin Pop
                              remote_node=opts.new_secondary)
356 a8083063 Iustin Pop
  SubmitOpCode(op)
357 a8083063 Iustin Pop
  return 0
358 a8083063 Iustin Pop
359 a8083063 Iustin Pop
360 a8083063 Iustin Pop
def FailoverInstance(opts, args):
361 a8083063 Iustin Pop
  """Failover an instance.
362 a8083063 Iustin Pop
363 a8083063 Iustin Pop
  The failover is done by shutting it down on its present node and
364 a8083063 Iustin Pop
  starting it on the secondary.
365 a8083063 Iustin Pop
366 a8083063 Iustin Pop
  Args:
367 a8083063 Iustin Pop
    opts - class with options as members
368 a8083063 Iustin Pop
    args - list with a single element, the instance name
369 a8083063 Iustin Pop
  Opts used:
370 a8083063 Iustin Pop
    force - whether to failover without asking questions.
371 a8083063 Iustin Pop
372 a8083063 Iustin Pop
  """
373 a8083063 Iustin Pop
  instance_name = args[0]
374 a8083063 Iustin Pop
  force = opts.force
375 a8083063 Iustin Pop
376 a8083063 Iustin Pop
  if not force:
377 a8083063 Iustin Pop
    usertext = ("Failover will happen to image %s."
378 a8083063 Iustin Pop
                " This requires a shutdown of the instance. Continue?" %
379 a8083063 Iustin Pop
                (instance_name,))
380 a8083063 Iustin Pop
    if not opts._ask_user(usertext):
381 a8083063 Iustin Pop
      return 1
382 a8083063 Iustin Pop
383 a8083063 Iustin Pop
  op = opcodes.OpFailoverInstance(instance_name=instance_name,
384 a8083063 Iustin Pop
                                  ignore_consistency=opts.ignore_consistency)
385 a8083063 Iustin Pop
  SubmitOpCode(op)
386 a8083063 Iustin Pop
  return 0
387 a8083063 Iustin Pop
388 a8083063 Iustin Pop
389 a8083063 Iustin Pop
def ConnectToInstanceConsole(opts, args):
390 a8083063 Iustin Pop
  """Connect to the console of an instance.
391 a8083063 Iustin Pop
392 a8083063 Iustin Pop
  Args:
393 a8083063 Iustin Pop
    opts - class with options as members
394 a8083063 Iustin Pop
    args - list with a single element, the instance name
395 a8083063 Iustin Pop
396 a8083063 Iustin Pop
  """
397 a8083063 Iustin Pop
  instance_name = args[0]
398 a8083063 Iustin Pop
399 a8083063 Iustin Pop
  op = opcodes.OpConnectConsole(instance_name=instance_name)
400 82122173 Iustin Pop
  cmd, argv = SubmitOpCode(op)
401 a8083063 Iustin Pop
  # drop lock and exec so other commands can run while we have console
402 a8083063 Iustin Pop
  utils.Unlock("cmd")
403 a8083063 Iustin Pop
  try:
404 82122173 Iustin Pop
    os.execvp(cmd, argv)
405 a8083063 Iustin Pop
  finally:
406 82122173 Iustin Pop
    sys.stderr.write("Can't run console command %s with arguments:\n'%s'" %
407 82122173 Iustin Pop
                     (cmd, " ".join(argv)))
408 a8083063 Iustin Pop
    os._exit(1)
409 a8083063 Iustin Pop
410 a8083063 Iustin Pop
411 a8083063 Iustin Pop
def _FormatBlockDevInfo(buf, dev, indent_level):
412 a8083063 Iustin Pop
  """Show block device information.
413 a8083063 Iustin Pop
414 a8083063 Iustin Pop
  This is only used by ShowInstanceConfig(), but it's too big to be
415 a8083063 Iustin Pop
  left for an inline definition.
416 a8083063 Iustin Pop
417 a8083063 Iustin Pop
  """
418 a8083063 Iustin Pop
  def helper(buf, dtype, status):
419 a8083063 Iustin Pop
    """Format one line for phsyical device status."""
420 a8083063 Iustin Pop
    if not status:
421 a8083063 Iustin Pop
      buf.write("not active\n")
422 a8083063 Iustin Pop
    else:
423 a8083063 Iustin Pop
      (path, major, minor, syncp, estt, degr) = status
424 a8083063 Iustin Pop
      buf.write("%s (%d:%d)" % (path, major, minor))
425 a8083063 Iustin Pop
      if dtype in ("md_raid1", "drbd"):
426 a8083063 Iustin Pop
        if syncp is not None:
427 a8083063 Iustin Pop
          sync_text = "*RECOVERING* %5.2f%%," % syncp
428 a8083063 Iustin Pop
          if estt:
429 a8083063 Iustin Pop
            sync_text += " ETA %ds" % estt
430 a8083063 Iustin Pop
          else:
431 a8083063 Iustin Pop
            sync_text += " ETA unknown"
432 a8083063 Iustin Pop
        else:
433 a8083063 Iustin Pop
          sync_text = "in sync"
434 a8083063 Iustin Pop
        if degr:
435 a8083063 Iustin Pop
          degr_text = "*DEGRADED*"
436 a8083063 Iustin Pop
        else:
437 a8083063 Iustin Pop
          degr_text = "ok"
438 a8083063 Iustin Pop
        buf.write(" %s, status %s" % (sync_text, degr_text))
439 a8083063 Iustin Pop
      buf.write("\n")
440 a8083063 Iustin Pop
441 a8083063 Iustin Pop
  if dev["iv_name"] is not None:
442 a8083063 Iustin Pop
    data = "  - %s, " % dev["iv_name"]
443 a8083063 Iustin Pop
  else:
444 a8083063 Iustin Pop
    data = "  - "
445 a8083063 Iustin Pop
  data += "type: %s" % dev["dev_type"]
446 a8083063 Iustin Pop
  if dev["logical_id"] is not None:
447 a8083063 Iustin Pop
    data += ", logical_id: %s" % (dev["logical_id"],)
448 a8083063 Iustin Pop
  elif dev["physical_id"] is not None:
449 a8083063 Iustin Pop
    data += ", physical_id: %s" % (dev["physical_id"],)
450 a8083063 Iustin Pop
  buf.write("%*s%s\n" % (2*indent_level, "", data))
451 a8083063 Iustin Pop
  buf.write("%*s    primary:   " % (2*indent_level, ""))
452 a8083063 Iustin Pop
  helper(buf, dev["dev_type"], dev["pstatus"])
453 a8083063 Iustin Pop
454 a8083063 Iustin Pop
  if dev["sstatus"]:
455 a8083063 Iustin Pop
    buf.write("%*s    secondary: " % (2*indent_level, ""))
456 a8083063 Iustin Pop
    helper(buf, dev["dev_type"], dev["sstatus"])
457 a8083063 Iustin Pop
458 a8083063 Iustin Pop
  if dev["children"]:
459 a8083063 Iustin Pop
    for child in dev["children"]:
460 a8083063 Iustin Pop
      _FormatBlockDevInfo(buf, child, indent_level+1)
461 a8083063 Iustin Pop
462 a8083063 Iustin Pop
463 a8083063 Iustin Pop
def ShowInstanceConfig(opts, args):
464 a8083063 Iustin Pop
  """Compute instance run-time status.
465 a8083063 Iustin Pop
466 a8083063 Iustin Pop
  """
467 a8083063 Iustin Pop
  retcode = 0
468 a8083063 Iustin Pop
  op = opcodes.OpQueryInstanceData(instances=args)
469 a8083063 Iustin Pop
  result = SubmitOpCode(op)
470 a8083063 Iustin Pop
471 a8083063 Iustin Pop
  if not result:
472 a8083063 Iustin Pop
    logger.ToStdout("No instances.")
473 a8083063 Iustin Pop
    return 1
474 a8083063 Iustin Pop
475 a8083063 Iustin Pop
  buf = StringIO()
476 a8083063 Iustin Pop
  retcode = 0
477 a8083063 Iustin Pop
  for instance_name in result:
478 a8083063 Iustin Pop
    instance = result[instance_name]
479 a8083063 Iustin Pop
    buf.write("Instance name: %s\n" % instance["name"])
480 a8083063 Iustin Pop
    buf.write("State: configured to be %s, actual state is %s\n" %
481 a8083063 Iustin Pop
              (instance["config_state"], instance["run_state"]))
482 a8083063 Iustin Pop
    buf.write("  Nodes:\n")
483 a8083063 Iustin Pop
    buf.write("    - primary: %s\n" % instance["pnode"])
484 a8083063 Iustin Pop
    buf.write("    - secondaries: %s\n" % ", ".join(instance["snodes"]))
485 a8083063 Iustin Pop
    buf.write("  Operating system: %s\n" % instance["os"])
486 a8083063 Iustin Pop
    buf.write("  Hardware:\n")
487 a8083063 Iustin Pop
    buf.write("    - memory: %dMiB\n" % instance["memory"])
488 a8083063 Iustin Pop
    buf.write("    - NICs: %s\n" %
489 a8083063 Iustin Pop
        ", ".join(["{MAC: %s, IP: %s, bridge: %s}" %
490 a8083063 Iustin Pop
                   (mac, ip, bridge)
491 a8083063 Iustin Pop
                     for mac, ip, bridge in instance["nics"]]))
492 a8083063 Iustin Pop
    buf.write("  Block devices:\n")
493 a8083063 Iustin Pop
494 a8083063 Iustin Pop
    for device in instance["disks"]:
495 a8083063 Iustin Pop
      _FormatBlockDevInfo(buf, device, 1)
496 a8083063 Iustin Pop
497 a8083063 Iustin Pop
  logger.ToStdout(buf.getvalue().rstrip('\n'))
498 a8083063 Iustin Pop
  return retcode
499 a8083063 Iustin Pop
500 a8083063 Iustin Pop
501 a8083063 Iustin Pop
def SetInstanceParms(opts, args):
502 a8083063 Iustin Pop
  """Modifies an instance.
503 a8083063 Iustin Pop
504 a8083063 Iustin Pop
  All parameters take effect only at the next restart of the instance.
505 a8083063 Iustin Pop
506 a8083063 Iustin Pop
  Args:
507 a8083063 Iustin Pop
    opts - class with options as members
508 a8083063 Iustin Pop
    args - list with a single element, the instance name
509 a8083063 Iustin Pop
  Opts used:
510 a8083063 Iustin Pop
    memory - the new memory size
511 a8083063 Iustin Pop
    vcpus - the new number of cpus
512 a8083063 Iustin Pop
513 a8083063 Iustin Pop
  """
514 a8083063 Iustin Pop
  if not opts.mem and not opts.vcpus and not opts.ip and not opts.bridge:
515 a8083063 Iustin Pop
    logger.ToStdout("Please give at least one of the parameters.")
516 a8083063 Iustin Pop
    return 1
517 a8083063 Iustin Pop
518 a8083063 Iustin Pop
  op = opcodes.OpSetInstanceParms(instance_name=args[0], mem=opts.mem,
519 a8083063 Iustin Pop
                                  vcpus=opts.vcpus, ip=opts.ip,
520 a8083063 Iustin Pop
                                  bridge=opts.bridge)
521 a8083063 Iustin Pop
  result = SubmitOpCode(op)
522 a8083063 Iustin Pop
523 a8083063 Iustin Pop
  if result:
524 a8083063 Iustin Pop
    logger.ToStdout("Modified instance %s" % args[0])
525 a8083063 Iustin Pop
    for param, data in result:
526 a8083063 Iustin Pop
      logger.ToStdout(" - %-5s -> %s" % (param, data))
527 a8083063 Iustin Pop
    logger.ToStdout("Please don't forget that these parameters take effect"
528 a8083063 Iustin Pop
                    " only at the next start of the instance.")
529 a8083063 Iustin Pop
  return 0
530 a8083063 Iustin Pop
531 a8083063 Iustin Pop
532 a8083063 Iustin Pop
# options used in more than one cmd
533 a8083063 Iustin Pop
node_opt = make_option("-n", "--node", dest="node", help="Target node",
534 a8083063 Iustin Pop
                       metavar="<node>")
535 a8083063 Iustin Pop
536 d0834de3 Michael Hanselmann
os_opt = cli_option("-o", "--os-type", dest="os", help="What OS to run",
537 739ecd56 Michael Hanselmann
                    metavar="<os>")
538 d0834de3 Michael Hanselmann
539 312ac745 Iustin Pop
# multi-instance selection options
540 312ac745 Iustin Pop
m_pri_node_opt = make_option("--primary", dest="multi_mode",
541 312ac745 Iustin Pop
                             help="Filter by nodes (primary only)",
542 312ac745 Iustin Pop
                             const=_SHUTDOWN_NODES_PRI, action="store_const")
543 312ac745 Iustin Pop
544 312ac745 Iustin Pop
m_sec_node_opt = make_option("--secondary", dest="multi_mode",
545 312ac745 Iustin Pop
                             help="Filter by nodes (secondary only)",
546 312ac745 Iustin Pop
                             const=_SHUTDOWN_NODES_SEC, action="store_const")
547 312ac745 Iustin Pop
548 312ac745 Iustin Pop
m_node_opt = make_option("--node", dest="multi_mode",
549 312ac745 Iustin Pop
                         help="Filter by nodes (primary and secondary)",
550 312ac745 Iustin Pop
                         const=_SHUTDOWN_NODES_BOTH, action="store_const")
551 312ac745 Iustin Pop
552 312ac745 Iustin Pop
m_clust_opt = make_option("--all", dest="multi_mode",
553 312ac745 Iustin Pop
                          help="Select all instances in the cluster",
554 312ac745 Iustin Pop
                          const=_SHUTDOWN_CLUSTER, action="store_const")
555 312ac745 Iustin Pop
556 312ac745 Iustin Pop
m_inst_opt = make_option("--instance", dest="multi_mode",
557 312ac745 Iustin Pop
                         help="Filter by instance name [default]",
558 312ac745 Iustin Pop
                         const=_SHUTDOWN_INSTANCES, action="store_const")
559 312ac745 Iustin Pop
560 312ac745 Iustin Pop
561 a8083063 Iustin Pop
# this is defined separately due to readability only
562 a8083063 Iustin Pop
add_opts = [
563 a8083063 Iustin Pop
  DEBUG_OPT,
564 a8083063 Iustin Pop
  node_opt,
565 b9ac33e9 Iustin Pop
  cli_option("-s", "--os-size", dest="size", help="Disk size, in MiB unless"
566 b9ac33e9 Iustin Pop
             " a suffix is used",
567 a8083063 Iustin Pop
             default=20 * 1024, type="unit", metavar="<size>"),
568 b9ac33e9 Iustin Pop
  cli_option("--swap-size", dest="swap", help="Swap size, in MiB unless a"
569 b9ac33e9 Iustin Pop
             " suffix is used",
570 a8083063 Iustin Pop
             default=4 * 1024, type="unit", metavar="<size>"),
571 d0834de3 Michael Hanselmann
  os_opt,
572 b9ac33e9 Iustin Pop
  cli_option("-m", "--memory", dest="mem", help="Memory size (in MiB)",
573 a8083063 Iustin Pop
              default=128, type="unit", metavar="<mem>"),
574 a8083063 Iustin Pop
  make_option("-p", "--cpu", dest="vcpus", help="Number of virtual CPUs",
575 a8083063 Iustin Pop
              default=1, type="int", metavar="<PROC>"),
576 a8083063 Iustin Pop
  make_option("-t", "--disk-template", dest="disk_template",
577 a8083063 Iustin Pop
              help="Custom disk setup (diskless, plain, local_raid1 or"
578 a8083063 Iustin Pop
              " remote_raid1)", default=None, metavar="TEMPL"),
579 a8083063 Iustin Pop
  make_option("-i", "--ip", dest="ip",
580 a8083063 Iustin Pop
              help="IP address ('none' [default], 'auto', or specify address)",
581 a8083063 Iustin Pop
              default='none', type="string", metavar="<ADDRESS>"),
582 a8083063 Iustin Pop
  make_option("--no-wait-for-sync", dest="wait_for_sync", default=True,
583 a8083063 Iustin Pop
              action="store_false", help="Don't wait for sync (DANGEROUS!)"),
584 a8083063 Iustin Pop
  make_option("--secondary-node", dest="snode",
585 a8083063 Iustin Pop
              help="Secondary node for remote_raid1 disk layout",
586 a8083063 Iustin Pop
              metavar="<node>"),
587 a8083063 Iustin Pop
  make_option("-b", "--bridge", dest="bridge",
588 a8083063 Iustin Pop
              help="Bridge to connect this instance to",
589 a8083063 Iustin Pop
              default=None, metavar="<bridge>")
590 a8083063 Iustin Pop
  ]
591 a8083063 Iustin Pop
592 a8083063 Iustin Pop
commands = {
593 a8083063 Iustin Pop
  'add': (AddInstance, ARGS_ONE, add_opts,
594 a8083063 Iustin Pop
          "[opts...] <name>",
595 a8083063 Iustin Pop
          "Creates and adds a new instance to the cluster"),
596 a8083063 Iustin Pop
  'add-mirror': (AddMDDRBDComponent, ARGS_ONE,
597 a8083063 Iustin Pop
                [DEBUG_OPT, node_opt,
598 a8083063 Iustin Pop
                 make_option("-b", "--disk", dest="disk", metavar="sdX",
599 a8083063 Iustin Pop
                             help=("The name of the instance disk for which to"
600 a8083063 Iustin Pop
                                   " add the mirror"))],
601 a8083063 Iustin Pop
                "-n node -b disk <instance>",
602 a8083063 Iustin Pop
                "Creates a new mirror for the instance"),
603 a8083063 Iustin Pop
  'console': (ConnectToInstanceConsole, ARGS_ONE, [DEBUG_OPT],
604 a8083063 Iustin Pop
              "<instance>",
605 a8083063 Iustin Pop
              "Opens a console on the specified instance"),
606 a8083063 Iustin Pop
  'failover': (FailoverInstance, ARGS_ONE,
607 fe7b0351 Michael Hanselmann
               [DEBUG_OPT, FORCE_OPT,
608 a8083063 Iustin Pop
                make_option("--ignore-consistency", dest="ignore_consistency",
609 a8083063 Iustin Pop
                            action="store_true", default=False,
610 a8083063 Iustin Pop
                            help="Ignore the consistency of the disks on"
611 a8083063 Iustin Pop
                            " the secondary"),
612 a8083063 Iustin Pop
                ],
613 a8083063 Iustin Pop
               "[-f] <instance>",
614 a8083063 Iustin Pop
               "Stops the instance and starts it on the backup node, using"
615 a8083063 Iustin Pop
               " the remote mirror (only for instances of type remote_raid1)"),
616 a8083063 Iustin Pop
  'info': (ShowInstanceConfig, ARGS_ANY, [DEBUG_OPT], "[<instance>...]",
617 a8083063 Iustin Pop
           "Show information on the specified instance"),
618 a8083063 Iustin Pop
  'list': (ListInstances, ARGS_NONE,
619 dcb93971 Michael Hanselmann
           [DEBUG_OPT, NOHDR_OPT, SEP_OPT, USEUNITS_OPT, FIELDS_OPT],
620 a8083063 Iustin Pop
           "", "Lists the instances and their status"),
621 d0834de3 Michael Hanselmann
  'reinstall': (ReinstallInstance, ARGS_ONE, [DEBUG_OPT, FORCE_OPT, os_opt],
622 fe7b0351 Michael Hanselmann
                "[-f] <instance>", "Reinstall the instance"),
623 fe7b0351 Michael Hanselmann
  'remove': (RemoveInstance, ARGS_ONE, [DEBUG_OPT, FORCE_OPT],
624 a8083063 Iustin Pop
             "[-f] <instance>", "Shuts down the instance and removes it"),
625 a8083063 Iustin Pop
  'remove-mirror': (RemoveMDDRBDComponent, ARGS_ONE,
626 a8083063 Iustin Pop
                   [DEBUG_OPT, node_opt,
627 a8083063 Iustin Pop
                    make_option("-b", "--disk", dest="disk", metavar="sdX",
628 a8083063 Iustin Pop
                                help=("The name of the instance disk"
629 a8083063 Iustin Pop
                                      " for which to add the mirror")),
630 a8083063 Iustin Pop
                    make_option("-p", "--port", dest="port", metavar="PORT",
631 a8083063 Iustin Pop
                                help=("The port of the drbd device"
632 a8083063 Iustin Pop
                                      " which to remove from the mirror"),
633 a8083063 Iustin Pop
                                type="int"),
634 a8083063 Iustin Pop
                    ],
635 a8083063 Iustin Pop
                   "-b disk -p port <instance>",
636 a8083063 Iustin Pop
                   "Removes a mirror from the instance"),
637 a8083063 Iustin Pop
  'replace-disks': (ReplaceDisks, ARGS_ONE,
638 a8083063 Iustin Pop
                    [DEBUG_OPT,
639 a8083063 Iustin Pop
                     make_option("-n", "--new-secondary", dest="new_secondary",
640 a8083063 Iustin Pop
                                 metavar="NODE",
641 a8083063 Iustin Pop
                                 help=("New secondary node (if you want to"
642 a8083063 Iustin Pop
                                       " change the secondary)"))],
643 a8083063 Iustin Pop
                    "[-n NODE] <instance>",
644 a8083063 Iustin Pop
                    "Replaces all disks for the instance"),
645 a8083063 Iustin Pop
  'modify': (SetInstanceParms, ARGS_ONE,
646 fe7b0351 Michael Hanselmann
             [DEBUG_OPT, FORCE_OPT,
647 a8083063 Iustin Pop
              cli_option("-m", "--memory", dest="mem",
648 a8083063 Iustin Pop
                         help="Memory size",
649 a8083063 Iustin Pop
                         default=None, type="unit", metavar="<mem>"),
650 a8083063 Iustin Pop
              make_option("-p", "--cpu", dest="vcpus",
651 a8083063 Iustin Pop
                          help="Number of virtual CPUs",
652 a8083063 Iustin Pop
                          default=None, type="int", metavar="<PROC>"),
653 a8083063 Iustin Pop
              make_option("-i", "--ip", dest="ip",
654 a8083063 Iustin Pop
                          help="IP address ('none' or numeric IP)",
655 a8083063 Iustin Pop
                          default=None, type="string", metavar="<ADDRESS>"),
656 a8083063 Iustin Pop
              make_option("-b", "--bridge", dest="bridge",
657 a8083063 Iustin Pop
                          help="Bridge to connect this instance to",
658 a8083063 Iustin Pop
                          default=None, type="string", metavar="<bridge>")
659 a8083063 Iustin Pop
              ],
660 a8083063 Iustin Pop
             "<instance>", "Alters the parameters of an instance"),
661 312ac745 Iustin Pop
  'shutdown': (ShutdownInstance, ARGS_ANY,
662 312ac745 Iustin Pop
               [DEBUG_OPT, m_node_opt, m_pri_node_opt, m_sec_node_opt,
663 312ac745 Iustin Pop
                m_clust_opt, m_inst_opt],
664 a8083063 Iustin Pop
               "<instance>", "Stops an instance"),
665 312ac745 Iustin Pop
  'startup': (StartupInstance, ARGS_ANY,
666 fe7b0351 Michael Hanselmann
              [DEBUG_OPT, FORCE_OPT,
667 a8083063 Iustin Pop
               make_option("-e", "--extra", dest="extra_args",
668 a8083063 Iustin Pop
                           help="Extra arguments for the instance's kernel",
669 a8083063 Iustin Pop
                           default=None, type="string", metavar="<PARAMS>"),
670 312ac745 Iustin Pop
               m_node_opt, m_pri_node_opt, m_sec_node_opt,
671 312ac745 Iustin Pop
               m_clust_opt, m_inst_opt,
672 a8083063 Iustin Pop
               ],
673 a8083063 Iustin Pop
            "<instance>", "Starts an instance"),
674 a8083063 Iustin Pop
  'activate-disks': (ActivateDisks, ARGS_ONE, [DEBUG_OPT],
675 a8083063 Iustin Pop
                     "<instance>",
676 a8083063 Iustin Pop
                     "Activate an instance's disks"),
677 a8083063 Iustin Pop
  'deactivate-disks': (DeactivateDisks, ARGS_ONE, [DEBUG_OPT],
678 a8083063 Iustin Pop
                       "<instance>",
679 a8083063 Iustin Pop
                       "Deactivate an instance's disks"),
680 a8083063 Iustin Pop
  }
681 a8083063 Iustin Pop
682 a8083063 Iustin Pop
if __name__ == '__main__':
683 3ecf6786 Iustin Pop
  sys.exit(GenericMain(commands))