Statistics
| Branch: | Tag: | Revision:

root / lib / client / gnt_network.py @ effc9cad

History | View | Annotate | Download (12 kB)

1
#
2
#
3

    
4
# Copyright (C) 2011, 2012 Google Inc.
5
#
6
# This program is free software; you can redistribute it and/or modify
7
# it under the terms of the GNU General Public License as published by
8
# the Free Software Foundation; either version 2 of the License, or
9
# (at your option) any later version.
10
#
11
# This program is distributed in the hope that it will be useful, but
12
# WITHOUT ANY WARRANTY; without even the implied warranty of
13
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14
# General Public License for more details.
15
#
16
# You should have received a copy of the GNU General Public License
17
# along with this program; if not, write to the Free Software
18
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
19
# 02110-1301, USA.
20

    
21
"""IP pool related commands"""
22

    
23
# pylint: disable=W0401,W0614
24
# W0401: Wildcard import ganeti.cli
25
# W0614: Unused import %s from wildcard import (since we need cli)
26

    
27
import textwrap
28
import itertools
29

    
30
from ganeti.cli import *
31
from ganeti import constants
32
from ganeti import opcodes
33
from ganeti import utils
34
from ganeti import errors
35

    
36

    
37
#: default list of fields for L{ListNetworks}
38
_LIST_DEF_FIELDS = ["name", "network", "gateway",
39
                    "network_type", "mac_prefix", "group_list", "tags"]
40

    
41

    
42
def _HandleReservedIPs(ips):
43
  if ips is None:
44
    return None
45
  elif not ips:
46
    return []
47
  else:
48
    return utils.UnescapeAndSplit(ips, sep=",")
49

    
50

    
51
def AddNetwork(opts, args):
52
  """Add a network to the cluster.
53

54
  @param opts: the command line options selected by the user
55
  @type args: list
56
  @param args: a list of length 1 with the network name to create
57
  @rtype: int
58
  @return: the desired exit code
59

60
  """
61
  (network_name, ) = args
62

    
63
  if opts.network is None:
64
    raise errors.OpPrereqError("The --network option must be given",
65
                               errors.ECODE_INVAL)
66

    
67
  if opts.tags is not None:
68
    tags = opts.tags.split(",")
69
  else:
70
    tags = []
71

    
72
  reserved_ips = _HandleReservedIPs(opts.add_reserved_ips)
73

    
74
  op = opcodes.OpNetworkAdd(network_name=network_name,
75
                            gateway=opts.gateway,
76
                            network=opts.network,
77
                            gateway6=opts.gateway6,
78
                            network6=opts.network6,
79
                            mac_prefix=opts.mac_prefix,
80
                            network_type=opts.network_type,
81
                            add_reserved_ips=reserved_ips,
82
                            conflicts_check=opts.conflicts_check,
83
                            tags=tags)
84
  SubmitOrSend(op, opts)
85

    
86

    
87
def _GetDefaultGroups(cl, groups):
88
  """Gets list of groups to operate on.
89

90
  If C{groups} doesn't contain groups, a list of all groups in the cluster is
91
  returned.
92

93
  @type cl: L{luxi.Client}
94
  @type groups: list
95
  @rtype: list
96

97
  """
98
  if groups:
99
    return groups
100

    
101
  return list(itertools.chain(*cl.QueryGroups([], ["uuid"], False)))
102

    
103

    
104
def ConnectNetwork(opts, args):
105
  """Map a network to a node group.
106

107
  @param opts: the command line options selected by the user
108
  @type args: list
109
  @param args: Network, mode, physlink and node groups
110
  @rtype: int
111
  @return: the desired exit code
112

113
  """
114
  cl = GetClient()
115

    
116
  (network, mode, link) = args[:3]
117
  groups = _GetDefaultGroups(cl, args[3:])
118

    
119
  # TODO: Change logic to support "--submit"
120
  for group in groups:
121
    op = opcodes.OpNetworkConnect(group_name=group,
122
                                  network_name=network,
123
                                  network_mode=mode,
124
                                  network_link=link,
125
                                  conflicts_check=opts.conflicts_check)
126
    SubmitOpCode(op, opts=opts, cl=cl)
127

    
128

    
129
def DisconnectNetwork(opts, args):
130
  """Unmap a network from a node group.
131

132
  @param opts: the command line options selected by the user
133
  @type args: list
134
  @param args: Network and node groups
135
  @rtype: int
136
  @return: the desired exit code
137

138
  """
139
  cl = GetClient()
140

    
141
  (network, ) = args[:1]
142
  groups = _GetDefaultGroups(cl, args[1:])
143

    
144
  # TODO: Change logic to support "--submit"
145
  for group in groups:
146
    op = opcodes.OpNetworkDisconnect(group_name=group,
147
                                     network_name=network,
148
                                     conflicts_check=opts.conflicts_check)
149
    SubmitOpCode(op, opts=opts, cl=cl)
150

    
151

    
152
def ListNetworks(opts, args):
153
  """List Ip pools and their properties.
154

155
  @param opts: the command line options selected by the user
156
  @type args: list
157
  @param args: networks to list, or empty for all
158
  @rtype: int
159
  @return: the desired exit code
160

161
  """
162
  desired_fields = ParseFields(opts.output, _LIST_DEF_FIELDS)
163
  fmtoverride = {
164
    "group_list":
165
      (lambda data: utils.CommaJoin("%s (%s, %s)" % (name, mode, link)
166
                                    for (name, mode, link) in data),
167
       False),
168
    "inst_list": (",".join, False),
169
    "tags": (",".join, False),
170
    }
171

    
172
  cl = GetClient()
173
  return GenericList(constants.QR_NETWORK, desired_fields, args, None,
174
                     opts.separator, not opts.no_headers,
175
                     verbose=opts.verbose, format_override=fmtoverride,
176
                     cl=cl)
177

    
178

    
179
def ListNetworkFields(opts, args):
180
  """List network fields.
181

182
  @param opts: the command line options selected by the user
183
  @type args: list
184
  @param args: fields to list, or empty for all
185
  @rtype: int
186
  @return: the desired exit code
187

188
  """
189
  return GenericListFields(constants.QR_NETWORK, args, opts.separator,
190
                           not opts.no_headers)
191

    
192

    
193
def ShowNetworkConfig(_, args):
194
  """Show network information.
195

196
  @type args: list
197
  @param args: should either be an empty list, in which case
198
      we show information about all nodes, or should contain
199
      a list of networks (names or UUIDs) to be queried for information
200
  @rtype: int
201
  @return: the desired exit code
202

203
  """
204
  cl = GetClient()
205
  result = cl.QueryNetworks(fields=["name", "network", "gateway",
206
                                    "network6", "gateway6",
207
                                    "mac_prefix", "network_type",
208
                                    "free_count", "reserved_count",
209
                                    "map", "group_list", "inst_list",
210
                                    "external_reservations",
211
                                    "serial_no", "uuid"],
212
                            names=args, use_locking=False)
213

    
214
  for (name, network, gateway, network6, gateway6,
215
       mac_prefix, network_type, free_count, reserved_count,
216
       mapping, group_list, instances, ext_res, serial, uuid) in result:
217
    size = free_count + reserved_count
218
    ToStdout("Network name: %s", name)
219
    ToStdout("UUID: %s", uuid)
220
    ToStdout("Serial number: %d", serial)
221
    ToStdout("  Subnet: %s", network)
222
    ToStdout("  Gateway: %s", gateway)
223
    ToStdout("  IPv6 Subnet: %s", network6)
224
    ToStdout("  IPv6 Gateway: %s", gateway6)
225
    ToStdout("  Mac Prefix: %s", mac_prefix)
226
    ToStdout("  Type: %s", network_type)
227
    ToStdout("  Size: %d", size)
228
    ToStdout("  Free: %d (%.2f%%)", free_count,
229
             100 * float(free_count) / float(size))
230
    ToStdout("  Usage map:")
231
    idx = 0
232
    for line in textwrap.wrap(mapping, width=64):
233
      ToStdout("     %s %s %d", str(idx).rjust(3), line.ljust(64), idx + 63)
234
      idx += 64
235
    ToStdout("         (X) used    (.) free")
236

    
237
    if ext_res:
238
      ToStdout("  externally reserved IPs:")
239
      for line in textwrap.wrap(ext_res, width=64):
240
        ToStdout("    %s" % line)
241

    
242
    if group_list:
243
      ToStdout("  connected to node groups:")
244
      for group in group_list:
245
        ToStdout("    %s", group)
246
    else:
247
      ToStdout("  not connected to any node group")
248

    
249
    if instances:
250
      ToStdout("  used by %d instances:", len(instances))
251
      for inst in instances:
252
        ((ips, networks), ) = cl.QueryInstances([inst],
253
                                                ["nic.ips", "nic.networks"],
254
                                                use_locking=False)
255

    
256
        l = lambda value: ", ".join(str(idx) + ":" + str(ip)
257
                                    for idx, (ip, net) in enumerate(value)
258
                                      if net == name)
259

    
260
        ToStdout("    %s : %s", inst, l(zip(ips, networks)))
261
    else:
262
      ToStdout("  not used by any instances")
263

    
264

    
265
def SetNetworkParams(opts, args):
266
  """Modifies an IP address pool's parameters.
267

268
  @param opts: the command line options selected by the user
269
  @type args: list
270
  @param args: should contain only one element, the node group name
271

272
  @rtype: int
273
  @return: the desired exit code
274

275
  """
276
  # TODO: add "network": opts.network,
277
  all_changes = {
278
    "gateway": opts.gateway,
279
    "add_reserved_ips": _HandleReservedIPs(opts.add_reserved_ips),
280
    "remove_reserved_ips": _HandleReservedIPs(opts.remove_reserved_ips),
281
    "mac_prefix": opts.mac_prefix,
282
    "network_type": opts.network_type,
283
    "gateway6": opts.gateway6,
284
    "network6": opts.network6,
285
  }
286

    
287
  if all_changes.values().count(None) == len(all_changes):
288
    ToStderr("Please give at least one of the parameters.")
289
    return 1
290

    
291
  # pylint: disable=W0142
292
  op = opcodes.OpNetworkSetParams(network_name=args[0], **all_changes)
293

    
294
  # TODO: add feedback to user, e.g. list the modifications
295
  SubmitOrSend(op, opts)
296

    
297

    
298
def RemoveNetwork(opts, args):
299
  """Remove an IP address pool from the cluster.
300

301
  @param opts: the command line options selected by the user
302
  @type args: list
303
  @param args: a list of length 1 with the id of the IP address pool to remove
304
  @rtype: int
305
  @return: the desired exit code
306

307
  """
308
  (network_name,) = args
309
  op = opcodes.OpNetworkRemove(network_name=network_name, force=opts.force)
310
  SubmitOrSend(op, opts)
311

    
312

    
313
commands = {
314
  "add": (
315
    AddNetwork, ARGS_ONE_NETWORK,
316
    [DRY_RUN_OPT, NETWORK_OPT, GATEWAY_OPT, ADD_RESERVED_IPS_OPT,
317
     MAC_PREFIX_OPT, NETWORK_TYPE_OPT, NETWORK6_OPT, GATEWAY6_OPT,
318
     NOCONFLICTSCHECK_OPT, TAG_ADD_OPT, PRIORITY_OPT, SUBMIT_OPT],
319
    "<network_name>", "Add a new IP network to the cluster"),
320
  "list": (
321
    ListNetworks, ARGS_MANY_NETWORKS,
322
    [NOHDR_OPT, SEP_OPT, FIELDS_OPT, VERBOSE_OPT],
323
    "[<network_id>...]",
324
    "Lists the IP networks in the cluster. The available fields can be shown"
325
    " using the \"list-fields\" command (see the man page for details)."
326
    " The default list is (in order): %s." % utils.CommaJoin(_LIST_DEF_FIELDS)),
327
  "list-fields": (
328
    ListNetworkFields, [ArgUnknown()], [NOHDR_OPT, SEP_OPT], "[fields...]",
329
    "Lists all available fields for networks"),
330
  "info": (
331
    ShowNetworkConfig, ARGS_MANY_NETWORKS, [],
332
    "[<network_name>...]", "Show information about the network(s)"),
333
  "modify": (
334
    SetNetworkParams, ARGS_ONE_NETWORK,
335
    [DRY_RUN_OPT, SUBMIT_OPT, ADD_RESERVED_IPS_OPT, REMOVE_RESERVED_IPS_OPT,
336
     GATEWAY_OPT, MAC_PREFIX_OPT, NETWORK_TYPE_OPT, NETWORK6_OPT, GATEWAY6_OPT,
337
     PRIORITY_OPT],
338
    "<network_name>", "Alters the parameters of a network"),
339
  "connect": (
340
    ConnectNetwork,
341
    [ArgNetwork(min=1, max=1),
342
     ArgChoice(min=1, max=1, choices=constants.NIC_VALID_MODES),
343
     ArgUnknown(min=1, max=1),
344
     ArgGroup()],
345
    [NOCONFLICTSCHECK_OPT, PRIORITY_OPT],
346
    "<network_name> <mode> <link> [<node_group>...]",
347
    "Map a given network to the specified node group"
348
    " with given mode and link (netparams)"),
349
  "disconnect": (
350
    DisconnectNetwork,
351
    [ArgNetwork(min=1, max=1), ArgGroup()],
352
    [NOCONFLICTSCHECK_OPT, PRIORITY_OPT],
353
    "<network_name> [<node_group>...]",
354
    "Unmap a given network from a specified node group"),
355
  "remove": (
356
    RemoveNetwork, ARGS_ONE_NETWORK,
357
    [FORCE_OPT, DRY_RUN_OPT, SUBMIT_OPT, PRIORITY_OPT],
358
    "[--dry-run] <network_id>",
359
    "Remove an (empty) network from the cluster"),
360
  "list-tags": (
361
    ListTags, ARGS_ONE_NETWORK, [],
362
    "<network_name>", "List the tags of the given network"),
363
  "add-tags": (
364
    AddTags, [ArgNetwork(min=1, max=1), ArgUnknown()],
365
    [TAG_SRC_OPT, PRIORITY_OPT, SUBMIT_OPT],
366
    "<network_name> tag...", "Add tags to the given network"),
367
  "remove-tags": (
368
    RemoveTags, [ArgNetwork(min=1, max=1), ArgUnknown()],
369
    [TAG_SRC_OPT, PRIORITY_OPT, SUBMIT_OPT],
370
    "<network_name> tag...", "Remove tags from given network"),
371
}
372

    
373

    
374
def Main():
375
  return GenericMain(commands, override={"tag_type": constants.TAG_NETWORK})