-
Notifications
You must be signed in to change notification settings - Fork 85
/
Copy pathvmware_host.py
278 lines (229 loc) · 11.3 KB
/
vmware_host.py
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
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
#!/usr/bin/env python
"""vmware_host - Exports JSON files with CPU and RAM data for VMware ESX hosts
# Requires VMware Python SDK: pyvmomi
# https://code.google.com/p/pysphere/
# pip install pyvmomi
"""
from credentials import VMWARE_VCENTER_USERNAME
from credentials import VMWARE_VCENTER_PASSWORD
import operator
import time
import json
import logging.config
from pyVim.connect import SmartConnect
from pchelper import collect_properties
from pchelper import get_container_view
import pyVmomi
import ssl
__author__ = 'scott@flakshack.com (Scott Vintinner)'
# =================================SETTINGS======================================
# VCenter Servers
VCENTER_SERVERS = [
{"name": "vcenter", "username": VMWARE_VCENTER_USERNAME, "password": VMWARE_VCENTER_PASSWORD},
{"name": "view-vcenter", "username": VMWARE_VCENTER_USERNAME, "password": VMWARE_VCENTER_PASSWORD}
]
SAMPLE_INTERVAL = 60
MAX_DATAPOINTS = 30
MAX_HOST_RESULTS = 11
# ===============================================================================
class MonitorJSON:
"""This is a simple class passed to Monitor threads so we can access the current JSON data in that thread"""
def __init__(self):
host_data = [{"name": "----", "status": 0, "cpu": [0, 0, 0, 0], "ram": 0}]
self.json = json.dumps({"hosts": host_data}, indent=4)
self.vcenter_servers = VCENTER_SERVERS
class ESXHost:
all_hosts = [] # Static array containing all hosts
def __init__(self, managed_object_reference, name):
self.managed_object_reference = managed_object_reference
self.name = name
self.status = 0
self.cpu_datapoints = []
self.ram = 0
self.ram_percent = 0
self.relative_weight = 1
self.__class__.all_hosts.append(self) # Add self to static array
def update_relative_weight(self):
"""The relative weight is used to determine how much we want to see the data of this Host."""
self.relative_weight = 1
# Add up all of the historical cpu datapoints (higher CPU = more weight)
for i in self.cpu_datapoints:
self.relative_weight += i
# Multiply by the status value (so VMs with red alarm have most weight)
self.relative_weight *= (self.status * 10)
@classmethod
def find_by_name(cls, managed_object_reference, name):
for host in cls.all_hosts:
if host.name == name:
return host
# if not found, create one and return it instead
return ESXHost(managed_object_reference, name)
def hostname_from_fqdn(fqdn):
"""Will take a fully qualified domain name and return only the hostname."""
split_fqdn = fqdn.split('.', 1) # Split fqdn at periods, but only bother doing first split
return split_fqdn[0]
def connect_vcenter(vcenter_server, vcenter_username, vcenter_password):
"""This function will connect to the specified vCenter server."""
logger = logging.getLogger(__name__)
# Disable certificate verification otherwise it will error
ssl_context = ssl.SSLContext(ssl.PROTOCOL_TLS)
ssl_context.verify_mode = ssl.CERT_NONE
server_instance = None
try:
logger.debug("Connecting to: " + vcenter_server)
server_instance = SmartConnect(host=vcenter_server, user=vcenter_username,
pwd=vcenter_password, sslContext=ssl_context)
server_instance._stub.connectionPoolTimeout = -1 # Turn the connection timeout off (default 900 sec)
except Exception as error:
logger.error("Error connecting to " + vcenter_server + str(error))
return server_instance
def update_host_data(server):
"""This function is called to update the HOST data for the specified vcenter server"""
logger = logging.getLogger(__name__)
# API: HostSystem -
# https://pubs.vmware.com/vsphere-51/index.jsp#com.vmware.wssdk.apiref.doc/vim.HostSystem.html
# summary.overallStatus: general "health" value: gray, green, red, yellow
# summary.quickStats.overallCpuUsage: Aggregated CPU usage across all cores on the host in MHz.
# summary.quickStats.overallMemoryUsage: Physical memory usage on the host in MB.
# hardware.memorySize: Total available RAM in bytes.
try:
# Fast way of getting Host properties using PropertyCollector
# https://github.com/vmware/pyvmomi/blob/master/docs/vmodl/query/PropertyCollector.rst
logger.debug("Getting PropertyCollector for " + server["name"])
container_view = get_container_view(server["conn"], obj_type=[pyVmomi.vim.HostSystem])
query = [
"name",
"summary.overallStatus",
"summary.quickStats.overallCpuUsage",
"summary.quickStats.overallMemoryUsage",
"hardware.memorySize"
]
props = collect_properties(server["conn"], container_view,
pyVmomi.vim.HostSystem, query, include_mors=True)
except Exception as error:
logger.error("Error collecting VMware Host data from " + server["name"] + str(error))
raise
# Loop through all of the ESX servers in props
for prop_set in props:
host_mor = prop_set["obj"] # Managed object reference
host_name = prop_set["name"]
host_name = hostname_from_fqdn(host_name) # trim out the domain name
host_status = prop_set["summary.overallStatus"]
host_cpu = prop_set["summary.quickStats.overallCpuUsage"]
host_ram = prop_set["summary.quickStats.overallMemoryUsage"]
host_ram_max = prop_set["hardware.memorySize"]
host_ram_max = int(host_ram_max / 1024 / 1024) # Convert to Megabytes to match overallMemoryUsage
host_ram_percent = int((host_ram / host_ram_max) * 100) # Calculate RAM percentage
logger.debug(host_name + " RAM: " + str(host_ram_percent) + "% CPU: " + str(host_cpu))
# Convert ram into Gigabytes and round to 1 decimal place
host_ram = int(host_ram / 1024)
# Find/Create this host in our list of hosts and update the object's data
host = ESXHost.find_by_name(host_mor, host_name)
if host_status == "green":
host.status = 1
elif host_status == "yellow":
host.status = 2
elif host_status == "red":
host.status = 3
else:
host.status = 0
# Add the raw data to the ESXHost object
# For RAM datapoints, we want to do a bar graph, so only include the current value
host.ram = host_ram
host.ram_percent = host_ram_percent
# For CPU datapoints, we want to do a line graph, so we need a history
if len(host.cpu_datapoints) == 0: # first time through, initialize the list
host.cpu_datapoints = [host_cpu]
else:
host.cpu_datapoints.append(host_cpu)
# If we already have the max number of datapoints in our list, delete the oldest item
if len(host.cpu_datapoints) >= MAX_DATAPOINTS:
del(host.cpu_datapoints[0])
# Update ranking value of this Host to determine if we should show it
host.update_relative_weight()
#
#
#
#
def generate_json(vmware_monitor):
"""This is the main function. It will connect to the vCenter server, obtain perf data and output files"""
logger = logging.getLogger("vmware_host")
# Process each vcenter server
for server in vmware_monitor.vcenter_servers:
logger.debug("Starting " + server["name"])
if "conn" not in server:
logger.debug(server["name"] + " not currently connected.")
server["conn"] = connect_vcenter(server["name"], server["username"], server["password"])
if server["conn"] is None:
logger.warning("Unable to connect to " + server["name"] + " will retry in " +
str(SAMPLE_INTERVAL) + " seconds.")
vmware_monitor.json = json.dumps({"vms": [{"error": "Unable to connect to " + server["name"] +
" will retry in " + str(SAMPLE_INTERVAL) +
" seconds."}]}, indent=4)
return vmware_monitor # Could not connect so return
# Final test if we're connected
try:
if server["conn"].content.sessionManager.currentSession:
logger.debug("Connected to " + server["name"] + " at " +
str(server["conn"].content.sessionManager.currentSession.loginTime))
except Exception as error:
logger.error("Final test: Error connecting to " + server["name"] + str(error))
vmware_monitor.json = json.dumps({"vms": [{"error": "Unable to connect to " + server["name"] +
" will retry in " + str(SAMPLE_INTERVAL) +
" seconds."}]}, indent=4)
return vmware_monitor # Could not connect so return
# Update all the ESX host objects for the specified vCenter server
try:
update_host_data(server)
except Exception as error:
logger.error("Error updating data from " + server["name"] + str(error))
vmware_monitor.json = json.dumps({"vms": [{"error": "Error updating data from " + server["name"] +
" will retry in " + str(SAMPLE_INTERVAL) +
" seconds."}]}, indent=4)
return vmware_monitor
# Sort by relative weight
ESXHost.all_hosts.sort(key=operator.attrgetter('relative_weight'), reverse=True)
# We have all the data we need, so format and set output
host_data = []
for i, host in enumerate(ESXHost.all_hosts):
# Generate the data sequence
host_data.append({
"name": host.name,
"status": host.status,
"cpu": host.cpu_datapoints,
"ram": host.ram,
"ram_percent": host.ram_percent
})
if i >= (MAX_HOST_RESULTS - 1): # Don't return more hosts than we need
break
vmware_monitor.json = json.dumps({"hosts": host_data})
if __debug__:
logger.debug(vmware_monitor.json)
# ======================================================
# __main__
#
# If you run this module by itself, it will instantiate
# the MonitorJSON class and start an infinite loop
# printing data.
# ======================================================
#
if __name__ == '__main__':
# When run by itself, we need to create the logger object (which is normally created in webserver.py)
try:
f = open("log_settings.json", 'rt')
log_config = json.load(f)
f.close()
logging.config.dictConfig(log_config)
except FileNotFoundError as e:
print("Log configuration file not found: " + str(e))
logging.basicConfig(level=logging.DEBUG) # fallback to basic settings
except json.decoder.JSONDecodeError as e:
print("Error parsing logger config file: " + str(e))
raise
monitor = MonitorJSON()
while True:
main_logger = logging.getLogger(__name__)
generate_json(monitor)
# Wait X seconds for the next iteration
main_logger.debug("Waiting for " + str(SAMPLE_INTERVAL) + " seconds")
time.sleep(SAMPLE_INTERVAL)