Statistics
| Branch: | Tag: | Revision:

root / scripts / gnt-os @ e16dfb5b

History | View | Annotate | Download (4.5 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

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

    
26
import sys
27

    
28
from ganeti.cli import *
29
from ganeti import opcodes
30
from ganeti import utils
31
from ganeti import constants
32

    
33

    
34
def ListOS(opts, args):
35
  """List the valid OSes in the cluster.
36

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

    
43
  """
44
  op = opcodes.OpDiagnoseOS(output_fields=["name", "valid"], names=[])
45
  result = SubmitOpCode(op)
46

    
47
  if not result:
48
    ToStderr("Can't get the OS list")
49
    return 1
50

    
51
  if not opts.no_headers:
52
    headers = {"name": "Name"}
53
  else:
54
    headers = None
55

    
56
  data = GenerateTable(separator=None, headers=headers, fields=["name"],
57
                       data=[[row[0]] for row in result if row[1]],
58
                       units=None)
59

    
60
  for line in data:
61
    ToStdout(line)
62

    
63
  return 0
64

    
65

    
66
def _OsStatus(status, diagnose):
67
  """Beautifier function for OS status.
68

    
69
  @type status: boolean
70
  @param status: is the OS valid
71
  @type diagnose: string
72
  @param diagnose: the error message for invalid OSes
73
  @rtype: string
74
  @return: a formatted status
75

    
76
  """
77
  if status:
78
    return "valid"
79
  else:
80
    return "invalid - %s" % diagnose
81

    
82
def DiagnoseOS(opts, args):
83
  """Analyse all OSes on this cluster.
84

    
85
  @param opts: the command line options selected by the user
86
  @type args: list
87
  @param args: should be an empty list
88
  @rtype: int
89
  @return: the desired exit code
90

    
91
  """
92
  op = opcodes.OpDiagnoseOS(output_fields=["name", "valid", "variants",
93
                                           "node_status"], names=[])
94
  result = SubmitOpCode(op)
95

    
96
  if not result:
97
    ToStderr("Can't get the OS list")
98
    return 1
99

    
100
  has_bad = False
101

    
102
  for os_name, os_valid, os_variants, node_data in result:
103
    nodes_valid = {}
104
    nodes_bad = {}
105
    nodes_hidden = {}
106
    for node_name, node_info in node_data.iteritems():
107
      nodes_hidden[node_name] = []
108
      if node_info: # at least one entry in the per-node list
109
        (first_os_path, first_os_status, first_os_msg,
110
         first_os_variants) = node_info.pop(0)
111
        if not first_os_variants:
112
          first_os_variants = []
113
        first_os_msg = ("%s (path: %s) [variants: %s]" %
114
                        (_OsStatus(first_os_status, first_os_msg),
115
                         first_os_path, ', '.join(first_os_variants)))
116
        if first_os_status:
117
          nodes_valid[node_name] = first_os_msg
118
        else:
119
          nodes_bad[node_name] = first_os_msg
120
        for hpath, hstatus, hmsg in node_info:
121
          nodes_hidden[node_name].append("    [hidden] path: %s, status: %s" %
122
                                         (hpath, _OsStatus(hstatus, hmsg)))
123
      else:
124
        nodes_bad[node_name] = "OS not found"
125

    
126
    if nodes_valid and not nodes_bad:
127
      status = "valid"
128
    elif not nodes_valid and nodes_bad:
129
      status = "invalid"
130
      has_bad = True
131
    else:
132
      status = "partial valid"
133
      has_bad = True
134

    
135
    def _OutputPerNodeOSStatus(msg_map):
136
      map_k = utils.NiceSort(msg_map.keys())
137
      for node_name in map_k:
138
        ToStdout("  Node: %s, status: %s", node_name, msg_map[node_name])
139
        for msg in nodes_hidden[node_name]:
140
          ToStdout(msg)
141

    
142
    ToStdout("OS: %s [global status: %s]", os_name, status)
143
    if os_variants:
144
      ToStdout("  Variants: [%s]" % ', '.join(os_variants))
145
    _OutputPerNodeOSStatus(nodes_valid)
146
    _OutputPerNodeOSStatus(nodes_bad)
147
    ToStdout("")
148

    
149
  return int(has_bad)
150

    
151

    
152
commands = {
153
  'list': (
154
    ListOS, ARGS_NONE, [NOHDR_OPT], "", "Lists all valid OSes on the master"),
155
  'diagnose': (
156
    DiagnoseOS, ARGS_NONE, [], "", "Diagnose all OSes"),
157
  }
158

    
159
if __name__ == '__main__':
160
  sys.exit(GenericMain(commands))