Statistics
| Branch: | Tag: | Revision:

root / lib / client / gnt_instance.py @ 364c350f

History | View | Annotate | Download (55.7 kB)

1 e792102d Michael Hanselmann
#
2 a8083063 Iustin Pop
#
3 a8083063 Iustin Pop
4 ef8270dc Iustin Pop
# Copyright (C) 2006, 2007, 2008, 2009, 2010, 2011, 2012 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 7260cfbe Iustin Pop
"""Instance related commands"""
22 a8083063 Iustin Pop
23 b459a848 Andrea Spadaccini
# pylint: disable=W0401,W0614,C0103
24 2f79bd34 Iustin Pop
# W0401: Wildcard import ganeti.cli
25 2f79bd34 Iustin Pop
# W0614: Unused import %s from wildcard import (since we need cli)
26 7260cfbe Iustin Pop
# C0103: Invalid name gnt-instance
27 2f79bd34 Iustin Pop
28 83d4ba5e René Nussbaumer
import copy
29 312ac745 Iustin Pop
import itertools
30 0d0e9090 René Nussbaumer
import simplejson
31 25ce3ec4 Michael Hanselmann
import logging
32 a8083063 Iustin Pop
33 a8083063 Iustin Pop
from ganeti.cli import *
34 a8083063 Iustin Pop
from ganeti import opcodes
35 a8083063 Iustin Pop
from ganeti import constants
36 e2736e40 Guido Trotter
from ganeti import compat
37 a8083063 Iustin Pop
from ganeti import utils
38 312ac745 Iustin Pop
from ganeti import errors
39 a744b676 Manuel Franceschini
from ganeti import netutils
40 25ce3ec4 Michael Hanselmann
from ganeti import ssh
41 25ce3ec4 Michael Hanselmann
from ganeti import objects
42 735e1318 Michael Hanselmann
from ganeti import ht
43 312ac745 Iustin Pop
44 312ac745 Iustin Pop
45 c20efaa8 Michael Hanselmann
_EXPAND_CLUSTER = "cluster"
46 c20efaa8 Michael Hanselmann
_EXPAND_NODES_BOTH = "nodes"
47 c20efaa8 Michael Hanselmann
_EXPAND_NODES_PRI = "nodes-pri"
48 c20efaa8 Michael Hanselmann
_EXPAND_NODES_SEC = "nodes-sec"
49 c20efaa8 Michael Hanselmann
_EXPAND_NODES_BOTH_BY_TAGS = "nodes-by-tags"
50 c20efaa8 Michael Hanselmann
_EXPAND_NODES_PRI_BY_TAGS = "nodes-pri-by-tags"
51 c20efaa8 Michael Hanselmann
_EXPAND_NODES_SEC_BY_TAGS = "nodes-sec-by-tags"
52 c20efaa8 Michael Hanselmann
_EXPAND_INSTANCES = "instances"
53 c20efaa8 Michael Hanselmann
_EXPAND_INSTANCES_BY_TAGS = "instances-by-tags"
54 c20efaa8 Michael Hanselmann
55 b8028dcf Michael Hanselmann
_EXPAND_NODES_TAGS_MODES = compat.UniqueFrozenset([
56 c20efaa8 Michael Hanselmann
  _EXPAND_NODES_BOTH_BY_TAGS,
57 c20efaa8 Michael Hanselmann
  _EXPAND_NODES_PRI_BY_TAGS,
58 c20efaa8 Michael Hanselmann
  _EXPAND_NODES_SEC_BY_TAGS,
59 c20efaa8 Michael Hanselmann
  ])
60 312ac745 Iustin Pop
61 7232c04c Iustin Pop
#: default list of options for L{ListInstances}
62 48c4dfa8 Iustin Pop
_LIST_DEF_FIELDS = [
63 e69d05fd Iustin Pop
  "name", "hypervisor", "os", "pnode", "status", "oper_ram",
64 48c4dfa8 Iustin Pop
  ]
65 48c4dfa8 Iustin Pop
66 a71f835e Michael Hanselmann
_MISSING = object()
67 b8028dcf Michael Hanselmann
_ENV_OVERRIDE = compat.UniqueFrozenset(["list"])
68 ef9fa5b9 René Nussbaumer
69 b0a8e8c2 René Nussbaumer
_INST_DATA_VAL = ht.TListOf(ht.TDict)
70 b0a8e8c2 René Nussbaumer
71 ef9fa5b9 René Nussbaumer
72 479636a3 Iustin Pop
def _ExpandMultiNames(mode, names, client=None):
73 312ac745 Iustin Pop
  """Expand the given names using the passed mode.
74 312ac745 Iustin Pop

75 c20efaa8 Michael Hanselmann
  For _EXPAND_CLUSTER, all instances will be returned. For
76 c20efaa8 Michael Hanselmann
  _EXPAND_NODES_PRI/SEC, all instances having those nodes as
77 c20efaa8 Michael Hanselmann
  primary/secondary will be returned. For _EXPAND_NODES_BOTH, all
78 312ac745 Iustin Pop
  instances having those nodes as either primary or secondary will be
79 c20efaa8 Michael Hanselmann
  returned. For _EXPAND_INSTANCES, the given instances will be
80 312ac745 Iustin Pop
  returned.
81 312ac745 Iustin Pop

82 c20efaa8 Michael Hanselmann
  @param mode: one of L{_EXPAND_CLUSTER}, L{_EXPAND_NODES_BOTH},
83 c20efaa8 Michael Hanselmann
      L{_EXPAND_NODES_PRI}, L{_EXPAND_NODES_SEC} or
84 c20efaa8 Michael Hanselmann
      L{_EXPAND_INSTANCES}
85 7232c04c Iustin Pop
  @param names: a list of names; for cluster, it must be empty,
86 7232c04c Iustin Pop
      and for node and instance it must be a list of valid item
87 7232c04c Iustin Pop
      names (short names are valid as usual, e.g. node1 instead of
88 7232c04c Iustin Pop
      node1.example.com)
89 7232c04c Iustin Pop
  @rtype: list
90 7232c04c Iustin Pop
  @return: the list of names after the expansion
91 7232c04c Iustin Pop
  @raise errors.ProgrammerError: for unknown selection type
92 7232c04c Iustin Pop
  @raise errors.OpPrereqError: for invalid input parameters
93 7232c04c Iustin Pop

94 312ac745 Iustin Pop
  """
95 b459a848 Andrea Spadaccini
  # pylint: disable=W0142
96 39dfd93e René Nussbaumer
97 479636a3 Iustin Pop
  if client is None:
98 479636a3 Iustin Pop
    client = GetClient()
99 c20efaa8 Michael Hanselmann
  if mode == _EXPAND_CLUSTER:
100 312ac745 Iustin Pop
    if names:
101 debac808 Iustin Pop
      raise errors.OpPrereqError("Cluster filter mode takes no arguments",
102 debac808 Iustin Pop
                                 errors.ECODE_INVAL)
103 ec79568d Iustin Pop
    idata = client.QueryInstances([], ["name"], False)
104 312ac745 Iustin Pop
    inames = [row[0] for row in idata]
105 312ac745 Iustin Pop
106 c20efaa8 Michael Hanselmann
  elif (mode in _EXPAND_NODES_TAGS_MODES or
107 c20efaa8 Michael Hanselmann
        mode in (_EXPAND_NODES_BOTH, _EXPAND_NODES_PRI, _EXPAND_NODES_SEC)):
108 c20efaa8 Michael Hanselmann
    if mode in _EXPAND_NODES_TAGS_MODES:
109 39dfd93e René Nussbaumer
      if not names:
110 39dfd93e René Nussbaumer
        raise errors.OpPrereqError("No node tags passed", errors.ECODE_INVAL)
111 39dfd93e René Nussbaumer
      ndata = client.QueryNodes([], ["name", "pinst_list",
112 39dfd93e René Nussbaumer
                                     "sinst_list", "tags"], False)
113 39dfd93e René Nussbaumer
      ndata = [row for row in ndata if set(row[3]).intersection(names)]
114 39dfd93e René Nussbaumer
    else:
115 39dfd93e René Nussbaumer
      if not names:
116 39dfd93e René Nussbaumer
        raise errors.OpPrereqError("No node names passed", errors.ECODE_INVAL)
117 39dfd93e René Nussbaumer
      ndata = client.QueryNodes(names, ["name", "pinst_list", "sinst_list"],
118 5ae4945a Iustin Pop
                                False)
119 39dfd93e René Nussbaumer
120 312ac745 Iustin Pop
    ipri = [row[1] for row in ndata]
121 312ac745 Iustin Pop
    pri_names = list(itertools.chain(*ipri))
122 312ac745 Iustin Pop
    isec = [row[2] for row in ndata]
123 312ac745 Iustin Pop
    sec_names = list(itertools.chain(*isec))
124 c20efaa8 Michael Hanselmann
    if mode in (_EXPAND_NODES_BOTH, _EXPAND_NODES_BOTH_BY_TAGS):
125 312ac745 Iustin Pop
      inames = pri_names + sec_names
126 c20efaa8 Michael Hanselmann
    elif mode in (_EXPAND_NODES_PRI, _EXPAND_NODES_PRI_BY_TAGS):
127 312ac745 Iustin Pop
      inames = pri_names
128 c20efaa8 Michael Hanselmann
    elif mode in (_EXPAND_NODES_SEC, _EXPAND_NODES_SEC_BY_TAGS):
129 312ac745 Iustin Pop
      inames = sec_names
130 312ac745 Iustin Pop
    else:
131 312ac745 Iustin Pop
      raise errors.ProgrammerError("Unhandled shutdown type")
132 c20efaa8 Michael Hanselmann
  elif mode == _EXPAND_INSTANCES:
133 312ac745 Iustin Pop
    if not names:
134 debac808 Iustin Pop
      raise errors.OpPrereqError("No instance names passed",
135 debac808 Iustin Pop
                                 errors.ECODE_INVAL)
136 ec79568d Iustin Pop
    idata = client.QueryInstances(names, ["name"], False)
137 312ac745 Iustin Pop
    inames = [row[0] for row in idata]
138 c20efaa8 Michael Hanselmann
  elif mode == _EXPAND_INSTANCES_BY_TAGS:
139 39dfd93e René Nussbaumer
    if not names:
140 39dfd93e René Nussbaumer
      raise errors.OpPrereqError("No instance tags passed",
141 39dfd93e René Nussbaumer
                                 errors.ECODE_INVAL)
142 39dfd93e René Nussbaumer
    idata = client.QueryInstances([], ["name", "tags"], False)
143 39dfd93e René Nussbaumer
    inames = [row[0] for row in idata if set(row[1]).intersection(names)]
144 312ac745 Iustin Pop
  else:
145 debac808 Iustin Pop
    raise errors.OpPrereqError("Unknown mode '%s'" % mode, errors.ECODE_INVAL)
146 312ac745 Iustin Pop
147 312ac745 Iustin Pop
  return inames
148 a8083063 Iustin Pop
149 a8083063 Iustin Pop
150 a76f0c4a Iustin Pop
def _EnsureInstancesExist(client, names):
151 a76f0c4a Iustin Pop
  """Check for and ensure the given instance names exist.
152 a76f0c4a Iustin Pop

153 a76f0c4a Iustin Pop
  This function will raise an OpPrereqError in case they don't
154 a76f0c4a Iustin Pop
  exist. Otherwise it will exit cleanly.
155 a76f0c4a Iustin Pop

156 f2fd87d7 Iustin Pop
  @type client: L{ganeti.luxi.Client}
157 a76f0c4a Iustin Pop
  @param client: the client to use for the query
158 a76f0c4a Iustin Pop
  @type names: list
159 a76f0c4a Iustin Pop
  @param names: the list of instance names to query
160 a76f0c4a Iustin Pop
  @raise errors.OpPrereqError: in case any instance is missing
161 a76f0c4a Iustin Pop

162 a76f0c4a Iustin Pop
  """
163 f2af0bec Iustin Pop
  # TODO: change LUInstanceQuery to that it actually returns None
164 a76f0c4a Iustin Pop
  # instead of raising an exception, or devise a better mechanism
165 ec79568d Iustin Pop
  result = client.QueryInstances(names, ["name"], False)
166 a76f0c4a Iustin Pop
  for orig_name, row in zip(names, result):
167 a76f0c4a Iustin Pop
    if row[0] is None:
168 debac808 Iustin Pop
      raise errors.OpPrereqError("Instance '%s' does not exist" % orig_name,
169 debac808 Iustin Pop
                                 errors.ECODE_NOENT)
170 a76f0c4a Iustin Pop
171 a76f0c4a Iustin Pop
172 1c5945b6 Iustin Pop
def GenericManyOps(operation, fn):
173 1c5945b6 Iustin Pop
  """Generic multi-instance operations.
174 1c5945b6 Iustin Pop

175 1c5945b6 Iustin Pop
  The will return a wrapper that processes the options and arguments
176 1c5945b6 Iustin Pop
  given, and uses the passed function to build the opcode needed for
177 1c5945b6 Iustin Pop
  the specific operation. Thus all the generic loop/confirmation code
178 1c5945b6 Iustin Pop
  is abstracted into this function.
179 1c5945b6 Iustin Pop

180 1c5945b6 Iustin Pop
  """
181 1c5945b6 Iustin Pop
  def realfn(opts, args):
182 1c5945b6 Iustin Pop
    if opts.multi_mode is None:
183 c20efaa8 Michael Hanselmann
      opts.multi_mode = _EXPAND_INSTANCES
184 1c5945b6 Iustin Pop
    cl = GetClient()
185 1c5945b6 Iustin Pop
    inames = _ExpandMultiNames(opts.multi_mode, args, client=cl)
186 1c5945b6 Iustin Pop
    if not inames:
187 c20efaa8 Michael Hanselmann
      if opts.multi_mode == _EXPAND_CLUSTER:
188 c37bb2c6 Stephen Shirley
        ToStdout("Cluster is empty, no instances to shutdown")
189 c37bb2c6 Stephen Shirley
        return 0
190 1c5945b6 Iustin Pop
      raise errors.OpPrereqError("Selection filter does not match"
191 debac808 Iustin Pop
                                 " any instances", errors.ECODE_INVAL)
192 c20efaa8 Michael Hanselmann
    multi_on = opts.multi_mode != _EXPAND_INSTANCES or len(inames) > 1
193 1c5945b6 Iustin Pop
    if not (opts.force_multi or not multi_on
194 25bd815c René Nussbaumer
            or ConfirmOperation(inames, "instances", operation)):
195 1c5945b6 Iustin Pop
      return 1
196 cb573a31 Iustin Pop
    jex = JobExecutor(verbose=multi_on, cl=cl, opts=opts)
197 1c5945b6 Iustin Pop
    for name in inames:
198 1c5945b6 Iustin Pop
      op = fn(name, opts)
199 1c5945b6 Iustin Pop
      jex.QueueJob(name, op)
200 b4e68848 Iustin Pop
    results = jex.WaitOrShow(not opts.submit_only)
201 b4e68848 Iustin Pop
    rcode = compat.all(row[0] for row in results)
202 b4e68848 Iustin Pop
    return int(not rcode)
203 1c5945b6 Iustin Pop
  return realfn
204 1c5945b6 Iustin Pop
205 1c5945b6 Iustin Pop
206 a8083063 Iustin Pop
def ListInstances(opts, args):
207 f5abe9bd Oleksiy Mishchenko
  """List instances and their properties.
208 a8083063 Iustin Pop

209 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
210 7232c04c Iustin Pop
  @type args: list
211 7232c04c Iustin Pop
  @param args: should be an empty list
212 7232c04c Iustin Pop
  @rtype: int
213 7232c04c Iustin Pop
  @return: the desired exit code
214 7232c04c Iustin Pop

215 a8083063 Iustin Pop
  """
216 a4ebd726 Michael Hanselmann
  selected_fields = ParseFields(opts.output, _LIST_DEF_FIELDS)
217 a8083063 Iustin Pop
218 b82c5ff5 Michael Hanselmann
  fmtoverride = dict.fromkeys(["tags", "disk.sizes", "nic.macs", "nic.ips",
219 b82c5ff5 Michael Hanselmann
                               "nic.modes", "nic.links", "nic.bridges",
220 d4117a72 Apollon Oikonomopoulos
                               "nic.networks",
221 fab9573b Michael Hanselmann
                               "snodes", "snodes.group", "snodes.group.uuid"],
222 b82c5ff5 Michael Hanselmann
                              (lambda value: ",".join(str(item)
223 b82c5ff5 Michael Hanselmann
                                                      for item in value),
224 b82c5ff5 Michael Hanselmann
                               False))
225 a8083063 Iustin Pop
226 b82c5ff5 Michael Hanselmann
  return GenericList(constants.QR_INSTANCE, selected_fields, args, opts.units,
227 b82c5ff5 Michael Hanselmann
                     opts.separator, not opts.no_headers,
228 87e87959 Michael Hanselmann
                     format_override=fmtoverride, verbose=opts.verbose,
229 87e87959 Michael Hanselmann
                     force_filter=opts.force_filter)
230 b82c5ff5 Michael Hanselmann
231 b82c5ff5 Michael Hanselmann
232 b82c5ff5 Michael Hanselmann
def ListInstanceFields(opts, args):
233 b82c5ff5 Michael Hanselmann
  """List instance fields.
234 b82c5ff5 Michael Hanselmann

235 b82c5ff5 Michael Hanselmann
  @param opts: the command line options selected by the user
236 b82c5ff5 Michael Hanselmann
  @type args: list
237 b82c5ff5 Michael Hanselmann
  @param args: fields to list, or empty for all
238 b82c5ff5 Michael Hanselmann
  @rtype: int
239 b82c5ff5 Michael Hanselmann
  @return: the desired exit code
240 b82c5ff5 Michael Hanselmann

241 b82c5ff5 Michael Hanselmann
  """
242 b82c5ff5 Michael Hanselmann
  return GenericListFields(constants.QR_INSTANCE, args, opts.separator,
243 b82c5ff5 Michael Hanselmann
                           not opts.no_headers)
244 a8083063 Iustin Pop
245 a8083063 Iustin Pop
246 a8083063 Iustin Pop
def AddInstance(opts, args):
247 a8083063 Iustin Pop
  """Add an instance to the cluster.
248 a8083063 Iustin Pop

249 d77490c5 Iustin Pop
  This is just a wrapper over GenericInstanceCreate.
250 a8083063 Iustin Pop

251 a8083063 Iustin Pop
  """
252 d77490c5 Iustin Pop
  return GenericInstanceCreate(constants.INSTANCE_CREATE, opts, args)
253 a8083063 Iustin Pop
254 a8083063 Iustin Pop
255 0d0e9090 René Nussbaumer
def BatchCreate(opts, args):
256 7232c04c Iustin Pop
  """Create instances using a definition file.
257 7232c04c Iustin Pop

258 b0a8e8c2 René Nussbaumer
  This function reads a json file with L{opcodes.OpInstanceCreate}
259 b0a8e8c2 René Nussbaumer
  serialisations.
260 7232c04c Iustin Pop

261 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
262 7232c04c Iustin Pop
  @type args: list
263 7232c04c Iustin Pop
  @param args: should contain one element, the json filename
264 7232c04c Iustin Pop
  @rtype: int
265 7232c04c Iustin Pop
  @return: the desired exit code
266 0d0e9090 René Nussbaumer

267 0d0e9090 René Nussbaumer
  """
268 b0a8e8c2 René Nussbaumer
  (json_filename,) = args
269 b0a8e8c2 René Nussbaumer
  cl = GetClient()
270 0d0e9090 René Nussbaumer
271 0d0e9090 René Nussbaumer
  try:
272 13998ef2 Michael Hanselmann
    instance_data = simplejson.loads(utils.ReadFile(json_filename))
273 b459a848 Andrea Spadaccini
  except Exception, err: # pylint: disable=W0703
274 4082e6f9 Iustin Pop
    ToStderr("Can't parse the instance definition file: %s" % str(err))
275 4082e6f9 Iustin Pop
    return 1
276 0d0e9090 René Nussbaumer
277 b0a8e8c2 René Nussbaumer
  if not _INST_DATA_VAL(instance_data):
278 b0a8e8c2 René Nussbaumer
    ToStderr("The instance definition file is not %s" % _INST_DATA_VAL)
279 fe7c59d5 Guido Trotter
    return 1
280 fe7c59d5 Guido Trotter
281 b0a8e8c2 René Nussbaumer
  instances = []
282 b0a8e8c2 René Nussbaumer
  possible_params = set(opcodes.OpInstanceCreate.GetAllSlots())
283 b0a8e8c2 René Nussbaumer
  for (idx, inst) in enumerate(instance_data):
284 b0a8e8c2 René Nussbaumer
    unknown = set(inst.keys()) - possible_params
285 d4dd4b74 Iustin Pop
286 b0a8e8c2 René Nussbaumer
    if unknown:
287 b0a8e8c2 René Nussbaumer
      # TODO: Suggest closest match for more user friendly experience
288 3779121c René Nussbaumer
      raise errors.OpPrereqError("Unknown fields in definition %s: %s" %
289 3779121c René Nussbaumer
                                 (idx, utils.CommaJoin(unknown)),
290 3779121c René Nussbaumer
                                 errors.ECODE_INVAL)
291 0d0e9090 René Nussbaumer
292 3779121c René Nussbaumer
    op = opcodes.OpInstanceCreate(**inst) # pylint: disable=W0142
293 b0a8e8c2 René Nussbaumer
    op.Validate(False)
294 b0a8e8c2 René Nussbaumer
    instances.append(op)
295 0d0e9090 René Nussbaumer
296 b0a8e8c2 René Nussbaumer
  op = opcodes.OpInstanceMultiAlloc(iallocator=opts.iallocator,
297 b0a8e8c2 René Nussbaumer
                                    instances=instances)
298 b0a8e8c2 René Nussbaumer
  result = SubmitOrSend(op, opts, cl=cl)
299 0d0e9090 René Nussbaumer
300 b0a8e8c2 René Nussbaumer
  # Keep track of submitted jobs
301 b0a8e8c2 René Nussbaumer
  jex = JobExecutor(cl=cl, opts=opts)
302 b0a8e8c2 René Nussbaumer
303 b0a8e8c2 René Nussbaumer
  for (status, job_id) in result[constants.JOB_IDS_KEY]:
304 b0a8e8c2 René Nussbaumer
    jex.AddJobId(None, status, job_id)
305 b0a8e8c2 René Nussbaumer
306 b0a8e8c2 René Nussbaumer
  results = jex.GetResults()
307 b0a8e8c2 René Nussbaumer
  bad_cnt = len([row for row in results if not row[0]])
308 b0a8e8c2 René Nussbaumer
  if bad_cnt == 0:
309 b0a8e8c2 René Nussbaumer
    ToStdout("All instances created successfully.")
310 b0a8e8c2 René Nussbaumer
    rcode = constants.EXIT_SUCCESS
311 b0a8e8c2 René Nussbaumer
  else:
312 b0a8e8c2 René Nussbaumer
    ToStdout("There were %s errors during the creation.", bad_cnt)
313 b0a8e8c2 René Nussbaumer
    rcode = constants.EXIT_FAILURE
314 b0a8e8c2 René Nussbaumer
315 b0a8e8c2 René Nussbaumer
  return rcode
316 0d0e9090 René Nussbaumer
317 0d0e9090 René Nussbaumer
318 fe7b0351 Michael Hanselmann
def ReinstallInstance(opts, args):
319 fe7b0351 Michael Hanselmann
  """Reinstall an instance.
320 fe7b0351 Michael Hanselmann

321 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
322 7232c04c Iustin Pop
  @type args: list
323 7232c04c Iustin Pop
  @param args: should contain only one element, the name of the
324 7232c04c Iustin Pop
      instance to be reinstalled
325 7232c04c Iustin Pop
  @rtype: int
326 7232c04c Iustin Pop
  @return: the desired exit code
327 fe7b0351 Michael Hanselmann

328 fe7b0351 Michael Hanselmann
  """
329 55efe6da Iustin Pop
  # first, compute the desired name list
330 55efe6da Iustin Pop
  if opts.multi_mode is None:
331 c20efaa8 Michael Hanselmann
    opts.multi_mode = _EXPAND_INSTANCES
332 55efe6da Iustin Pop
333 55efe6da Iustin Pop
  inames = _ExpandMultiNames(opts.multi_mode, args)
334 55efe6da Iustin Pop
  if not inames:
335 debac808 Iustin Pop
    raise errors.OpPrereqError("Selection filter does not match any instances",
336 debac808 Iustin Pop
                               errors.ECODE_INVAL)
337 fe7b0351 Michael Hanselmann
338 55efe6da Iustin Pop
  # second, if requested, ask for an OS
339 20e23543 Alexander Schreiber
  if opts.select_os is True:
340 da2d02e7 Iustin Pop
    op = opcodes.OpOsDiagnose(output_fields=["name", "variants"], names=[])
341 400ca2f7 Iustin Pop
    result = SubmitOpCode(op, opts=opts)
342 20e23543 Alexander Schreiber
343 20e23543 Alexander Schreiber
    if not result:
344 3a24c527 Iustin Pop
      ToStdout("Can't get the OS list")
345 20e23543 Alexander Schreiber
      return 1
346 20e23543 Alexander Schreiber
347 3a24c527 Iustin Pop
    ToStdout("Available OS templates:")
348 20e23543 Alexander Schreiber
    number = 0
349 20e23543 Alexander Schreiber
    choices = []
350 d22dfef7 Iustin Pop
    for (name, variants) in result:
351 d22dfef7 Iustin Pop
      for entry in CalculateOSNames(name, variants):
352 d22dfef7 Iustin Pop
        ToStdout("%3s: %s", number, entry)
353 d22dfef7 Iustin Pop
        choices.append(("%s" % number, entry, entry))
354 d22dfef7 Iustin Pop
        number += 1
355 20e23543 Alexander Schreiber
356 d0c8c01d Iustin Pop
    choices.append(("x", "exit", "Exit gnt-instance reinstall"))
357 949bdabe Iustin Pop
    selected = AskUser("Enter OS template number (or x to abort):",
358 20e23543 Alexander Schreiber
                       choices)
359 20e23543 Alexander Schreiber
360 d0c8c01d Iustin Pop
    if selected == "exit":
361 55efe6da Iustin Pop
      ToStderr("User aborted reinstall, exiting")
362 20e23543 Alexander Schreiber
      return 1
363 20e23543 Alexander Schreiber
364 2f79bd34 Iustin Pop
    os_name = selected
365 f86426f5 Iustin Pop
    os_msg = "change the OS to '%s'" % selected
366 20e23543 Alexander Schreiber
  else:
367 2f79bd34 Iustin Pop
    os_name = opts.os
368 f86426f5 Iustin Pop
    if opts.os is not None:
369 f86426f5 Iustin Pop
      os_msg = "change the OS to '%s'" % os_name
370 f86426f5 Iustin Pop
    else:
371 f86426f5 Iustin Pop
      os_msg = "keep the same OS"
372 20e23543 Alexander Schreiber
373 297ddce9 Iustin Pop
  # third, get confirmation: multi-reinstall requires --force-multi,
374 297ddce9 Iustin Pop
  # single-reinstall either --force or --force-multi (--force-multi is
375 297ddce9 Iustin Pop
  # a stronger --force)
376 c20efaa8 Michael Hanselmann
  multi_on = opts.multi_mode != _EXPAND_INSTANCES or len(inames) > 1
377 55efe6da Iustin Pop
  if multi_on:
378 f86426f5 Iustin Pop
    warn_msg = ("Note: this will remove *all* data for the"
379 f86426f5 Iustin Pop
                " below instances! It will %s.\n" % os_msg)
380 297ddce9 Iustin Pop
    if not (opts.force_multi or
381 25bd815c René Nussbaumer
            ConfirmOperation(inames, "instances", "reinstall", extra=warn_msg)):
382 fe7b0351 Michael Hanselmann
      return 1
383 55efe6da Iustin Pop
  else:
384 297ddce9 Iustin Pop
    if not (opts.force or opts.force_multi):
385 f86426f5 Iustin Pop
      usertext = ("This will reinstall the instance '%s' (and %s) which"
386 f86426f5 Iustin Pop
                  " removes all data. Continue?") % (inames[0], os_msg)
387 55efe6da Iustin Pop
      if not AskUser(usertext):
388 55efe6da Iustin Pop
        return 1
389 55efe6da Iustin Pop
390 cb573a31 Iustin Pop
  jex = JobExecutor(verbose=multi_on, opts=opts)
391 55efe6da Iustin Pop
  for instance_name in inames:
392 5073fd8f Iustin Pop
    op = opcodes.OpInstanceReinstall(instance_name=instance_name,
393 06073e85 Guido Trotter
                                     os_type=os_name,
394 8d8c4eff Michael Hanselmann
                                     force_variant=opts.force_variant,
395 8d8c4eff Michael Hanselmann
                                     osparams=opts.osparams)
396 55efe6da Iustin Pop
    jex.QueueJob(instance_name, op)
397 fe7b0351 Michael Hanselmann
398 64be07b1 Michael Hanselmann
  results = jex.WaitOrShow(not opts.submit_only)
399 64be07b1 Michael Hanselmann
400 64be07b1 Michael Hanselmann
  if compat.all(map(compat.fst, results)):
401 64be07b1 Michael Hanselmann
    return constants.EXIT_SUCCESS
402 64be07b1 Michael Hanselmann
  else:
403 64be07b1 Michael Hanselmann
    return constants.EXIT_FAILURE
404 fe7b0351 Michael Hanselmann
405 fe7b0351 Michael Hanselmann
406 a8083063 Iustin Pop
def RemoveInstance(opts, args):
407 a8083063 Iustin Pop
  """Remove an instance.
408 a8083063 Iustin Pop

409 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
410 7232c04c Iustin Pop
  @type args: list
411 7232c04c Iustin Pop
  @param args: should contain only one element, the name of
412 7232c04c Iustin Pop
      the instance to be removed
413 7232c04c Iustin Pop
  @rtype: int
414 7232c04c Iustin Pop
  @return: the desired exit code
415 a8083063 Iustin Pop

416 a8083063 Iustin Pop
  """
417 a8083063 Iustin Pop
  instance_name = args[0]
418 a8083063 Iustin Pop
  force = opts.force
419 a76f0c4a Iustin Pop
  cl = GetClient()
420 a8083063 Iustin Pop
421 a8083063 Iustin Pop
  if not force:
422 a76f0c4a Iustin Pop
    _EnsureInstancesExist(cl, [instance_name])
423 a76f0c4a Iustin Pop
424 a8083063 Iustin Pop
    usertext = ("This will remove the volumes of the instance %s"
425 a8083063 Iustin Pop
                " (including mirrors), thus removing all the data"
426 a8083063 Iustin Pop
                " of the instance. Continue?") % instance_name
427 47988778 Iustin Pop
    if not AskUser(usertext):
428 a8083063 Iustin Pop
      return 1
429 a8083063 Iustin Pop
430 3cd2d4b1 Iustin Pop
  op = opcodes.OpInstanceRemove(instance_name=instance_name,
431 17c3f802 Guido Trotter
                                ignore_failures=opts.ignore_failures,
432 4d98c565 Guido Trotter
                                shutdown_timeout=opts.shutdown_timeout)
433 a76f0c4a Iustin Pop
  SubmitOrSend(op, opts, cl=cl)
434 a8083063 Iustin Pop
  return 0
435 a8083063 Iustin Pop
436 a8083063 Iustin Pop
437 decd5f45 Iustin Pop
def RenameInstance(opts, args):
438 4ab0b9e3 Guido Trotter
  """Rename an instance.
439 decd5f45 Iustin Pop

440 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
441 7232c04c Iustin Pop
  @type args: list
442 7232c04c Iustin Pop
  @param args: should contain two elements, the old and the
443 7232c04c Iustin Pop
      new instance names
444 7232c04c Iustin Pop
  @rtype: int
445 7232c04c Iustin Pop
  @return: the desired exit code
446 decd5f45 Iustin Pop

447 decd5f45 Iustin Pop
  """
448 90ed09b0 René Nussbaumer
  if not opts.name_check:
449 1b6dddc8 René Nussbaumer
    if not AskUser("As you disabled the check of the DNS entry, please verify"
450 1b6dddc8 René Nussbaumer
                   " that '%s' is a FQDN. Continue?" % args[1]):
451 1b6dddc8 René Nussbaumer
      return 1
452 1b6dddc8 René Nussbaumer
453 5659e2e2 Iustin Pop
  op = opcodes.OpInstanceRename(instance_name=args[0],
454 decd5f45 Iustin Pop
                                new_name=args[1],
455 3fe11ba3 Manuel Franceschini
                                ip_check=opts.ip_check,
456 3fe11ba3 Manuel Franceschini
                                name_check=opts.name_check)
457 6a016df9 Michael Hanselmann
  result = SubmitOrSend(op, opts)
458 6a016df9 Michael Hanselmann
459 48418fea Iustin Pop
  if result:
460 48418fea Iustin Pop
    ToStdout("Instance '%s' renamed to '%s'", args[0], result)
461 6a016df9 Michael Hanselmann
462 decd5f45 Iustin Pop
  return 0
463 decd5f45 Iustin Pop
464 decd5f45 Iustin Pop
465 a8083063 Iustin Pop
def ActivateDisks(opts, args):
466 a8083063 Iustin Pop
  """Activate an instance's disks.
467 a8083063 Iustin Pop

468 a8083063 Iustin Pop
  This serves two purposes:
469 7232c04c Iustin Pop
    - it allows (as long as the instance is not running)
470 7232c04c Iustin Pop
      mounting the disks and modifying them from the node
471 a8083063 Iustin Pop
    - it repairs inactive secondary drbds
472 a8083063 Iustin Pop

473 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
474 7232c04c Iustin Pop
  @type args: list
475 7232c04c Iustin Pop
  @param args: should contain only one element, the instance name
476 7232c04c Iustin Pop
  @rtype: int
477 7232c04c Iustin Pop
  @return: the desired exit code
478 7232c04c Iustin Pop

479 a8083063 Iustin Pop
  """
480 a8083063 Iustin Pop
  instance_name = args[0]
481 83f5d475 Iustin Pop
  op = opcodes.OpInstanceActivateDisks(instance_name=instance_name,
482 f30d8165 Iustin Pop
                                       ignore_size=opts.ignore_size,
483 f30d8165 Iustin Pop
                                       wait_for_sync=opts.wait_for_sync)
484 6340bb0a Iustin Pop
  disks_info = SubmitOrSend(op, opts)
485 a8083063 Iustin Pop
  for host, iname, nname in disks_info:
486 3a24c527 Iustin Pop
    ToStdout("%s:%s:%s", host, iname, nname)
487 a8083063 Iustin Pop
  return 0
488 a8083063 Iustin Pop
489 a8083063 Iustin Pop
490 a8083063 Iustin Pop
def DeactivateDisks(opts, args):
491 bd315bfa Iustin Pop
  """Deactivate an instance's disks.
492 a8083063 Iustin Pop

493 a8083063 Iustin Pop
  This function takes the instance name, looks for its primary node
494 a8083063 Iustin Pop
  and the tries to shutdown its block devices on that node.
495 a8083063 Iustin Pop

496 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
497 7232c04c Iustin Pop
  @type args: list
498 7232c04c Iustin Pop
  @param args: should contain only one element, the instance name
499 7232c04c Iustin Pop
  @rtype: int
500 7232c04c Iustin Pop
  @return: the desired exit code
501 7232c04c Iustin Pop

502 a8083063 Iustin Pop
  """
503 a8083063 Iustin Pop
  instance_name = args[0]
504 c9c41373 Iustin Pop
  op = opcodes.OpInstanceDeactivateDisks(instance_name=instance_name,
505 c9c41373 Iustin Pop
                                         force=opts.force)
506 6340bb0a Iustin Pop
  SubmitOrSend(op, opts)
507 a8083063 Iustin Pop
  return 0
508 a8083063 Iustin Pop
509 a8083063 Iustin Pop
510 bd315bfa Iustin Pop
def RecreateDisks(opts, args):
511 bd315bfa Iustin Pop
  """Recreate an instance's disks.
512 bd315bfa Iustin Pop

513 bd315bfa Iustin Pop
  @param opts: the command line options selected by the user
514 bd315bfa Iustin Pop
  @type args: list
515 bd315bfa Iustin Pop
  @param args: should contain only one element, the instance name
516 bd315bfa Iustin Pop
  @rtype: int
517 bd315bfa Iustin Pop
  @return: the desired exit code
518 bd315bfa Iustin Pop

519 bd315bfa Iustin Pop
  """
520 bd315bfa Iustin Pop
  instance_name = args[0]
521 735e1318 Michael Hanselmann
522 735e1318 Michael Hanselmann
  disks = []
523 735e1318 Michael Hanselmann
524 bd315bfa Iustin Pop
  if opts.disks:
525 735e1318 Michael Hanselmann
    for didx, ddict in opts.disks:
526 735e1318 Michael Hanselmann
      didx = int(didx)
527 735e1318 Michael Hanselmann
528 735e1318 Michael Hanselmann
      if not ht.TDict(ddict):
529 735e1318 Michael Hanselmann
        msg = "Invalid disk/%d value: expected dict, got %s" % (didx, ddict)
530 2cfbc784 Iustin Pop
        raise errors.OpPrereqError(msg, errors.ECODE_INVAL)
531 735e1318 Michael Hanselmann
532 735e1318 Michael Hanselmann
      if constants.IDISK_SIZE in ddict:
533 735e1318 Michael Hanselmann
        try:
534 735e1318 Michael Hanselmann
          ddict[constants.IDISK_SIZE] = \
535 735e1318 Michael Hanselmann
            utils.ParseUnit(ddict[constants.IDISK_SIZE])
536 735e1318 Michael Hanselmann
        except ValueError, err:
537 735e1318 Michael Hanselmann
          raise errors.OpPrereqError("Invalid disk size for disk %d: %s" %
538 2cfbc784 Iustin Pop
                                     (didx, err), errors.ECODE_INVAL)
539 735e1318 Michael Hanselmann
540 735e1318 Michael Hanselmann
      disks.append((didx, ddict))
541 735e1318 Michael Hanselmann
542 735e1318 Michael Hanselmann
    # TODO: Verify modifyable parameters (already done in
543 735e1318 Michael Hanselmann
    # LUInstanceRecreateDisks, but it'd be nice to have in the client)
544 bd315bfa Iustin Pop
545 c8a96ae7 Iustin Pop
  if opts.node:
546 38db4e7c Adam Ingrassia
    if opts.iallocator:
547 38db4e7c Adam Ingrassia
      msg = "At most one of either --nodes or --iallocator can be passed"
548 38db4e7c Adam Ingrassia
      raise errors.OpPrereqError(msg, errors.ECODE_INVAL)
549 c8a96ae7 Iustin Pop
    pnode, snode = SplitNodeOption(opts.node)
550 c8a96ae7 Iustin Pop
    nodes = [pnode]
551 c8a96ae7 Iustin Pop
    if snode is not None:
552 c8a96ae7 Iustin Pop
      nodes.append(snode)
553 c8a96ae7 Iustin Pop
  else:
554 c8a96ae7 Iustin Pop
    nodes = []
555 c8a96ae7 Iustin Pop
556 6b273e78 Iustin Pop
  op = opcodes.OpInstanceRecreateDisks(instance_name=instance_name,
557 38db4e7c Adam Ingrassia
                                       disks=disks, nodes=nodes,
558 38db4e7c Adam Ingrassia
                                       iallocator=opts.iallocator)
559 bd315bfa Iustin Pop
  SubmitOrSend(op, opts)
560 735e1318 Michael Hanselmann
561 bd315bfa Iustin Pop
  return 0
562 bd315bfa Iustin Pop
563 bd315bfa Iustin Pop
564 c6e911bc Iustin Pop
def GrowDisk(opts, args):
565 7232c04c Iustin Pop
  """Grow an instance's disks.
566 c6e911bc Iustin Pop

567 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
568 7232c04c Iustin Pop
  @type args: list
569 30a83755 Guido Trotter
  @param args: should contain three elements, the target instance name,
570 30a83755 Guido Trotter
      the target disk id, and the target growth
571 7232c04c Iustin Pop
  @rtype: int
572 7232c04c Iustin Pop
  @return: the desired exit code
573 c6e911bc Iustin Pop

574 c6e911bc Iustin Pop
  """
575 c6e911bc Iustin Pop
  instance = args[0]
576 c6e911bc Iustin Pop
  disk = args[1]
577 ad24e046 Iustin Pop
  try:
578 ad24e046 Iustin Pop
    disk = int(disk)
579 691744c4 Iustin Pop
  except (TypeError, ValueError), err:
580 debac808 Iustin Pop
    raise errors.OpPrereqError("Invalid disk index: %s" % str(err),
581 debac808 Iustin Pop
                               errors.ECODE_INVAL)
582 c8bde61e Iustin Pop
  try:
583 c8bde61e Iustin Pop
    amount = utils.ParseUnit(args[2])
584 c8bde61e Iustin Pop
  except errors.UnitParseError:
585 c8bde61e Iustin Pop
    raise errors.OpPrereqError("Can't parse the given amount '%s'" % args[2],
586 c8bde61e Iustin Pop
                               errors.ECODE_INVAL)
587 60472d29 Iustin Pop
  op = opcodes.OpInstanceGrowDisk(instance_name=instance,
588 60472d29 Iustin Pop
                                  disk=disk, amount=amount,
589 ef8270dc Iustin Pop
                                  wait_for_sync=opts.wait_for_sync,
590 ef8270dc Iustin Pop
                                  absolute=opts.absolute)
591 6340bb0a Iustin Pop
  SubmitOrSend(op, opts)
592 c6e911bc Iustin Pop
  return 0
593 c6e911bc Iustin Pop
594 c6e911bc Iustin Pop
595 1c5945b6 Iustin Pop
def _StartupInstance(name, opts):
596 7232c04c Iustin Pop
  """Startup instances.
597 a8083063 Iustin Pop

598 1c5945b6 Iustin Pop
  This returns the opcode to start an instance, and its decorator will
599 1c5945b6 Iustin Pop
  wrap this into a loop starting all desired instances.
600 7232c04c Iustin Pop

601 1c5945b6 Iustin Pop
  @param name: the name of the instance to act on
602 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
603 1c5945b6 Iustin Pop
  @return: the opcode needed for the operation
604 a8083063 Iustin Pop

605 a8083063 Iustin Pop
  """
606 c873d91c Iustin Pop
  op = opcodes.OpInstanceStartup(instance_name=name,
607 b44bd844 Michael Hanselmann
                                 force=opts.force,
608 885a0fc4 Iustin Pop
                                 ignore_offline_nodes=opts.ignore_offline,
609 323f9095 Stephen Shirley
                                 no_remember=opts.no_remember,
610 323f9095 Stephen Shirley
                                 startup_paused=opts.startup_paused)
611 1c5945b6 Iustin Pop
  # do not add these parameters to the opcode unless they're defined
612 1c5945b6 Iustin Pop
  if opts.hvparams:
613 1c5945b6 Iustin Pop
    op.hvparams = opts.hvparams
614 1c5945b6 Iustin Pop
  if opts.beparams:
615 1c5945b6 Iustin Pop
    op.beparams = opts.beparams
616 1c5945b6 Iustin Pop
  return op
617 a8083063 Iustin Pop
618 7c0d6283 Michael Hanselmann
619 1c5945b6 Iustin Pop
def _RebootInstance(name, opts):
620 7232c04c Iustin Pop
  """Reboot instance(s).
621 7232c04c Iustin Pop

622 1c5945b6 Iustin Pop
  This returns the opcode to reboot an instance, and its decorator
623 1c5945b6 Iustin Pop
  will wrap this into a loop rebooting all desired instances.
624 579d4337 Alexander Schreiber

625 1c5945b6 Iustin Pop
  @param name: the name of the instance to act on
626 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
627 1c5945b6 Iustin Pop
  @return: the opcode needed for the operation
628 579d4337 Alexander Schreiber

629 579d4337 Alexander Schreiber
  """
630 90ab1a95 Iustin Pop
  return opcodes.OpInstanceReboot(instance_name=name,
631 579d4337 Alexander Schreiber
                                  reboot_type=opts.reboot_type,
632 17c3f802 Guido Trotter
                                  ignore_secondaries=opts.ignore_secondaries,
633 aa922d64 Michele Tartara
                                  shutdown_timeout=opts.shutdown_timeout)
634 a8083063 Iustin Pop
635 7c0d6283 Michael Hanselmann
636 1c5945b6 Iustin Pop
def _ShutdownInstance(name, opts):
637 a8083063 Iustin Pop
  """Shutdown an instance.
638 a8083063 Iustin Pop

639 1c5945b6 Iustin Pop
  This returns the opcode to shutdown an instance, and its decorator
640 1c5945b6 Iustin Pop
  will wrap this into a loop shutting down all desired instances.
641 1c5945b6 Iustin Pop

642 1c5945b6 Iustin Pop
  @param name: the name of the instance to act on
643 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
644 1c5945b6 Iustin Pop
  @return: the opcode needed for the operation
645 a8083063 Iustin Pop

646 a8083063 Iustin Pop
  """
647 ee3e37a7 Iustin Pop
  return opcodes.OpInstanceShutdown(instance_name=name,
648 0d57ce24 Guido Trotter
                                    force=opts.force,
649 b44bd844 Michael Hanselmann
                                    timeout=opts.timeout,
650 885a0fc4 Iustin Pop
                                    ignore_offline_nodes=opts.ignore_offline,
651 885a0fc4 Iustin Pop
                                    no_remember=opts.no_remember)
652 a8083063 Iustin Pop
653 a8083063 Iustin Pop
654 a8083063 Iustin Pop
def ReplaceDisks(opts, args):
655 a8083063 Iustin Pop
  """Replace the disks of an instance
656 a8083063 Iustin Pop

657 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
658 7232c04c Iustin Pop
  @type args: list
659 7232c04c Iustin Pop
  @param args: should contain only one element, the instance name
660 7232c04c Iustin Pop
  @rtype: int
661 7232c04c Iustin Pop
  @return: the desired exit code
662 a8083063 Iustin Pop

663 a8083063 Iustin Pop
  """
664 a14db5ff Iustin Pop
  new_2ndary = opts.dst_node
665 b6e82a65 Iustin Pop
  iallocator = opts.iallocator
666 a9e0c397 Iustin Pop
  if opts.disks is None:
667 54155f52 Iustin Pop
    disks = []
668 a9e0c397 Iustin Pop
  else:
669 54155f52 Iustin Pop
    try:
670 54155f52 Iustin Pop
      disks = [int(i) for i in opts.disks.split(",")]
671 691744c4 Iustin Pop
    except (TypeError, ValueError), err:
672 debac808 Iustin Pop
      raise errors.OpPrereqError("Invalid disk index passed: %s" % str(err),
673 debac808 Iustin Pop
                                 errors.ECODE_INVAL)
674 05d47e33 Michael Hanselmann
  cnt = [opts.on_primary, opts.on_secondary, opts.auto,
675 7e9366f7 Iustin Pop
         new_2ndary is not None, iallocator is not None].count(True)
676 7e9366f7 Iustin Pop
  if cnt != 1:
677 d8d838cb Michael Hanselmann
    raise errors.OpPrereqError("One and only one of the -p, -s, -a, -n and -I"
678 debac808 Iustin Pop
                               " options must be passed", errors.ECODE_INVAL)
679 7e9366f7 Iustin Pop
  elif opts.on_primary:
680 a9e0c397 Iustin Pop
    mode = constants.REPLACE_DISK_PRI
681 7e9366f7 Iustin Pop
  elif opts.on_secondary:
682 a9e0c397 Iustin Pop
    mode = constants.REPLACE_DISK_SEC
683 05d47e33 Michael Hanselmann
  elif opts.auto:
684 05d47e33 Michael Hanselmann
    mode = constants.REPLACE_DISK_AUTO
685 05d47e33 Michael Hanselmann
    if disks:
686 05d47e33 Michael Hanselmann
      raise errors.OpPrereqError("Cannot specify disks when using automatic"
687 debac808 Iustin Pop
                                 " mode", errors.ECODE_INVAL)
688 7e9366f7 Iustin Pop
  elif new_2ndary is not None or iallocator is not None:
689 7e9366f7 Iustin Pop
    # replace secondary
690 7e9366f7 Iustin Pop
    mode = constants.REPLACE_DISK_CHG
691 a9e0c397 Iustin Pop
692 668f755d Iustin Pop
  op = opcodes.OpInstanceReplaceDisks(instance_name=args[0], disks=disks,
693 668f755d Iustin Pop
                                      remote_node=new_2ndary, mode=mode,
694 668f755d Iustin Pop
                                      iallocator=iallocator,
695 893e8f49 René Nussbaumer
                                      early_release=opts.early_release,
696 893e8f49 René Nussbaumer
                                      ignore_ipolicy=opts.ignore_ipolicy)
697 6340bb0a Iustin Pop
  SubmitOrSend(op, opts)
698 a8083063 Iustin Pop
  return 0
699 a8083063 Iustin Pop
700 a8083063 Iustin Pop
701 a8083063 Iustin Pop
def FailoverInstance(opts, args):
702 a8083063 Iustin Pop
  """Failover an instance.
703 a8083063 Iustin Pop

704 a8083063 Iustin Pop
  The failover is done by shutting it down on its present node and
705 a8083063 Iustin Pop
  starting it on the secondary.
706 a8083063 Iustin Pop

707 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
708 7232c04c Iustin Pop
  @type args: list
709 7232c04c Iustin Pop
  @param args: should contain only one element, the instance name
710 7232c04c Iustin Pop
  @rtype: int
711 7232c04c Iustin Pop
  @return: the desired exit code
712 a8083063 Iustin Pop

713 a8083063 Iustin Pop
  """
714 a76f0c4a Iustin Pop
  cl = GetClient()
715 80de0e3f Iustin Pop
  instance_name = args[0]
716 80de0e3f Iustin Pop
  force = opts.force
717 1b7761fd Apollon Oikonomopoulos
  iallocator = opts.iallocator
718 1b7761fd Apollon Oikonomopoulos
  target_node = opts.dst_node
719 1b7761fd Apollon Oikonomopoulos
720 1b7761fd Apollon Oikonomopoulos
  if iallocator and target_node:
721 1b7761fd Apollon Oikonomopoulos
    raise errors.OpPrereqError("Specify either an iallocator (-I), or a target"
722 1b7761fd Apollon Oikonomopoulos
                               " node (-n) but not both", errors.ECODE_INVAL)
723 a8083063 Iustin Pop
724 80de0e3f Iustin Pop
  if not force:
725 a76f0c4a Iustin Pop
    _EnsureInstancesExist(cl, [instance_name])
726 a76f0c4a Iustin Pop
727 80de0e3f Iustin Pop
    usertext = ("Failover will happen to image %s."
728 80de0e3f Iustin Pop
                " This requires a shutdown of the instance. Continue?" %
729 80de0e3f Iustin Pop
                (instance_name,))
730 80de0e3f Iustin Pop
    if not AskUser(usertext):
731 80de0e3f Iustin Pop
      return 1
732 a8083063 Iustin Pop
733 019dbee1 Iustin Pop
  op = opcodes.OpInstanceFailover(instance_name=instance_name,
734 17c3f802 Guido Trotter
                                  ignore_consistency=opts.ignore_consistency,
735 1b7761fd Apollon Oikonomopoulos
                                  shutdown_timeout=opts.shutdown_timeout,
736 1b7761fd Apollon Oikonomopoulos
                                  iallocator=iallocator,
737 b6aaf437 René Nussbaumer
                                  target_node=target_node,
738 b6aaf437 René Nussbaumer
                                  ignore_ipolicy=opts.ignore_ipolicy)
739 a76f0c4a Iustin Pop
  SubmitOrSend(op, opts, cl=cl)
740 80de0e3f Iustin Pop
  return 0
741 a8083063 Iustin Pop
742 a8083063 Iustin Pop
743 53c776b5 Iustin Pop
def MigrateInstance(opts, args):
744 53c776b5 Iustin Pop
  """Migrate an instance.
745 53c776b5 Iustin Pop

746 53c776b5 Iustin Pop
  The migrate is done without shutdown.
747 53c776b5 Iustin Pop

748 2f907a8c Iustin Pop
  @param opts: the command line options selected by the user
749 2f907a8c Iustin Pop
  @type args: list
750 2f907a8c Iustin Pop
  @param args: should contain only one element, the instance name
751 2f907a8c Iustin Pop
  @rtype: int
752 2f907a8c Iustin Pop
  @return: the desired exit code
753 53c776b5 Iustin Pop

754 53c776b5 Iustin Pop
  """
755 a76f0c4a Iustin Pop
  cl = GetClient()
756 53c776b5 Iustin Pop
  instance_name = args[0]
757 53c776b5 Iustin Pop
  force = opts.force
758 1b7761fd Apollon Oikonomopoulos
  iallocator = opts.iallocator
759 1b7761fd Apollon Oikonomopoulos
  target_node = opts.dst_node
760 1b7761fd Apollon Oikonomopoulos
761 1b7761fd Apollon Oikonomopoulos
  if iallocator and target_node:
762 1b7761fd Apollon Oikonomopoulos
    raise errors.OpPrereqError("Specify either an iallocator (-I), or a target"
763 1b7761fd Apollon Oikonomopoulos
                               " node (-n) but not both", errors.ECODE_INVAL)
764 53c776b5 Iustin Pop
765 53c776b5 Iustin Pop
  if not force:
766 a76f0c4a Iustin Pop
    _EnsureInstancesExist(cl, [instance_name])
767 a76f0c4a Iustin Pop
768 53c776b5 Iustin Pop
    if opts.cleanup:
769 53c776b5 Iustin Pop
      usertext = ("Instance %s will be recovered from a failed migration."
770 53c776b5 Iustin Pop
                  " Note that the migration procedure (including cleanup)" %
771 53c776b5 Iustin Pop
                  (instance_name,))
772 53c776b5 Iustin Pop
    else:
773 53c776b5 Iustin Pop
      usertext = ("Instance %s will be migrated. Note that migration" %
774 53c776b5 Iustin Pop
                  (instance_name,))
775 cf29cfb6 Iustin Pop
    usertext += (" might impact the instance if anything goes wrong"
776 cf29cfb6 Iustin Pop
                 " (e.g. due to bugs in the hypervisor). Continue?")
777 53c776b5 Iustin Pop
    if not AskUser(usertext):
778 53c776b5 Iustin Pop
      return 1
779 53c776b5 Iustin Pop
780 e71b9ef4 Iustin Pop
  # this should be removed once --non-live is deprecated
781 783a6c0b Iustin Pop
  if not opts.live and opts.migration_mode is not None:
782 e71b9ef4 Iustin Pop
    raise errors.OpPrereqError("Only one of the --non-live and "
783 783a6c0b Iustin Pop
                               "--migration-mode options can be passed",
784 e71b9ef4 Iustin Pop
                               errors.ECODE_INVAL)
785 e71b9ef4 Iustin Pop
  if not opts.live: # --non-live passed
786 8c35561f Iustin Pop
    mode = constants.HT_MIGRATION_NONLIVE
787 e71b9ef4 Iustin Pop
  else:
788 8c35561f Iustin Pop
    mode = opts.migration_mode
789 e71b9ef4 Iustin Pop
790 75c866c2 Iustin Pop
  op = opcodes.OpInstanceMigrate(instance_name=instance_name, mode=mode,
791 1b7761fd Apollon Oikonomopoulos
                                 cleanup=opts.cleanup, iallocator=iallocator,
792 e9c487be René Nussbaumer
                                 target_node=target_node,
793 3ed23330 René Nussbaumer
                                 allow_failover=opts.allow_failover,
794 8c0b16f6 Guido Trotter
                                 allow_runtime_changes=opts.allow_runtime_chgs,
795 3ed23330 René Nussbaumer
                                 ignore_ipolicy=opts.ignore_ipolicy)
796 f70bb622 Michael Hanselmann
  SubmitOrSend(op, cl=cl, opts=opts)
797 53c776b5 Iustin Pop
  return 0
798 53c776b5 Iustin Pop
799 53c776b5 Iustin Pop
800 fbf5a861 Iustin Pop
def MoveInstance(opts, args):
801 fbf5a861 Iustin Pop
  """Move an instance.
802 fbf5a861 Iustin Pop

803 fbf5a861 Iustin Pop
  @param opts: the command line options selected by the user
804 fbf5a861 Iustin Pop
  @type args: list
805 fbf5a861 Iustin Pop
  @param args: should contain only one element, the instance name
806 fbf5a861 Iustin Pop
  @rtype: int
807 fbf5a861 Iustin Pop
  @return: the desired exit code
808 fbf5a861 Iustin Pop

809 fbf5a861 Iustin Pop
  """
810 fbf5a861 Iustin Pop
  cl = GetClient()
811 fbf5a861 Iustin Pop
  instance_name = args[0]
812 fbf5a861 Iustin Pop
  force = opts.force
813 fbf5a861 Iustin Pop
814 fbf5a861 Iustin Pop
  if not force:
815 fbf5a861 Iustin Pop
    usertext = ("Instance %s will be moved."
816 fbf5a861 Iustin Pop
                " This requires a shutdown of the instance. Continue?" %
817 fbf5a861 Iustin Pop
                (instance_name,))
818 fbf5a861 Iustin Pop
    if not AskUser(usertext):
819 fbf5a861 Iustin Pop
      return 1
820 fbf5a861 Iustin Pop
821 0091b480 Iustin Pop
  op = opcodes.OpInstanceMove(instance_name=instance_name,
822 17c3f802 Guido Trotter
                              target_node=opts.node,
823 bb851c63 Iustin Pop
                              shutdown_timeout=opts.shutdown_timeout,
824 92cf62e3 René Nussbaumer
                              ignore_consistency=opts.ignore_consistency,
825 92cf62e3 René Nussbaumer
                              ignore_ipolicy=opts.ignore_ipolicy)
826 fbf5a861 Iustin Pop
  SubmitOrSend(op, opts, cl=cl)
827 fbf5a861 Iustin Pop
  return 0
828 fbf5a861 Iustin Pop
829 fbf5a861 Iustin Pop
830 a8083063 Iustin Pop
def ConnectToInstanceConsole(opts, args):
831 a8083063 Iustin Pop
  """Connect to the console of an instance.
832 a8083063 Iustin Pop

833 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
834 7232c04c Iustin Pop
  @type args: list
835 7232c04c Iustin Pop
  @param args: should contain only one element, the instance name
836 7232c04c Iustin Pop
  @rtype: int
837 7232c04c Iustin Pop
  @return: the desired exit code
838 a8083063 Iustin Pop

839 a8083063 Iustin Pop
  """
840 a8083063 Iustin Pop
  instance_name = args[0]
841 a8083063 Iustin Pop
842 25ce3ec4 Michael Hanselmann
  cl = GetClient()
843 25ce3ec4 Michael Hanselmann
  try:
844 25ce3ec4 Michael Hanselmann
    cluster_name = cl.QueryConfigValues(["cluster_name"])[0]
845 d6f46b6a Michael Hanselmann
    ((console_data, oper_state), ) = \
846 d6f46b6a Michael Hanselmann
      cl.QueryInstances([instance_name], ["console", "oper_state"], False)
847 25ce3ec4 Michael Hanselmann
  finally:
848 25ce3ec4 Michael Hanselmann
    # Ensure client connection is closed while external commands are run
849 25ce3ec4 Michael Hanselmann
    cl.Close()
850 25ce3ec4 Michael Hanselmann
851 25ce3ec4 Michael Hanselmann
  del cl
852 25ce3ec4 Michael Hanselmann
853 d6f46b6a Michael Hanselmann
  if not console_data:
854 d6f46b6a Michael Hanselmann
    if oper_state:
855 d6f46b6a Michael Hanselmann
      # Instance is running
856 d6f46b6a Michael Hanselmann
      raise errors.OpExecError("Console information for instance %s is"
857 d6f46b6a Michael Hanselmann
                               " unavailable" % instance_name)
858 d6f46b6a Michael Hanselmann
    else:
859 d6f46b6a Michael Hanselmann
      raise errors.OpExecError("Instance %s is not running, can't get console" %
860 d6f46b6a Michael Hanselmann
                               instance_name)
861 d6f46b6a Michael Hanselmann
862 25ce3ec4 Michael Hanselmann
  return _DoConsole(objects.InstanceConsole.FromDict(console_data),
863 25ce3ec4 Michael Hanselmann
                    opts.show_command, cluster_name)
864 25ce3ec4 Michael Hanselmann
865 25ce3ec4 Michael Hanselmann
866 25ce3ec4 Michael Hanselmann
def _DoConsole(console, show_command, cluster_name, feedback_fn=ToStdout,
867 25ce3ec4 Michael Hanselmann
               _runcmd_fn=utils.RunCmd):
868 cc0dec7b Iustin Pop
  """Acts based on the result of L{opcodes.OpInstanceConsole}.
869 25ce3ec4 Michael Hanselmann

870 25ce3ec4 Michael Hanselmann
  @type console: L{objects.InstanceConsole}
871 25ce3ec4 Michael Hanselmann
  @param console: Console object
872 25ce3ec4 Michael Hanselmann
  @type show_command: bool
873 25ce3ec4 Michael Hanselmann
  @param show_command: Whether to just display commands
874 25ce3ec4 Michael Hanselmann
  @type cluster_name: string
875 25ce3ec4 Michael Hanselmann
  @param cluster_name: Cluster name as retrieved from master daemon
876 25ce3ec4 Michael Hanselmann

877 25ce3ec4 Michael Hanselmann
  """
878 25ce3ec4 Michael Hanselmann
  assert console.Validate()
879 25ce3ec4 Michael Hanselmann
880 25ce3ec4 Michael Hanselmann
  if console.kind == constants.CONS_MESSAGE:
881 25ce3ec4 Michael Hanselmann
    feedback_fn(console.message)
882 25ce3ec4 Michael Hanselmann
  elif console.kind == constants.CONS_VNC:
883 25ce3ec4 Michael Hanselmann
    feedback_fn("Instance %s has VNC listening on %s:%s (display %s),"
884 25ce3ec4 Michael Hanselmann
                " URL <vnc://%s:%s/>",
885 25ce3ec4 Michael Hanselmann
                console.instance, console.host, console.port,
886 25ce3ec4 Michael Hanselmann
                console.display, console.host, console.port)
887 4d2cdb5a Andrea Spadaccini
  elif console.kind == constants.CONS_SPICE:
888 4d2cdb5a Andrea Spadaccini
    feedback_fn("Instance %s has SPICE listening on %s:%s", console.instance,
889 4d2cdb5a Andrea Spadaccini
                console.host, console.port)
890 25ce3ec4 Michael Hanselmann
  elif console.kind == constants.CONS_SSH:
891 25ce3ec4 Michael Hanselmann
    # Convert to string if not already one
892 25ce3ec4 Michael Hanselmann
    if isinstance(console.command, basestring):
893 25ce3ec4 Michael Hanselmann
      cmd = console.command
894 25ce3ec4 Michael Hanselmann
    else:
895 25ce3ec4 Michael Hanselmann
      cmd = utils.ShellQuoteArgs(console.command)
896 25ce3ec4 Michael Hanselmann
897 25ce3ec4 Michael Hanselmann
    srun = ssh.SshRunner(cluster_name=cluster_name)
898 25ce3ec4 Michael Hanselmann
    ssh_cmd = srun.BuildCmd(console.host, console.user, cmd,
899 25ce3ec4 Michael Hanselmann
                            batch=True, quiet=False, tty=True)
900 25ce3ec4 Michael Hanselmann
901 25ce3ec4 Michael Hanselmann
    if show_command:
902 25ce3ec4 Michael Hanselmann
      feedback_fn(utils.ShellQuoteArgs(ssh_cmd))
903 25ce3ec4 Michael Hanselmann
    else:
904 25ce3ec4 Michael Hanselmann
      result = _runcmd_fn(ssh_cmd, interactive=True)
905 25ce3ec4 Michael Hanselmann
      if result.failed:
906 25ce3ec4 Michael Hanselmann
        logging.error("Console command \"%s\" failed with reason '%s' and"
907 25ce3ec4 Michael Hanselmann
                      " output %r", result.cmd, result.fail_reason,
908 25ce3ec4 Michael Hanselmann
                      result.output)
909 25ce3ec4 Michael Hanselmann
        raise errors.OpExecError("Connection to console of instance %s failed,"
910 25ce3ec4 Michael Hanselmann
                                 " please check cluster configuration" %
911 25ce3ec4 Michael Hanselmann
                                 console.instance)
912 51c6e7b5 Michael Hanselmann
  else:
913 25ce3ec4 Michael Hanselmann
    raise errors.GenericError("Unknown console type '%s'" % console.kind)
914 678aa6d3 Michael Hanselmann
915 678aa6d3 Michael Hanselmann
  return constants.EXIT_SUCCESS
916 a8083063 Iustin Pop
917 a8083063 Iustin Pop
918 e2736e40 Guido Trotter
def _FormatLogicalID(dev_type, logical_id, roman):
919 19708787 Iustin Pop
  """Formats the logical_id of a disk.
920 19708787 Iustin Pop

921 19708787 Iustin Pop
  """
922 19708787 Iustin Pop
  if dev_type == constants.LD_DRBD8:
923 19708787 Iustin Pop
    node_a, node_b, port, minor_a, minor_b, key = logical_id
924 19708787 Iustin Pop
    data = [
925 e2736e40 Guido Trotter
      ("nodeA", "%s, minor=%s" % (node_a, compat.TryToRoman(minor_a,
926 e2736e40 Guido Trotter
                                                            convert=roman))),
927 e2736e40 Guido Trotter
      ("nodeB", "%s, minor=%s" % (node_b, compat.TryToRoman(minor_b,
928 e2736e40 Guido Trotter
                                                            convert=roman))),
929 2cbcf95d Bernardo Dal Seno
      ("port", str(compat.TryToRoman(port, convert=roman))),
930 2cbcf95d Bernardo Dal Seno
      ("auth key", str(key)),
931 19708787 Iustin Pop
      ]
932 19708787 Iustin Pop
  elif dev_type == constants.LD_LV:
933 19708787 Iustin Pop
    vg_name, lv_name = logical_id
934 19708787 Iustin Pop
    data = ["%s/%s" % (vg_name, lv_name)]
935 19708787 Iustin Pop
  else:
936 19708787 Iustin Pop
    data = [str(logical_id)]
937 19708787 Iustin Pop
938 19708787 Iustin Pop
  return data
939 19708787 Iustin Pop
940 19708787 Iustin Pop
941 2cbcf95d Bernardo Dal Seno
def _FormatListInfo(data):
942 2cbcf95d Bernardo Dal Seno
  return list(str(i) for i in data)
943 2cbcf95d Bernardo Dal Seno
944 2cbcf95d Bernardo Dal Seno
945 f965260c Michael Hanselmann
def _FormatBlockDevInfo(idx, top_level, dev, roman):
946 a8083063 Iustin Pop
  """Show block device information.
947 a8083063 Iustin Pop

948 7232c04c Iustin Pop
  This is only used by L{ShowInstanceConfig}, but it's too big to be
949 a8083063 Iustin Pop
  left for an inline definition.
950 a8083063 Iustin Pop

951 19708787 Iustin Pop
  @type idx: int
952 19708787 Iustin Pop
  @param idx: the index of the current disk
953 19708787 Iustin Pop
  @type top_level: boolean
954 19708787 Iustin Pop
  @param top_level: if this a top-level disk?
955 7232c04c Iustin Pop
  @type dev: dict
956 7232c04c Iustin Pop
  @param dev: dictionary with disk information
957 e2736e40 Guido Trotter
  @type roman: boolean
958 e2736e40 Guido Trotter
  @param roman: whether to try to use roman integers
959 19708787 Iustin Pop
  @return: a list of either strings, tuples or lists
960 19708787 Iustin Pop
      (which should be formatted at a higher indent level)
961 7232c04c Iustin Pop

962 a8083063 Iustin Pop
  """
963 19708787 Iustin Pop
  def helper(dtype, status):
964 7232c04c Iustin Pop
    """Format one line for physical device status.
965 7232c04c Iustin Pop

966 7232c04c Iustin Pop
    @type dtype: str
967 7232c04c Iustin Pop
    @param dtype: a constant from the L{constants.LDS_BLOCK} set
968 7232c04c Iustin Pop
    @type status: tuple
969 7232c04c Iustin Pop
    @param status: a tuple as returned from L{backend.FindBlockDevice}
970 19708787 Iustin Pop
    @return: the string representing the status
971 7232c04c Iustin Pop

972 7232c04c Iustin Pop
    """
973 a8083063 Iustin Pop
    if not status:
974 19708787 Iustin Pop
      return "not active"
975 19708787 Iustin Pop
    txt = ""
976 f208978a Michael Hanselmann
    (path, major, minor, syncp, estt, degr, ldisk_status) = status
977 19708787 Iustin Pop
    if major is None:
978 19708787 Iustin Pop
      major_string = "N/A"
979 a8083063 Iustin Pop
    else:
980 e2736e40 Guido Trotter
      major_string = str(compat.TryToRoman(major, convert=roman))
981 fd38ef95 Manuel Franceschini
982 19708787 Iustin Pop
    if minor is None:
983 19708787 Iustin Pop
      minor_string = "N/A"
984 19708787 Iustin Pop
    else:
985 e2736e40 Guido Trotter
      minor_string = str(compat.TryToRoman(minor, convert=roman))
986 19708787 Iustin Pop
987 19708787 Iustin Pop
    txt += ("%s (%s:%s)" % (path, major_string, minor_string))
988 19708787 Iustin Pop
    if dtype in (constants.LD_DRBD8, ):
989 19708787 Iustin Pop
      if syncp is not None:
990 19708787 Iustin Pop
        sync_text = "*RECOVERING* %5.2f%%," % syncp
991 19708787 Iustin Pop
        if estt:
992 e2736e40 Guido Trotter
          sync_text += " ETA %ss" % compat.TryToRoman(estt, convert=roman)
993 9db6dbce Iustin Pop
        else:
994 19708787 Iustin Pop
          sync_text += " ETA unknown"
995 19708787 Iustin Pop
      else:
996 19708787 Iustin Pop
        sync_text = "in sync"
997 19708787 Iustin Pop
      if degr:
998 19708787 Iustin Pop
        degr_text = "*DEGRADED*"
999 19708787 Iustin Pop
      else:
1000 19708787 Iustin Pop
        degr_text = "ok"
1001 f208978a Michael Hanselmann
      if ldisk_status == constants.LDS_FAULTY:
1002 19708787 Iustin Pop
        ldisk_text = " *MISSING DISK*"
1003 f208978a Michael Hanselmann
      elif ldisk_status == constants.LDS_UNKNOWN:
1004 f208978a Michael Hanselmann
        ldisk_text = " *UNCERTAIN STATE*"
1005 19708787 Iustin Pop
      else:
1006 19708787 Iustin Pop
        ldisk_text = ""
1007 19708787 Iustin Pop
      txt += (" %s, status %s%s" % (sync_text, degr_text, ldisk_text))
1008 19708787 Iustin Pop
    elif dtype == constants.LD_LV:
1009 f208978a Michael Hanselmann
      if ldisk_status == constants.LDS_FAULTY:
1010 19708787 Iustin Pop
        ldisk_text = " *FAILED* (failed drive?)"
1011 19708787 Iustin Pop
      else:
1012 19708787 Iustin Pop
        ldisk_text = ""
1013 19708787 Iustin Pop
      txt += ldisk_text
1014 19708787 Iustin Pop
    return txt
1015 19708787 Iustin Pop
1016 19708787 Iustin Pop
  # the header
1017 19708787 Iustin Pop
  if top_level:
1018 19708787 Iustin Pop
    if dev["iv_name"] is not None:
1019 19708787 Iustin Pop
      txt = dev["iv_name"]
1020 19708787 Iustin Pop
    else:
1021 e2736e40 Guido Trotter
      txt = "disk %s" % compat.TryToRoman(idx, convert=roman)
1022 a8083063 Iustin Pop
  else:
1023 e2736e40 Guido Trotter
    txt = "child %s" % compat.TryToRoman(idx, convert=roman)
1024 c98162a7 Iustin Pop
  if isinstance(dev["size"], int):
1025 c98162a7 Iustin Pop
    nice_size = utils.FormatUnit(dev["size"], "h")
1026 c98162a7 Iustin Pop
  else:
1027 2cbcf95d Bernardo Dal Seno
    nice_size = str(dev["size"])
1028 2cbcf95d Bernardo Dal Seno
  data = [(txt, "%s, size %s" % (dev["dev_type"], nice_size))]
1029 19708787 Iustin Pop
  if top_level:
1030 19708787 Iustin Pop
    data.append(("access mode", dev["mode"]))
1031 a8083063 Iustin Pop
  if dev["logical_id"] is not None:
1032 19708787 Iustin Pop
    try:
1033 e2736e40 Guido Trotter
      l_id = _FormatLogicalID(dev["dev_type"], dev["logical_id"], roman)
1034 19708787 Iustin Pop
    except ValueError:
1035 19708787 Iustin Pop
      l_id = [str(dev["logical_id"])]
1036 19708787 Iustin Pop
    if len(l_id) == 1:
1037 19708787 Iustin Pop
      data.append(("logical_id", l_id[0]))
1038 19708787 Iustin Pop
    else:
1039 19708787 Iustin Pop
      data.extend(l_id)
1040 a8083063 Iustin Pop
  elif dev["physical_id"] is not None:
1041 2cbcf95d Bernardo Dal Seno
    data.append(("physical_id:", _FormatListInfo(dev["physical_id"])))
1042 f965260c Michael Hanselmann
1043 f965260c Michael Hanselmann
  if dev["pstatus"]:
1044 19708787 Iustin Pop
    data.append(("on primary", helper(dev["dev_type"], dev["pstatus"])))
1045 f965260c Michael Hanselmann
1046 f965260c Michael Hanselmann
  if dev["sstatus"]:
1047 19708787 Iustin Pop
    data.append(("on secondary", helper(dev["dev_type"], dev["sstatus"])))
1048 a8083063 Iustin Pop
1049 fcbd025c Christos Stavrakakis
  data.append(("name", dev["name"]))
1050 fcbd025c Christos Stavrakakis
  data.append(("UUID", dev["uuid"]))
1051 fcbd025c Christos Stavrakakis
1052 a8083063 Iustin Pop
  if dev["children"]:
1053 2cbcf95d Bernardo Dal Seno
    data.append(("child devices", [
1054 2cbcf95d Bernardo Dal Seno
      _FormatBlockDevInfo(c_idx, False, child, roman)
1055 2cbcf95d Bernardo Dal Seno
      for c_idx, child in enumerate(dev["children"])
1056 2cbcf95d Bernardo Dal Seno
      ]))
1057 2cbcf95d Bernardo Dal Seno
  return data
1058 19708787 Iustin Pop
1059 19708787 Iustin Pop
1060 2cbcf95d Bernardo Dal Seno
def _FormatInstanceNicInfo(idx, nic):
1061 2cbcf95d Bernardo Dal Seno
  """Helper function for L{_FormatInstanceInfo()}"""
1062 fcbd025c Christos Stavrakakis
  (name, uuid, ip, mac, mode, link, _, netinfo) = nic
1063 2cbcf95d Bernardo Dal Seno
  network_name = None
1064 2cbcf95d Bernardo Dal Seno
  if netinfo:
1065 2cbcf95d Bernardo Dal Seno
    network_name = netinfo["name"]
1066 2cbcf95d Bernardo Dal Seno
  return [
1067 2cbcf95d Bernardo Dal Seno
    ("nic/%d" % idx, ""),
1068 2cbcf95d Bernardo Dal Seno
    ("MAC", str(mac)),
1069 2cbcf95d Bernardo Dal Seno
    ("IP", str(ip)),
1070 2cbcf95d Bernardo Dal Seno
    ("mode", str(mode)),
1071 2cbcf95d Bernardo Dal Seno
    ("link", str(link)),
1072 2cbcf95d Bernardo Dal Seno
    ("network", str(network_name)),
1073 fcbd025c Christos Stavrakakis
    ("UUID", str(uuid)),
1074 fcbd025c Christos Stavrakakis
    ("name", str(name)),
1075 2cbcf95d Bernardo Dal Seno
    ]
1076 2cbcf95d Bernardo Dal Seno
1077 2cbcf95d Bernardo Dal Seno
1078 2cbcf95d Bernardo Dal Seno
def _FormatInstanceNodesInfo(instance):
1079 2cbcf95d Bernardo Dal Seno
  """Helper function for L{_FormatInstanceInfo()}"""
1080 2cbcf95d Bernardo Dal Seno
  pgroup = ("%s (UUID %s)" %
1081 2cbcf95d Bernardo Dal Seno
            (instance["pnode_group_name"], instance["pnode_group_uuid"]))
1082 2cbcf95d Bernardo Dal Seno
  secs = utils.CommaJoin(("%s (group %s, group UUID %s)" %
1083 2cbcf95d Bernardo Dal Seno
                          (name, group_name, group_uuid))
1084 2cbcf95d Bernardo Dal Seno
                         for (name, group_name, group_uuid) in
1085 2cbcf95d Bernardo Dal Seno
                           zip(instance["snodes"],
1086 2cbcf95d Bernardo Dal Seno
                               instance["snodes_group_names"],
1087 2cbcf95d Bernardo Dal Seno
                               instance["snodes_group_uuids"]))
1088 2cbcf95d Bernardo Dal Seno
  return [
1089 2cbcf95d Bernardo Dal Seno
    [
1090 2cbcf95d Bernardo Dal Seno
      ("primary", instance["pnode"]),
1091 2cbcf95d Bernardo Dal Seno
      ("group", pgroup),
1092 2cbcf95d Bernardo Dal Seno
      ],
1093 2cbcf95d Bernardo Dal Seno
    [("secondaries", secs)],
1094 2cbcf95d Bernardo Dal Seno
    ]
1095 2cbcf95d Bernardo Dal Seno
1096 2cbcf95d Bernardo Dal Seno
1097 2cbcf95d Bernardo Dal Seno
def _GetVncConsoleInfo(instance):
1098 2cbcf95d Bernardo Dal Seno
  """Helper function for L{_FormatInstanceInfo()}"""
1099 2cbcf95d Bernardo Dal Seno
  vnc_bind_address = instance["hv_actual"].get(constants.HV_VNC_BIND_ADDRESS,
1100 2cbcf95d Bernardo Dal Seno
                                               None)
1101 2cbcf95d Bernardo Dal Seno
  if vnc_bind_address:
1102 2cbcf95d Bernardo Dal Seno
    port = instance["network_port"]
1103 2cbcf95d Bernardo Dal Seno
    display = int(port) - constants.VNC_BASE_PORT
1104 2cbcf95d Bernardo Dal Seno
    if display > 0 and vnc_bind_address == constants.IP4_ADDRESS_ANY:
1105 2cbcf95d Bernardo Dal Seno
      vnc_console_port = "%s:%s (display %s)" % (instance["pnode"],
1106 2cbcf95d Bernardo Dal Seno
                                                 port,
1107 2cbcf95d Bernardo Dal Seno
                                                 display)
1108 2cbcf95d Bernardo Dal Seno
    elif display > 0 and netutils.IP4Address.IsValid(vnc_bind_address):
1109 2cbcf95d Bernardo Dal Seno
      vnc_console_port = ("%s:%s (node %s) (display %s)" %
1110 2cbcf95d Bernardo Dal Seno
                           (vnc_bind_address, port,
1111 2cbcf95d Bernardo Dal Seno
                            instance["pnode"], display))
1112 2cbcf95d Bernardo Dal Seno
    else:
1113 2cbcf95d Bernardo Dal Seno
      # vnc bind address is a file
1114 2cbcf95d Bernardo Dal Seno
      vnc_console_port = "%s:%s" % (instance["pnode"],
1115 2cbcf95d Bernardo Dal Seno
                                    vnc_bind_address)
1116 2cbcf95d Bernardo Dal Seno
    ret = "vnc to %s" % vnc_console_port
1117 2cbcf95d Bernardo Dal Seno
  else:
1118 2cbcf95d Bernardo Dal Seno
    ret = None
1119 2cbcf95d Bernardo Dal Seno
  return ret
1120 2cbcf95d Bernardo Dal Seno
1121 2cbcf95d Bernardo Dal Seno
1122 2cbcf95d Bernardo Dal Seno
def _FormatInstanceInfo(instance, roman_integers):
1123 2cbcf95d Bernardo Dal Seno
  """Format instance information for L{cli.PrintGenericInfo()}"""
1124 2cbcf95d Bernardo Dal Seno
  istate = "configured to be %s" % instance["config_state"]
1125 2cbcf95d Bernardo Dal Seno
  if instance["run_state"]:
1126 2cbcf95d Bernardo Dal Seno
    istate += ", actual state is %s" % instance["run_state"]
1127 2cbcf95d Bernardo Dal Seno
  info = [
1128 2cbcf95d Bernardo Dal Seno
    ("Instance name", instance["name"]),
1129 2cbcf95d Bernardo Dal Seno
    ("UUID", instance["uuid"]),
1130 2cbcf95d Bernardo Dal Seno
    ("Serial number",
1131 2cbcf95d Bernardo Dal Seno
     str(compat.TryToRoman(instance["serial_no"], convert=roman_integers))),
1132 2cbcf95d Bernardo Dal Seno
    ("Creation time", utils.FormatTime(instance["ctime"])),
1133 2cbcf95d Bernardo Dal Seno
    ("Modification time", utils.FormatTime(instance["mtime"])),
1134 2cbcf95d Bernardo Dal Seno
    ("State", istate),
1135 2cbcf95d Bernardo Dal Seno
    ("Nodes", _FormatInstanceNodesInfo(instance)),
1136 2cbcf95d Bernardo Dal Seno
    ("Operating system", instance["os"]),
1137 2cbcf95d Bernardo Dal Seno
    ("Operating system parameters",
1138 2cbcf95d Bernardo Dal Seno
     FormatParamsDictInfo(instance["os_instance"], instance["os_actual"])),
1139 2cbcf95d Bernardo Dal Seno
    ]
1140 2cbcf95d Bernardo Dal Seno
1141 2cbcf95d Bernardo Dal Seno
  if "network_port" in instance:
1142 2cbcf95d Bernardo Dal Seno
    info.append(("Allocated network port",
1143 2cbcf95d Bernardo Dal Seno
                 str(compat.TryToRoman(instance["network_port"],
1144 2cbcf95d Bernardo Dal Seno
                                       convert=roman_integers))))
1145 2cbcf95d Bernardo Dal Seno
  info.append(("Hypervisor", instance["hypervisor"]))
1146 2cbcf95d Bernardo Dal Seno
  console = _GetVncConsoleInfo(instance)
1147 2cbcf95d Bernardo Dal Seno
  if console:
1148 2cbcf95d Bernardo Dal Seno
    info.append(("console connection", console))
1149 2cbcf95d Bernardo Dal Seno
  # deprecated "memory" value, kept for one version for compatibility
1150 2cbcf95d Bernardo Dal Seno
  # TODO(ganeti 2.7) remove.
1151 2cbcf95d Bernardo Dal Seno
  be_actual = copy.deepcopy(instance["be_actual"])
1152 2cbcf95d Bernardo Dal Seno
  be_actual["memory"] = be_actual[constants.BE_MAXMEM]
1153 2cbcf95d Bernardo Dal Seno
  info.extend([
1154 2cbcf95d Bernardo Dal Seno
    ("Hypervisor parameters",
1155 2cbcf95d Bernardo Dal Seno
     FormatParamsDictInfo(instance["hv_instance"], instance["hv_actual"])),
1156 2cbcf95d Bernardo Dal Seno
    ("Back-end parameters",
1157 2cbcf95d Bernardo Dal Seno
     FormatParamsDictInfo(instance["be_instance"], be_actual)),
1158 2cbcf95d Bernardo Dal Seno
    ("NICs", [
1159 2cbcf95d Bernardo Dal Seno
      _FormatInstanceNicInfo(idx, nic)
1160 2cbcf95d Bernardo Dal Seno
      for (idx, nic) in enumerate(instance["nics"])
1161 2cbcf95d Bernardo Dal Seno
      ]),
1162 2cbcf95d Bernardo Dal Seno
    ("Disk template", instance["disk_template"]),
1163 2cbcf95d Bernardo Dal Seno
    ("Disks", [
1164 2cbcf95d Bernardo Dal Seno
      _FormatBlockDevInfo(idx, True, device, roman_integers)
1165 2cbcf95d Bernardo Dal Seno
      for (idx, device) in enumerate(instance["disks"])
1166 2cbcf95d Bernardo Dal Seno
      ]),
1167 2cbcf95d Bernardo Dal Seno
    ])
1168 2cbcf95d Bernardo Dal Seno
  return info
1169 19708787 Iustin Pop
1170 98825740 Michael Hanselmann
1171 a8083063 Iustin Pop
def ShowInstanceConfig(opts, args):
1172 a8083063 Iustin Pop
  """Compute instance run-time status.
1173 a8083063 Iustin Pop

1174 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
1175 7232c04c Iustin Pop
  @type args: list
1176 7232c04c Iustin Pop
  @param args: either an empty list, and then we query all
1177 7232c04c Iustin Pop
      instances, or should contain a list of instance names
1178 7232c04c Iustin Pop
  @rtype: int
1179 7232c04c Iustin Pop
  @return: the desired exit code
1180 7232c04c Iustin Pop

1181 a8083063 Iustin Pop
  """
1182 220cde0b Guido Trotter
  if not args and not opts.show_all:
1183 220cde0b Guido Trotter
    ToStderr("No instance selected."
1184 220cde0b Guido Trotter
             " Please pass in --all if you want to query all instances.\n"
1185 220cde0b Guido Trotter
             "Note that this can take a long time on a big cluster.")
1186 220cde0b Guido Trotter
    return 1
1187 220cde0b Guido Trotter
  elif args and opts.show_all:
1188 220cde0b Guido Trotter
    ToStderr("Cannot use --all if you specify instance names.")
1189 220cde0b Guido Trotter
    return 1
1190 220cde0b Guido Trotter
1191 a8083063 Iustin Pop
  retcode = 0
1192 5c097318 Iustin Pop
  op = opcodes.OpInstanceQueryData(instances=args, static=opts.static,
1193 5c097318 Iustin Pop
                                   use_locking=not opts.static)
1194 400ca2f7 Iustin Pop
  result = SubmitOpCode(op, opts=opts)
1195 a8083063 Iustin Pop
  if not result:
1196 3a24c527 Iustin Pop
    ToStdout("No instances.")
1197 a8083063 Iustin Pop
    return 1
1198 a8083063 Iustin Pop
1199 2cbcf95d Bernardo Dal Seno
  PrintGenericInfo([
1200 2cbcf95d Bernardo Dal Seno
    _FormatInstanceInfo(instance, opts.roman_integers)
1201 2cbcf95d Bernardo Dal Seno
    for instance in result.values()
1202 2cbcf95d Bernardo Dal Seno
    ])
1203 a8083063 Iustin Pop
  return retcode
1204 a8083063 Iustin Pop
1205 a8083063 Iustin Pop
1206 a71f835e Michael Hanselmann
def _ConvertNicDiskModifications(mods):
1207 a71f835e Michael Hanselmann
  """Converts NIC/disk modifications from CLI to opcode.
1208 a71f835e Michael Hanselmann

1209 a71f835e Michael Hanselmann
  When L{opcodes.OpInstanceSetParams} was changed to support adding/removing
1210 a71f835e Michael Hanselmann
  disks at arbitrary indices, its parameter format changed. This function
1211 a71f835e Michael Hanselmann
  converts legacy requests (e.g. "--net add" or "--disk add:size=4G") to the
1212 a71f835e Michael Hanselmann
  newer format and adds support for new-style requests (e.g. "--new 4:add").
1213 a71f835e Michael Hanselmann

1214 a71f835e Michael Hanselmann
  @type mods: list of tuples
1215 a71f835e Michael Hanselmann
  @param mods: Modifications as given by command line parser
1216 a71f835e Michael Hanselmann
  @rtype: list of tuples
1217 a71f835e Michael Hanselmann
  @return: Modifications as understood by L{opcodes.OpInstanceSetParams}
1218 a71f835e Michael Hanselmann

1219 a71f835e Michael Hanselmann
  """
1220 a71f835e Michael Hanselmann
  result = []
1221 a71f835e Michael Hanselmann
1222 b21d488b Christos Stavrakakis
  for (identifier, params) in mods:
1223 b21d488b Christos Stavrakakis
    if identifier == constants.DDM_ADD:
1224 a71f835e Michael Hanselmann
      # Add item as last item (legacy interface)
1225 a71f835e Michael Hanselmann
      action = constants.DDM_ADD
1226 b21d488b Christos Stavrakakis
      identifier = -1
1227 b21d488b Christos Stavrakakis
    elif identifier == constants.DDM_REMOVE:
1228 a71f835e Michael Hanselmann
      # Remove last item (legacy interface)
1229 a71f835e Michael Hanselmann
      action = constants.DDM_REMOVE
1230 b21d488b Christos Stavrakakis
      identifier = -1
1231 a71f835e Michael Hanselmann
    else:
1232 a71f835e Michael Hanselmann
      # Modifications and adding/removing at arbitrary indices
1233 a71f835e Michael Hanselmann
      add = params.pop(constants.DDM_ADD, _MISSING)
1234 a71f835e Michael Hanselmann
      remove = params.pop(constants.DDM_REMOVE, _MISSING)
1235 f0d22861 Constantinos Venetsanopoulos
      modify = params.pop(constants.DDM_MODIFY, _MISSING)
1236 f0d22861 Constantinos Venetsanopoulos
1237 f0d22861 Constantinos Venetsanopoulos
      if modify is _MISSING:
1238 f0d22861 Constantinos Venetsanopoulos
        if not (add is _MISSING or remove is _MISSING):
1239 f0d22861 Constantinos Venetsanopoulos
          raise errors.OpPrereqError("Cannot add and remove at the same time",
1240 f0d22861 Constantinos Venetsanopoulos
                                     errors.ECODE_INVAL)
1241 f0d22861 Constantinos Venetsanopoulos
        elif add is not _MISSING:
1242 f0d22861 Constantinos Venetsanopoulos
          action = constants.DDM_ADD
1243 f0d22861 Constantinos Venetsanopoulos
        elif remove is not _MISSING:
1244 f0d22861 Constantinos Venetsanopoulos
          action = constants.DDM_REMOVE
1245 f0d22861 Constantinos Venetsanopoulos
        else:
1246 f0d22861 Constantinos Venetsanopoulos
          action = constants.DDM_MODIFY
1247 a71f835e Michael Hanselmann
1248 7a70541e Michael Hanselmann
      elif add is _MISSING and remove is _MISSING:
1249 7a70541e Michael Hanselmann
        action = constants.DDM_MODIFY
1250 a71f835e Michael Hanselmann
      else:
1251 7a70541e Michael Hanselmann
        raise errors.OpPrereqError("Cannot modify and add/remove at the"
1252 7a70541e Michael Hanselmann
                                   " same time", errors.ECODE_INVAL)
1253 a71f835e Michael Hanselmann
1254 a71f835e Michael Hanselmann
      assert not (constants.DDMS_VALUES_WITH_MODIFY & set(params.keys()))
1255 a71f835e Michael Hanselmann
1256 a71f835e Michael Hanselmann
    if action == constants.DDM_REMOVE and params:
1257 a71f835e Michael Hanselmann
      raise errors.OpPrereqError("Not accepting parameters on removal",
1258 a71f835e Michael Hanselmann
                                 errors.ECODE_INVAL)
1259 a71f835e Michael Hanselmann
1260 b21d488b Christos Stavrakakis
    result.append((action, identifier, params))
1261 a71f835e Michael Hanselmann
1262 a71f835e Michael Hanselmann
  return result
1263 a71f835e Michael Hanselmann
1264 a71f835e Michael Hanselmann
1265 a71f835e Michael Hanselmann
def _ParseDiskSizes(mods):
1266 a71f835e Michael Hanselmann
  """Parses disk sizes in parameters.
1267 a71f835e Michael Hanselmann

1268 a71f835e Michael Hanselmann
  """
1269 a71f835e Michael Hanselmann
  for (action, _, params) in mods:
1270 a71f835e Michael Hanselmann
    if params and constants.IDISK_SIZE in params:
1271 a71f835e Michael Hanselmann
      params[constants.IDISK_SIZE] = \
1272 a71f835e Michael Hanselmann
        utils.ParseUnit(params[constants.IDISK_SIZE])
1273 a71f835e Michael Hanselmann
    elif action == constants.DDM_ADD:
1274 a71f835e Michael Hanselmann
      raise errors.OpPrereqError("Missing required parameter 'size'",
1275 a71f835e Michael Hanselmann
                                 errors.ECODE_INVAL)
1276 a71f835e Michael Hanselmann
1277 a71f835e Michael Hanselmann
  return mods
1278 a71f835e Michael Hanselmann
1279 a71f835e Michael Hanselmann
1280 7767bbf5 Manuel Franceschini
def SetInstanceParams(opts, args):
1281 a8083063 Iustin Pop
  """Modifies an instance.
1282 a8083063 Iustin Pop

1283 a8083063 Iustin Pop
  All parameters take effect only at the next restart of the instance.
1284 a8083063 Iustin Pop

1285 7232c04c Iustin Pop
  @param opts: the command line options selected by the user
1286 7232c04c Iustin Pop
  @type args: list
1287 7232c04c Iustin Pop
  @param args: should contain only one element, the instance name
1288 7232c04c Iustin Pop
  @rtype: int
1289 7232c04c Iustin Pop
  @return: the desired exit code
1290 a8083063 Iustin Pop

1291 a8083063 Iustin Pop
  """
1292 e29e9550 Iustin Pop
  if not (opts.nics or opts.disks or opts.disk_template or
1293 57de31c0 Agata Murawska
          opts.hvparams or opts.beparams or opts.os or opts.osparams or
1294 3438e1f8 Klaus Aehlig
          opts.offline_inst or opts.online_inst or opts.runtime_mem or
1295 3438e1f8 Klaus Aehlig
          opts.new_primary_node):
1296 3a24c527 Iustin Pop
    ToStderr("Please give at least one of the parameters.")
1297 a8083063 Iustin Pop
    return 1
1298 a8083063 Iustin Pop
1299 467ae11e Guido Trotter
  for param in opts.beparams:
1300 e9d622bc Guido Trotter
    if isinstance(opts.beparams[param], basestring):
1301 e9d622bc Guido Trotter
      if opts.beparams[param].lower() == "default":
1302 e9d622bc Guido Trotter
        opts.beparams[param] = constants.VALUE_DEFAULT
1303 a5728081 Guido Trotter
1304 b2e233a5 Guido Trotter
  utils.ForceDictType(opts.beparams, constants.BES_PARAMETER_COMPAT,
1305 a5728081 Guido Trotter
                      allowed_values=[constants.VALUE_DEFAULT])
1306 467ae11e Guido Trotter
1307 48f212d7 Iustin Pop
  for param in opts.hvparams:
1308 48f212d7 Iustin Pop
    if isinstance(opts.hvparams[param], basestring):
1309 48f212d7 Iustin Pop
      if opts.hvparams[param].lower() == "default":
1310 48f212d7 Iustin Pop
        opts.hvparams[param] = constants.VALUE_DEFAULT
1311 a5728081 Guido Trotter
1312 48f212d7 Iustin Pop
  utils.ForceDictType(opts.hvparams, constants.HVS_PARAMETER_TYPES,
1313 a5728081 Guido Trotter
                      allowed_values=[constants.VALUE_DEFAULT])
1314 61be6ba4 Iustin Pop
1315 a71f835e Michael Hanselmann
  nics = _ConvertNicDiskModifications(opts.nics)
1316 a71f835e Michael Hanselmann
  disks = _ParseDiskSizes(_ConvertNicDiskModifications(opts.disks))
1317 24991749 Iustin Pop
1318 e29e9550 Iustin Pop
  if (opts.disk_template and
1319 3429a076 Apollon Oikonomopoulos
      opts.disk_template in constants.DTS_INT_MIRROR and
1320 e29e9550 Iustin Pop
      not opts.node):
1321 e29e9550 Iustin Pop
    ToStderr("Changing the disk template to a mirrored one requires"
1322 e29e9550 Iustin Pop
             " specifying a secondary node")
1323 e29e9550 Iustin Pop
    return 1
1324 e29e9550 Iustin Pop
1325 3016bc1f Michael Hanselmann
  if opts.offline_inst:
1326 3016bc1f Michael Hanselmann
    offline = True
1327 3016bc1f Michael Hanselmann
  elif opts.online_inst:
1328 3016bc1f Michael Hanselmann
    offline = False
1329 3016bc1f Michael Hanselmann
  else:
1330 3016bc1f Michael Hanselmann
    offline = None
1331 3016bc1f Michael Hanselmann
1332 9a3cc7ae Iustin Pop
  op = opcodes.OpInstanceSetParams(instance_name=args[0],
1333 a71f835e Michael Hanselmann
                                   nics=nics,
1334 a71f835e Michael Hanselmann
                                   disks=disks,
1335 e29e9550 Iustin Pop
                                   disk_template=opts.disk_template,
1336 e29e9550 Iustin Pop
                                   remote_node=opts.node,
1337 3438e1f8 Klaus Aehlig
                                   pnode=opts.new_primary_node,
1338 48f212d7 Iustin Pop
                                   hvparams=opts.hvparams,
1339 338e51e8 Iustin Pop
                                   beparams=opts.beparams,
1340 2c0af7da Guido Trotter
                                   runtime_mem=opts.runtime_mem,
1341 96b39bcc Iustin Pop
                                   os_name=opts.os,
1342 1052d622 Iustin Pop
                                   osparams=opts.osparams,
1343 96b39bcc Iustin Pop
                                   force_variant=opts.force_variant,
1344 456798ab Iustin Pop
                                   force=opts.force,
1345 57de31c0 Agata Murawska
                                   wait_for_sync=opts.wait_for_sync,
1346 3016bc1f Michael Hanselmann
                                   offline=offline,
1347 9c784fb3 Dimitris Aragiorgis
                                   conflicts_check=opts.conflicts_check,
1348 1559e1e7 René Nussbaumer
                                   ignore_ipolicy=opts.ignore_ipolicy)
1349 31a853d2 Iustin Pop
1350 6340bb0a Iustin Pop
  # even if here we process the result, we allow submit only
1351 6340bb0a Iustin Pop
  result = SubmitOrSend(op, opts)
1352 a8083063 Iustin Pop
1353 a8083063 Iustin Pop
  if result:
1354 3a24c527 Iustin Pop
    ToStdout("Modified instance %s", args[0])
1355 a8083063 Iustin Pop
    for param, data in result:
1356 3a24c527 Iustin Pop
      ToStdout(" - %-5s -> %s", param, data)
1357 e29e9550 Iustin Pop
    ToStdout("Please don't forget that most parameters take effect"
1358 d976957d Iustin Pop
             " only at the next (re)start of the instance initiated by"
1359 d976957d Iustin Pop
             " ganeti; restarting from within the instance will"
1360 d976957d Iustin Pop
             " not be enough.")
1361 a8083063 Iustin Pop
  return 0
1362 a8083063 Iustin Pop
1363 a8083063 Iustin Pop
1364 bd2a5569 Michael Hanselmann
def ChangeGroup(opts, args):
1365 bd2a5569 Michael Hanselmann
  """Moves an instance to another group.
1366 bd2a5569 Michael Hanselmann

1367 bd2a5569 Michael Hanselmann
  """
1368 bd2a5569 Michael Hanselmann
  (instance_name, ) = args
1369 bd2a5569 Michael Hanselmann
1370 bd2a5569 Michael Hanselmann
  cl = GetClient()
1371 bd2a5569 Michael Hanselmann
1372 bd2a5569 Michael Hanselmann
  op = opcodes.OpInstanceChangeGroup(instance_name=instance_name,
1373 bd2a5569 Michael Hanselmann
                                     iallocator=opts.iallocator,
1374 bd2a5569 Michael Hanselmann
                                     target_groups=opts.to,
1375 bd2a5569 Michael Hanselmann
                                     early_release=opts.early_release)
1376 f70bb622 Michael Hanselmann
  result = SubmitOrSend(op, opts, cl=cl)
1377 bd2a5569 Michael Hanselmann
1378 bd2a5569 Michael Hanselmann
  # Keep track of submitted jobs
1379 bd2a5569 Michael Hanselmann
  jex = JobExecutor(cl=cl, opts=opts)
1380 bd2a5569 Michael Hanselmann
1381 bd2a5569 Michael Hanselmann
  for (status, job_id) in result[constants.JOB_IDS_KEY]:
1382 bd2a5569 Michael Hanselmann
    jex.AddJobId(None, status, job_id)
1383 bd2a5569 Michael Hanselmann
1384 bd2a5569 Michael Hanselmann
  results = jex.GetResults()
1385 bd2a5569 Michael Hanselmann
  bad_cnt = len([row for row in results if not row[0]])
1386 bd2a5569 Michael Hanselmann
  if bad_cnt == 0:
1387 bd2a5569 Michael Hanselmann
    ToStdout("Instance '%s' changed group successfully.", instance_name)
1388 bd2a5569 Michael Hanselmann
    rcode = constants.EXIT_SUCCESS
1389 bd2a5569 Michael Hanselmann
  else:
1390 bd2a5569 Michael Hanselmann
    ToStdout("There were %s errors while changing group of instance '%s'.",
1391 bd2a5569 Michael Hanselmann
             bad_cnt, instance_name)
1392 bd2a5569 Michael Hanselmann
    rcode = constants.EXIT_FAILURE
1393 bd2a5569 Michael Hanselmann
1394 bd2a5569 Michael Hanselmann
  return rcode
1395 bd2a5569 Michael Hanselmann
1396 bd2a5569 Michael Hanselmann
1397 312ac745 Iustin Pop
# multi-instance selection options
1398 c38c44ad Michael Hanselmann
m_force_multi = cli_option("--force-multiple", dest="force_multi",
1399 c38c44ad Michael Hanselmann
                           help="Do not ask for confirmation when more than"
1400 c38c44ad Michael Hanselmann
                           " one instance is affected",
1401 c38c44ad Michael Hanselmann
                           action="store_true", default=False)
1402 804a1e8e Iustin Pop
1403 c38c44ad Michael Hanselmann
m_pri_node_opt = cli_option("--primary", dest="multi_mode",
1404 c38c44ad Michael Hanselmann
                            help="Filter by nodes (primary only)",
1405 c20efaa8 Michael Hanselmann
                            const=_EXPAND_NODES_PRI, action="store_const")
1406 312ac745 Iustin Pop
1407 c38c44ad Michael Hanselmann
m_sec_node_opt = cli_option("--secondary", dest="multi_mode",
1408 c38c44ad Michael Hanselmann
                            help="Filter by nodes (secondary only)",
1409 c20efaa8 Michael Hanselmann
                            const=_EXPAND_NODES_SEC, action="store_const")
1410 312ac745 Iustin Pop
1411 c38c44ad Michael Hanselmann
m_node_opt = cli_option("--node", dest="multi_mode",
1412 c38c44ad Michael Hanselmann
                        help="Filter by nodes (primary and secondary)",
1413 c20efaa8 Michael Hanselmann
                        const=_EXPAND_NODES_BOTH, action="store_const")
1414 312ac745 Iustin Pop
1415 c38c44ad Michael Hanselmann
m_clust_opt = cli_option("--all", dest="multi_mode",
1416 c38c44ad Michael Hanselmann
                         help="Select all instances in the cluster",
1417 c20efaa8 Michael Hanselmann
                         const=_EXPAND_CLUSTER, action="store_const")
1418 312ac745 Iustin Pop
1419 c38c44ad Michael Hanselmann
m_inst_opt = cli_option("--instance", dest="multi_mode",
1420 c38c44ad Michael Hanselmann
                        help="Filter by instance name [default]",
1421 c20efaa8 Michael Hanselmann
                        const=_EXPAND_INSTANCES, action="store_const")
1422 312ac745 Iustin Pop
1423 39dfd93e René Nussbaumer
m_node_tags_opt = cli_option("--node-tags", dest="multi_mode",
1424 39dfd93e René Nussbaumer
                             help="Filter by node tag",
1425 c20efaa8 Michael Hanselmann
                             const=_EXPAND_NODES_BOTH_BY_TAGS,
1426 39dfd93e René Nussbaumer
                             action="store_const")
1427 39dfd93e René Nussbaumer
1428 39dfd93e René Nussbaumer
m_pri_node_tags_opt = cli_option("--pri-node-tags", dest="multi_mode",
1429 39dfd93e René Nussbaumer
                                 help="Filter by primary node tag",
1430 c20efaa8 Michael Hanselmann
                                 const=_EXPAND_NODES_PRI_BY_TAGS,
1431 39dfd93e René Nussbaumer
                                 action="store_const")
1432 39dfd93e René Nussbaumer
1433 39dfd93e René Nussbaumer
m_sec_node_tags_opt = cli_option("--sec-node-tags", dest="multi_mode",
1434 39dfd93e René Nussbaumer
                                 help="Filter by secondary node tag",
1435 c20efaa8 Michael Hanselmann
                                 const=_EXPAND_NODES_SEC_BY_TAGS,
1436 39dfd93e René Nussbaumer
                                 action="store_const")
1437 39dfd93e René Nussbaumer
1438 39dfd93e René Nussbaumer
m_inst_tags_opt = cli_option("--tags", dest="multi_mode",
1439 39dfd93e René Nussbaumer
                             help="Filter by instance tag",
1440 c20efaa8 Michael Hanselmann
                             const=_EXPAND_INSTANCES_BY_TAGS,
1441 39dfd93e René Nussbaumer
                             action="store_const")
1442 312ac745 Iustin Pop
1443 a8083063 Iustin Pop
# this is defined separately due to readability only
1444 a8083063 Iustin Pop
add_opts = [
1445 064c21f8 Iustin Pop
  NOSTART_OPT,
1446 064c21f8 Iustin Pop
  OS_OPT,
1447 06073e85 Guido Trotter
  FORCE_VARIANT_OPT,
1448 25a8792c Iustin Pop
  NO_INSTALL_OPT,
1449 10889e0c René Nussbaumer
  IGNORE_IPOLICY_OPT,
1450 a8083063 Iustin Pop
  ]
1451 a8083063 Iustin Pop
1452 a8083063 Iustin Pop
commands = {
1453 d0c8c01d Iustin Pop
  "add": (
1454 eb28ecf6 Guido Trotter
    AddInstance, [ArgHost(min=1, max=1)], COMMON_CREATE_OPTS + add_opts,
1455 6ea815cf Iustin Pop
    "[...] -t disk-type -n node[:secondary-node] -o os-type <name>",
1456 6ea815cf Iustin Pop
    "Creates and adds a new instance to the cluster"),
1457 d0c8c01d Iustin Pop
  "batch-create": (
1458 b0a8e8c2 René Nussbaumer
    BatchCreate, [ArgFile(min=1, max=1)],
1459 b0a8e8c2 René Nussbaumer
    [DRY_RUN_OPT, PRIORITY_OPT, IALLOCATOR_OPT, SUBMIT_OPT],
1460 6ea815cf Iustin Pop
    "<instances.json>",
1461 6ea815cf Iustin Pop
    "Create a bunch of instances based on specs in the file."),
1462 d0c8c01d Iustin Pop
  "console": (
1463 6ea815cf Iustin Pop
    ConnectToInstanceConsole, ARGS_ONE_INSTANCE,
1464 aa06f8c6 Michael Hanselmann
    [SHOWCMD_OPT, PRIORITY_OPT],
1465 6ea815cf Iustin Pop
    "[--show-cmd] <instance>", "Opens a console on the specified instance"),
1466 d0c8c01d Iustin Pop
  "failover": (
1467 6ea815cf Iustin Pop
    FailoverInstance, ARGS_ONE_INSTANCE,
1468 db5a8a2d Iustin Pop
    [FORCE_OPT, IGNORE_CONSIST_OPT, SUBMIT_OPT, SHUTDOWN_TIMEOUT_OPT,
1469 b6aaf437 René Nussbaumer
     DRY_RUN_OPT, PRIORITY_OPT, DST_NODE_OPT, IALLOCATOR_OPT,
1470 aa7a5c90 Michele Tartara
     IGNORE_IPOLICY_OPT, CLEANUP_OPT],
1471 a6a3efe4 Iustin Pop
    "[-f] <instance>", "Stops the instance, changes its primary node and"
1472 a6a3efe4 Iustin Pop
    " (if it was originally running) starts it on the new node"
1473 a6a3efe4 Iustin Pop
    " (the secondary for mirrored instances or any node"
1474 a6a3efe4 Iustin Pop
    " for shared storage)."),
1475 d0c8c01d Iustin Pop
  "migrate": (
1476 6ea815cf Iustin Pop
    MigrateInstance, ARGS_ONE_INSTANCE,
1477 aa06f8c6 Michael Hanselmann
    [FORCE_OPT, NONLIVE_OPT, MIGRATION_MODE_OPT, CLEANUP_OPT, DRY_RUN_OPT,
1478 3ed23330 René Nussbaumer
     PRIORITY_OPT, DST_NODE_OPT, IALLOCATOR_OPT, ALLOW_FAILOVER_OPT,
1479 f70bb622 Michael Hanselmann
     IGNORE_IPOLICY_OPT, NORUNTIME_CHGS_OPT, SUBMIT_OPT],
1480 6ea815cf Iustin Pop
    "[-f] <instance>", "Migrate instance to its secondary node"
1481 1b7761fd Apollon Oikonomopoulos
    " (only for mirrored instances)"),
1482 d0c8c01d Iustin Pop
  "move": (
1483 6ea815cf Iustin Pop
    MoveInstance, ARGS_ONE_INSTANCE,
1484 db5a8a2d Iustin Pop
    [FORCE_OPT, SUBMIT_OPT, SINGLE_NODE_OPT, SHUTDOWN_TIMEOUT_OPT,
1485 92cf62e3 René Nussbaumer
     DRY_RUN_OPT, PRIORITY_OPT, IGNORE_CONSIST_OPT, IGNORE_IPOLICY_OPT],
1486 6ea815cf Iustin Pop
    "[-f] <instance>", "Move instance to an arbitrary node"
1487 6ea815cf Iustin Pop
    " (only for instances of type file and lv)"),
1488 d0c8c01d Iustin Pop
  "info": (
1489 6ea815cf Iustin Pop
    ShowInstanceConfig, ARGS_MANY_INSTANCES,
1490 aa06f8c6 Michael Hanselmann
    [STATIC_OPT, ALL_OPT, ROMAN_OPT, PRIORITY_OPT],
1491 6ea815cf Iustin Pop
    "[-s] {--all | <instance>...}",
1492 6ea815cf Iustin Pop
    "Show information on the specified instance(s)"),
1493 d0c8c01d Iustin Pop
  "list": (
1494 6ea815cf Iustin Pop
    ListInstances, ARGS_MANY_INSTANCES,
1495 87e87959 Michael Hanselmann
    [NOHDR_OPT, SEP_OPT, USEUNITS_OPT, FIELDS_OPT, VERBOSE_OPT,
1496 87e87959 Michael Hanselmann
     FORCE_FILTER_OPT],
1497 6ea815cf Iustin Pop
    "[<instance>...]",
1498 b82c5ff5 Michael Hanselmann
    "Lists the instances and their status. The available fields can be shown"
1499 b82c5ff5 Michael Hanselmann
    " using the \"list-fields\" command (see the man page for details)."
1500 b82c5ff5 Michael Hanselmann
    " The default field list is (in order): %s." %
1501 b82c5ff5 Michael Hanselmann
    utils.CommaJoin(_LIST_DEF_FIELDS),
1502 6ea815cf Iustin Pop
    ),
1503 b82c5ff5 Michael Hanselmann
  "list-fields": (
1504 b82c5ff5 Michael Hanselmann
    ListInstanceFields, [ArgUnknown()],
1505 b82c5ff5 Michael Hanselmann
    [NOHDR_OPT, SEP_OPT],
1506 b82c5ff5 Michael Hanselmann
    "[fields...]",
1507 b82c5ff5 Michael Hanselmann
    "Lists all available fields for instances"),
1508 d0c8c01d Iustin Pop
  "reinstall": (
1509 3e54ace7 Iustin Pop
    ReinstallInstance, [ArgInstance()],
1510 06073e85 Guido Trotter
    [FORCE_OPT, OS_OPT, FORCE_VARIANT_OPT, m_force_multi, m_node_opt,
1511 39dfd93e René Nussbaumer
     m_pri_node_opt, m_sec_node_opt, m_clust_opt, m_inst_opt, m_node_tags_opt,
1512 39dfd93e René Nussbaumer
     m_pri_node_tags_opt, m_sec_node_tags_opt, m_inst_tags_opt, SELECT_OS_OPT,
1513 8d8c4eff Michael Hanselmann
     SUBMIT_OPT, DRY_RUN_OPT, PRIORITY_OPT, OSPARAMS_OPT],
1514 6ea815cf Iustin Pop
    "[-f] <instance>", "Reinstall a stopped instance"),
1515 d0c8c01d Iustin Pop
  "remove": (
1516 6ea815cf Iustin Pop
    RemoveInstance, ARGS_ONE_INSTANCE,
1517 db5a8a2d Iustin Pop
    [FORCE_OPT, SHUTDOWN_TIMEOUT_OPT, IGNORE_FAILURES_OPT, SUBMIT_OPT,
1518 aa06f8c6 Michael Hanselmann
     DRY_RUN_OPT, PRIORITY_OPT],
1519 6ea815cf Iustin Pop
    "[-f] <instance>", "Shuts down the instance and removes it"),
1520 d0c8c01d Iustin Pop
  "rename": (
1521 6ea815cf Iustin Pop
    RenameInstance,
1522 6ea815cf Iustin Pop
    [ArgInstance(min=1, max=1), ArgHost(min=1, max=1)],
1523 aa06f8c6 Michael Hanselmann
    [NOIPCHECK_OPT, NONAMECHECK_OPT, SUBMIT_OPT, DRY_RUN_OPT, PRIORITY_OPT],
1524 6ea815cf Iustin Pop
    "<instance> <new_name>", "Rename the instance"),
1525 d0c8c01d Iustin Pop
  "replace-disks": (
1526 6ea815cf Iustin Pop
    ReplaceDisks, ARGS_ONE_INSTANCE,
1527 7ea7bcf6 Iustin Pop
    [AUTO_REPLACE_OPT, DISKIDX_OPT, IALLOCATOR_OPT, EARLY_RELEASE_OPT,
1528 db5a8a2d Iustin Pop
     NEW_SECONDARY_OPT, ON_PRIMARY_OPT, ON_SECONDARY_OPT, SUBMIT_OPT,
1529 893e8f49 René Nussbaumer
     DRY_RUN_OPT, PRIORITY_OPT, IGNORE_IPOLICY_OPT],
1530 50c1e351 Bernardo Dal Seno
    "[-s|-p|-a|-n NODE|-I NAME] <instance>",
1531 50c1e351 Bernardo Dal Seno
    "Replaces disks for the instance"),
1532 d0c8c01d Iustin Pop
  "modify": (
1533 6ea815cf Iustin Pop
    SetInstanceParams, ARGS_ONE_INSTANCE,
1534 e29e9550 Iustin Pop
    [BACKEND_OPT, DISK_OPT, FORCE_OPT, HVOPTS_OPT, NET_OPT, SUBMIT_OPT,
1535 1052d622 Iustin Pop
     DISK_TEMPLATE_OPT, SINGLE_NODE_OPT, OS_OPT, FORCE_VARIANT_OPT,
1536 57de31c0 Agata Murawska
     OSPARAMS_OPT, DRY_RUN_OPT, PRIORITY_OPT, NWSYNC_OPT, OFFLINE_INST_OPT,
1537 9c784fb3 Dimitris Aragiorgis
     ONLINE_INST_OPT, IGNORE_IPOLICY_OPT, RUNTIME_MEM_OPT,
1538 3438e1f8 Klaus Aehlig
     NOCONFLICTSCHECK_OPT, NEW_PRIMARY_OPT],
1539 6ea815cf Iustin Pop
    "<instance>", "Alters the parameters of an instance"),
1540 d0c8c01d Iustin Pop
  "shutdown": (
1541 1c5945b6 Iustin Pop
    GenericManyOps("shutdown", _ShutdownInstance), [ArgInstance()],
1542 0d57ce24 Guido Trotter
    [FORCE_OPT, m_node_opt, m_pri_node_opt, m_sec_node_opt, m_clust_opt,
1543 39dfd93e René Nussbaumer
     m_node_tags_opt, m_pri_node_tags_opt, m_sec_node_tags_opt,
1544 db5a8a2d Iustin Pop
     m_inst_tags_opt, m_inst_opt, m_force_multi, TIMEOUT_OPT, SUBMIT_OPT,
1545 885a0fc4 Iustin Pop
     DRY_RUN_OPT, PRIORITY_OPT, IGNORE_OFFLINE_OPT, NO_REMEMBER_OPT],
1546 6ea815cf Iustin Pop
    "<instance>", "Stops an instance"),
1547 d0c8c01d Iustin Pop
  "startup": (
1548 1c5945b6 Iustin Pop
    GenericManyOps("startup", _StartupInstance), [ArgInstance()],
1549 39dfd93e René Nussbaumer
    [FORCE_OPT, m_force_multi, m_node_opt, m_pri_node_opt, m_sec_node_opt,
1550 39dfd93e René Nussbaumer
     m_node_tags_opt, m_pri_node_tags_opt, m_sec_node_tags_opt,
1551 39dfd93e René Nussbaumer
     m_inst_tags_opt, m_clust_opt, m_inst_opt, SUBMIT_OPT, HVOPTS_OPT,
1552 885a0fc4 Iustin Pop
     BACKEND_OPT, DRY_RUN_OPT, PRIORITY_OPT, IGNORE_OFFLINE_OPT,
1553 323f9095 Stephen Shirley
     NO_REMEMBER_OPT, STARTUP_PAUSED_OPT],
1554 6ea815cf Iustin Pop
    "<instance>", "Starts an instance"),
1555 d0c8c01d Iustin Pop
  "reboot": (
1556 1c5945b6 Iustin Pop
    GenericManyOps("reboot", _RebootInstance), [ArgInstance()],
1557 064c21f8 Iustin Pop
    [m_force_multi, REBOOT_TYPE_OPT, IGNORE_SECONDARIES_OPT, m_node_opt,
1558 17c3f802 Guido Trotter
     m_pri_node_opt, m_sec_node_opt, m_clust_opt, m_inst_opt, SUBMIT_OPT,
1559 39dfd93e René Nussbaumer
     m_node_tags_opt, m_pri_node_tags_opt, m_sec_node_tags_opt,
1560 aa922d64 Michele Tartara
     m_inst_tags_opt, SHUTDOWN_TIMEOUT_OPT, DRY_RUN_OPT, PRIORITY_OPT],
1561 6ea815cf Iustin Pop
    "<instance>", "Reboots an instance"),
1562 d0c8c01d Iustin Pop
  "activate-disks": (
1563 db5a8a2d Iustin Pop
    ActivateDisks, ARGS_ONE_INSTANCE,
1564 f30d8165 Iustin Pop
    [SUBMIT_OPT, IGNORE_SIZE_OPT, PRIORITY_OPT, WFSYNC_OPT],
1565 6ea815cf Iustin Pop
    "<instance>", "Activate an instance's disks"),
1566 d0c8c01d Iustin Pop
  "deactivate-disks": (
1567 aa06f8c6 Michael Hanselmann
    DeactivateDisks, ARGS_ONE_INSTANCE,
1568 c9c41373 Iustin Pop
    [FORCE_OPT, SUBMIT_OPT, DRY_RUN_OPT, PRIORITY_OPT],
1569 c9c41373 Iustin Pop
    "[-f] <instance>", "Deactivate an instance's disks"),
1570 d0c8c01d Iustin Pop
  "recreate-disks": (
1571 aa06f8c6 Michael Hanselmann
    RecreateDisks, ARGS_ONE_INSTANCE,
1572 38db4e7c Adam Ingrassia
    [SUBMIT_OPT, DISK_OPT, NODE_PLACEMENT_OPT, DRY_RUN_OPT, PRIORITY_OPT,
1573 38db4e7c Adam Ingrassia
     IALLOCATOR_OPT],
1574 6ea815cf Iustin Pop
    "<instance>", "Recreate an instance's disks"),
1575 d0c8c01d Iustin Pop
  "grow-disk": (
1576 6ea815cf Iustin Pop
    GrowDisk,
1577 6ea815cf Iustin Pop
    [ArgInstance(min=1, max=1), ArgUnknown(min=1, max=1),
1578 6ea815cf Iustin Pop
     ArgUnknown(min=1, max=1)],
1579 ef8270dc Iustin Pop
    [SUBMIT_OPT, NWSYNC_OPT, DRY_RUN_OPT, PRIORITY_OPT, ABSOLUTE_OPT],
1580 6ea815cf Iustin Pop
    "<instance> <disk> <size>", "Grow an instance's disk"),
1581 bd2a5569 Michael Hanselmann
  "change-group": (
1582 bd2a5569 Michael Hanselmann
    ChangeGroup, ARGS_ONE_INSTANCE,
1583 f70bb622 Michael Hanselmann
    [TO_GROUP_OPT, IALLOCATOR_OPT, EARLY_RELEASE_OPT, PRIORITY_OPT, SUBMIT_OPT],
1584 bd2a5569 Michael Hanselmann
    "[-I <iallocator>] [--to <group>]", "Change group of instance"),
1585 d0c8c01d Iustin Pop
  "list-tags": (
1586 6bc3ed14 Michael Hanselmann
    ListTags, ARGS_ONE_INSTANCE, [],
1587 6ea815cf Iustin Pop
    "<instance_name>", "List the tags of the given instance"),
1588 d0c8c01d Iustin Pop
  "add-tags": (
1589 6ea815cf Iustin Pop
    AddTags, [ArgInstance(min=1, max=1), ArgUnknown()],
1590 6bc3ed14 Michael Hanselmann
    [TAG_SRC_OPT, PRIORITY_OPT, SUBMIT_OPT],
1591 6ea815cf Iustin Pop
    "<instance_name> tag...", "Add tags to the given instance"),
1592 d0c8c01d Iustin Pop
  "remove-tags": (
1593 6ea815cf Iustin Pop
    RemoveTags, [ArgInstance(min=1, max=1), ArgUnknown()],
1594 6bc3ed14 Michael Hanselmann
    [TAG_SRC_OPT, PRIORITY_OPT, SUBMIT_OPT],
1595 6ea815cf Iustin Pop
    "<instance_name> tag...", "Remove tags from given instance"),
1596 a8083063 Iustin Pop
  }
1597 a8083063 Iustin Pop
1598 7232c04c Iustin Pop
#: dictionary with aliases for commands
1599 dbfd89dd Guido Trotter
aliases = {
1600 d0c8c01d Iustin Pop
  "start": "startup",
1601 d0c8c01d Iustin Pop
  "stop": "shutdown",
1602 96897af7 Alexander Schreiber
  "show": "info",
1603 dbfd89dd Guido Trotter
  }
1604 dbfd89dd Guido Trotter
1605 a8005e17 Michael Hanselmann
1606 e792102d Michael Hanselmann
def Main():
1607 e792102d Michael Hanselmann
  return GenericMain(commands, aliases=aliases,
1608 ef9fa5b9 René Nussbaumer
                     override={"tag_type": constants.TAG_INSTANCE},
1609 ef9fa5b9 René Nussbaumer
                     env_override=_ENV_OVERRIDE)