Statistics
| Branch: | Tag: | Revision:

root / scripts / gnt-os @ 2d54e29c

History | View | Annotate | Download (4.8 kB)

1
#!/usr/bin/python
2
#
3

    
4
# Copyright (C) 2006, 2007 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
"""OS scripts related commands"""
22

    
23
# pylint: disable-msg=W0401,W0613,W0614,C0103
24
# W0401: Wildcard import ganeti.cli
25
# W0613: Unused argument, since all functions follow the same API
26
# W0614: Unused import %s from wildcard import (since we need cli)
27
# C0103: Invalid name gnt-os
28

    
29
import sys
30

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

    
36

    
37
def ListOS(opts, args):
38
  """List the valid OSes in the cluster.
39

    
40
  @param opts: the command line options selected by the user
41
  @type args: list
42
  @param args: should be an empty list
43
  @rtype: int
44
  @return: the desired exit code
45

    
46
  """
47
  op = opcodes.OpDiagnoseOS(output_fields=["name", "valid", "variants"],
48
                            names=[])
49
  result = SubmitOpCode(op)
50

    
51
  if not result:
52
    ToStderr("Can't get the OS list")
53
    return 1
54

    
55
  if not opts.no_headers:
56
    headers = {"name": "Name"}
57
  else:
58
    headers = None
59

    
60
  os_names = []
61
  for (name, valid, variants) in result:
62
    if valid:
63
      os_names.extend([[n] for n in CalculateOSNames(name, variants)])
64

    
65
  data = GenerateTable(separator=None, headers=headers, fields=["name"],
66
                       data=os_names, units=None)
67

    
68
  for line in data:
69
    ToStdout(line)
70

    
71
  return 0
72

    
73

    
74
def _OsStatus(status, diagnose):
75
  """Beautifier function for OS status.
76

    
77
  @type status: boolean
78
  @param status: is the OS valid
79
  @type diagnose: string
80
  @param diagnose: the error message for invalid OSes
81
  @rtype: string
82
  @return: a formatted status
83

    
84
  """
85
  if status:
86
    return "valid"
87
  else:
88
    return "invalid - %s" % diagnose
89

    
90
def DiagnoseOS(opts, args):
91
  """Analyse all OSes on this cluster.
92

    
93
  @param opts: the command line options selected by the user
94
  @type args: list
95
  @param args: should be an empty list
96
  @rtype: int
97
  @return: the desired exit code
98

    
99
  """
100
  op = opcodes.OpDiagnoseOS(output_fields=["name", "valid", "variants",
101
                                           "node_status"], names=[])
102
  result = SubmitOpCode(op)
103

    
104
  if not result:
105
    ToStderr("Can't get the OS list")
106
    return 1
107

    
108
  has_bad = False
109

    
110
  for os_name, _, os_variants, node_data in result:
111
    nodes_valid = {}
112
    nodes_bad = {}
113
    nodes_hidden = {}
114
    for node_name, node_info in node_data.iteritems():
115
      nodes_hidden[node_name] = []
116
      if node_info: # at least one entry in the per-node list
117
        (first_os_path, first_os_status, first_os_msg,
118
         first_os_variants) = node_info.pop(0)
119
        if not first_os_variants:
120
          first_os_variants = []
121
        first_os_msg = ("%s (path: %s) [variants: %s]" %
122
                        (_OsStatus(first_os_status, first_os_msg),
123
                         first_os_path, utils.CommaJoin(first_os_variants)))
124
        if first_os_status:
125
          nodes_valid[node_name] = first_os_msg
126
        else:
127
          nodes_bad[node_name] = first_os_msg
128
        for hpath, hstatus, hmsg in node_info:
129
          nodes_hidden[node_name].append("    [hidden] path: %s, status: %s" %
130
                                         (hpath, _OsStatus(hstatus, hmsg)))
131
      else:
132
        nodes_bad[node_name] = "OS not found"
133

    
134
    if nodes_valid and not nodes_bad:
135
      status = "valid"
136
    elif not nodes_valid and nodes_bad:
137
      status = "invalid"
138
      has_bad = True
139
    else:
140
      status = "partial valid"
141
      has_bad = True
142

    
143
    def _OutputPerNodeOSStatus(msg_map):
144
      map_k = utils.NiceSort(msg_map.keys())
145
      for node_name in map_k:
146
        ToStdout("  Node: %s, status: %s", node_name, msg_map[node_name])
147
        for msg in nodes_hidden[node_name]:
148
          ToStdout(msg)
149

    
150
    ToStdout("OS: %s [global status: %s]", os_name, status)
151
    if os_variants:
152
      ToStdout("  Variants: [%s]" % utils.CommaJoin(os_variants))
153
    _OutputPerNodeOSStatus(nodes_valid)
154
    _OutputPerNodeOSStatus(nodes_bad)
155
    ToStdout("")
156

    
157
  return int(has_bad)
158

    
159

    
160
commands = {
161
  'list': (
162
    ListOS, ARGS_NONE, [NOHDR_OPT], "", "Lists all valid OSes on the master"),
163
  'diagnose': (
164
    DiagnoseOS, ARGS_NONE, [], "", "Diagnose all OSes"),
165
  }
166

    
167
if __name__ == '__main__':
168
  sys.exit(GenericMain(commands))