-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpowermanager.py
executable file
·148 lines (118 loc) · 4.33 KB
/
powermanager.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
#!/usr/bin/env python
from __future__ import print_function
import sys
import argparse
import os.path
import urllib2
from operator import methodcaller
from time import sleep
"""Manages the power for the IP Power 9258.
You are meant to create symlinks to this file for each port.
"""
class PowerManagerException(RuntimeError):
def __init__(self, reason):
super(PowerManagerException, self).__init__()
self.reason = reason
class PowerManager(object):
"""Manages the power for a single port of the IP Power 9258."""
def __init__(self, host, username, password, port):
self.host = host
self.username = username
self.password = password
self.port = port
"""Set-up urllib2"""
basic_handler = urllib2.HTTPBasicAuthHandler()
basic_handler.add_password(realm='IP9258',
uri='http://{}/'.format(self.host),
user=self.username,
passwd=self.password)
opener = urllib2.build_opener(basic_handler)
urllib2.install_opener(opener)
def __call_url(self, full_url):
try:
url_res = urllib2.urlopen(full_url)
txt = url_res.read()
url_res.close()
return txt
except urllib2.URLError as e:
raise PowerManagerException('Error opening URL: {}'.format(
e.reason))
def get_state(self):
full_url = 'http://{}/Set.cmd?CMD=GetPower'.format(self.host)
return self.__call_url(full_url)
def __call_url_power(self, port_value):
port_state = '{}={}'.format(self.port, port_value)
full_url = 'http://{}/Set.cmd?CMD=SetPower+{}'.format(self.host,
port_state)
txt = self.__call_url(full_url)
if -1 == txt.find(port_state):
"""Response should be the port_state we wanted, plus HTML."""
raise PowerManagerException('Bad response: {}'.format(txt))
def power_on(self):
print("Power on.")
self.__call_url_power('1')
def power_off(self):
print("Power off.")
self.__call_url_power('0')
def power_cycle(self):
self.power_off()
sleep(1)
self.power_on()
@classmethod
def dispatch_action(cls, power_manager, action):
ACTION_CALLBACKS = {'get': methodcaller('get_state'),
'on': methodcaller('power_on'),
'off': methodcaller('power_off'),
'cycle': methodcaller('power_cycle')}
actions = ACTION_CALLBACKS.keys()
try:
action_method = ACTION_CALLBACKS[action]
except KeyError:
raise ValueError('Invalid action: {}'.format(action))
ret = action_method(power_manager)
if ret is not None:
print(ret)
def parse_args():
p = argparse.ArgumentParser(description='Control power.')
p.add_argument('action', help='get|on|off|cycle')
return p.parse_args(sys.argv[1:])
def find_port():
"""Finds port to use based on script name.
Assumes the script is named <SOMETHING>-power.py, where the part before the
dash is a dict key.
"""
POWER_PORTS = { 'pound': 'p61', 'pandaboard': 'p62', 'odroidx': 'p63' }
script_name = os.path.basename(sys.argv[0])
dash_pos = script_name.find('-')
if -1 == dash_pos:
raise ValueError('No dash found in script name: {}'.format(script_name))
port_key = script_name[:dash_pos]
try:
return (port_key, POWER_PORTS[port_key])
except KeyError:
raise KeyError('Port key not found in dict: {}'.format(port_key))
def read_password():
fname = os.path.expanduser('~/.poundpower')
with open(fname) as f:
password = f.read().strip()
return password
def main():
HOST = 'pound-pwr.cs.unc.edu'
USERNAME = 'admin'
try:
password = read_password()
except IOError as e:
print('Error reading password file: {}'.format(e), file=sys.stderr)
sys.exit(1)
args = parse_args()
machine, port = find_port()
print('Operating on port {} machine {}.'.format(port, machine))
mgr = PowerManager(HOST, USERNAME, password, port)
try:
PowerManager.dispatch_action(mgr, args.action)
except PowerManagerException as e:
print('Exception: {}'.format(e.reason), file=sys.stderr)
sys.exit(1)
sys.exit(0)
if __name__ == '__main__':
main()