aboutsummaryrefslogtreecommitdiff
path: root/text_collector_examples/storcli.py
blob: 316629093d0f9713afa7d75580f2b1d270f95c7f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
#!/usr/bin/env python
"""
Script to parse StorCLI's JSON output and expose
MegaRAID health as Prometheus metrics.

Tested against StorCLI 'Ver 1.14.12 Nov 25, 2014'.

StorCLI reference manual:
http://docs.avagotech.com/docs/12352476

Advanced Software Options (ASO) not exposed as metrics currently.

JSON key abbreviations used by StorCLI are documented in the standard command
output, i.e.  when you omit the trailing 'J' from the command.
"""

from __future__ import print_function
import argparse
import json
import os
import subprocess

DESCRIPTION = """Parses StorCLI's JSON output and exposes MegaRAID health as
    Prometheus metrics."""
VERSION = '0.0.1'


def main(args):
    """ main """

    # exporter variables
    metric_prefix = 'megaraid_'
    metric_controller_labels = '{{controller="{}", model="{}"}}'

    data = json.loads(get_storcli_json(args.storcli_path))

    # It appears that the data we need will always be present in the first
    # item in the Controllers array
    status = data['Controllers'][0]

    metrics = {
        'status_code': status['Command Status']['Status Code'],
        'controllers': status['Response Data']['Number of Controllers'],
    }

    for name, value in metrics.iteritems():
        print('# HELP {}{} MegaRAID {}'.format(metric_prefix, name, name.replace('_', ' ')))
        print('# TYPE {}{} gauge'.format(metric_prefix, name))
        print("{}{} {}".format(metric_prefix, name, value))

    controller_info = []
    controller_metrics = {}
    overview = []

    try:
        overview = status['Response Data']['System Overview']
    except KeyError:
        pass

    for controller in overview:
        controller_index = controller['Ctl']
        model = controller['Model']
        controller_info.append(metric_controller_labels.format(controller_index, model))

        controller_metrics = {
            # FIXME: Parse dimmer switch options
            # 'dimmer_switch':          controller['DS'],

            'battery_backup_healthy':   int(controller['BBU'] == 'Opt'),
            'degraded':                 int(controller['Hlth'] == 'Dgd'),
            'drive_groups':             controller['DGs'],
            'emergency_hot_spare':      int(controller['EHS'] == 'Y'),
            'failed':                   int(controller['Hlth'] == 'Fld'),
            'healthy':                  int(controller['Hlth'] == 'Opt'),
            'physical_drives':          controller['PDs'],
            'ports':                    controller['Ports'],
            'scheduled_patrol_read':    int(controller['sPR'] == 'On'),
            'virtual_drives':           controller['VDs'],

            # Reverse StorCLI's logic to make metrics consistent
            'drive_groups_optimal':     int(controller['DNOpt'] == 0),
            'virtual_drives_optimal':   int(controller['VNOpt'] == 0),
            }

    for name, value in controller_metrics.iteritems():
        print('# HELP {}{} MegaRAID {}'.format(metric_prefix, name, name.replace('_', ' ')))
        print('# TYPE {}{} gauge'.format(metric_prefix, name))
        print('{}{}{{controller="{}"}} {}'.format(metric_prefix, name,
                                                  controller_index, value))

    if controller_info:
        print('# HELP {}{} MegaRAID controller info'.format(metric_prefix, 'controller_info'))
        print('# TYPE {}{} gauge'.format(metric_prefix, 'controller_info'))
    for labels in controller_info:
        print('{}{}{} {}'.format(metric_prefix, 'controller_info', labels, 1))


def get_storcli_json(storcli_path):
    """Get storcli output in JSON format."""

    # Check if storcli is installed
    if os.path.isfile(storcli_path) and os.access(storcli_path, os.X_OK):
        storcli_cmd = [storcli_path, 'show', 'all', 'J']
        proc = subprocess.Popen(storcli_cmd, shell=False,
                                stdout=subprocess.PIPE, stderr=subprocess.PIPE)
        output_json = proc.communicate()[0]
    else:
        # Create an empty dummy-JSON where storcli not installed.
        dummy_json = {"Controllers":[{
            "Command Status": {"Status Code": 0, "Status": "Success",
                               "Description": "None"},
            "Response Data": {"Number of Controllers": 0}}]}
        output_json = json.dumps(dummy_json)

    return output_json

if __name__ == "__main__":
    PARSER = argparse.ArgumentParser(description=DESCRIPTION,
                                     formatter_class=argparse.ArgumentDefaultsHelpFormatter)
    PARSER.add_argument('--storcli_path',
                        default='/opt/MegaRAID/storcli/storcli64',
                        help='path to StorCLi binary')
    PARSER.add_argument('--version',
                        action='version',
                        version='%(prog)s {}'.format(VERSION))
    ARGS = PARSER.parse_args()

    main(ARGS)