-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathseattleclearinghouse_xmlrpc.py
582 lines (472 loc) · 18.9 KB
/
seattleclearinghouse_xmlrpc.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
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
"""
<Program>
seattleclearinghouse_xmlrpc.py
<Started>
6/28/2009
<Author>
Jason Chen
Justin Samuel
<Purpose>
A client library for communicating with the SeattleClearinghouse XMLRPC Server.
Your Python scripts can import this library, create an instance of the
SeattleClearinghouseClient class, then call methods on the object to perform XMLRPC
calls through the SeattleClearinghouse XMLRPC API.
Full tutorials on using this library, see:
https://seattle.cs.washington.edu/wiki/SeattleGeniClientLib
In order to perform secure SSL communication with SeattleClearinghouse:
* You must have M2Crypto installed.
* You must set the value of CA_CERTIFICATES_FILE to the location of a PEM
file containing CA certificates that you trust. If you don't know where
this is on your own system, you can download this file from a site you
trust. One such place to download this file from is:
http://curl.haxx.se/ca/cacert.pem
If you can't fulfill the above requirements, you can still use this client with
XMLRPC servers that use https but you will be vulnerable to a man-in-the-middle
attack. To enable this insecure mode, include the argument:
allow_ssl_insecure=True
when creating a SeattleClearinghouseClient instance.
<Notes>
All methods of the client class may raise the following errors in addition to
any others described in the method's docstring:
CommunicationError
AuthenticationError
InvalidRequestError
InternalError
The safest way to be certain to catch any of these errors is to the catch
their base class:
SeattleClearinghouseError
"""
import os
import socket
import xmlrpclib
# If a user does not provide us with an API key, we'll need to load
# their private key instead.
from repyportability import add_dy_support
add_dy_support(locals())
# Location of a file containing one or more PEM-encoded CA certificates
# concatenated together. This is required if using allow_ssl_insecure=False.
# By default it looks for a cacert.pem file in the same directory as this
# python module is in.
DEFAULT_CA_CERTIFICATES_FILE = os.path.join(os.path.dirname(__file__), "cacert.pem")
# The location of the SeattleClearinghouse XMLRPC server to use.
DEFAULT_XMLRPC_URL = "https://seattleclearinghouse.poly.edu/xmlrpc/"
# SeattleClearinghouse XMLRPC Fault Code Constants
FAULTCODE_INTERNALERROR = 100
FAULTCODE_AUTHERROR = 101
FAULTCODE_INVALIDREQUEST = 102
FAULTCODE_NOTENOUGHCREDITS = 103
FAULTCODE_UNABLETOACQUIRE = 105
class SeattleClearinghouseClient(object):
"""
Implementation of an XMLRPC client for communicating with a SeattleClearinghouse
server. This uses the public API described at:
https://seattle.cs.washington.edu/wiki/SeattleGeniApi
"""
def __init__(self, username, api_key=None, private_key_string=None,
xmlrpc_url=None,
allow_ssl_insecure=None,
ca_certs_file=None):
if xmlrpc_url is None:
xmlrpc_url = DEFAULT_XMLRPC_URL
if allow_ssl_insecure is None:
allow_ssl_insecure = False
if ca_certs_file is None:
ca_certs_file = DEFAULT_CA_CERTIFICATES_FILE
if not isinstance(username, basestring):
raise TypeError("username must be a string")
if api_key is not None:
if not isinstance(api_key, basestring):
raise TypeError("api_key must be a string")
else:
if not private_key_string:
raise TypeError("private_key_string must be provided if api_key is not")
if not isinstance(private_key_string, basestring):
raise TypeError("private_key_string must be a string")
if not isinstance(xmlrpc_url, basestring):
raise TypeError("xmlrpc_url must be a string")
if not isinstance(allow_ssl_insecure, bool):
raise TypeError("allow_ssl_insecure must be True or False")
if not isinstance(ca_certs_file, basestring):
raise TypeError("ca_certs_file must be a string")
if allow_ssl_insecure:
self.proxy = xmlrpclib.Server(xmlrpc_url)
else:
ssl_transport = _get_ssl_transport(ca_certs_file)
self.proxy = xmlrpclib.Server(xmlrpc_url, transport=ssl_transport)
if not api_key:
api_key = self._get_api_key(username, private_key_string)
self.auth = {'username':username, 'api_key':api_key}
def _get_api_key(self, username, private_key_string):
# Normally we try not to import modules anywhere but globally,
# but I'd like to keep this xmlrpc client usable without repy files
# available when the user provides their api key and doesn't require
# it to be retrieved.
try:
dy_import_module_symbols("rsa.r2py")
except ImportError, e:
raise SeattleClearinghouseError("Unable to get API key from SeattleClearinghouse " +
"because a required python or repy module " +
"cannot be found:" + str(e))
# This will raise a ValueError if the private key is not valid.
private_key_dict = rsa_string_to_privatekey(private_key_string)
encrypted_data = self.proxy.get_encrypted_api_key(username)
decrypted_data = rsa_decrypt(encrypted_data, private_key_dict)
split_data = decrypted_data.split("!")
# The encrypted data has 20 bytes of random data followed by a "!" which
# is then followed by the actual API key. If the private key was the wrong
# key, we will end up with garbage data (if it was an invalid key, it
# might be empty, though). So, we're going to make the fairly safe
# assumption that the odds of a random decryption with the wrong key
# resulting in data that starts with 20 bytes which aren't exclamation
# marks followed by a single exclamation mark and no others is pretty low.
if len(split_data) != 2 or len(split_data[0]) != 20:
raise AuthenticationError("The provided private key does not appear " +
"to correspond to this account's public key: " +
"encrypted API key could not be decrypted.")
api_key = split_data[1]
return api_key
def _do_call(self, function, *args):
try:
return function(self.auth, *args)
except socket.error, err:
raise CommunicationError("XMLRPC failed: " + str(err))
except xmlrpclib.Fault, fault:
if fault.faultCode == FAULTCODE_AUTHERROR:
raise AuthenticationError
elif fault.faultCode == FAULTCODE_INVALIDREQUEST:
raise InvalidRequestError(fault.faultString)
elif fault.faultCode == FAULTCODE_NOTENOUGHCREDITS:
raise NotEnoughCreditsError(fault.faultString)
elif fault.faultCode == FAULTCODE_UNABLETOACQUIRE:
raise UnableToAcquireResourcesError(fault.faultString)
else:
raise InternalError(fault.faultString)
def _do_pwauth_call(self, function, password, *args):
"""For use by calls that require a password rather than an api key."""
pwauth = {'username':self.auth['username'], 'password':password}
try:
return function(pwauth, *args)
except socket.error, err:
raise CommunicationError("XMLRPC failed: " + str(err))
except xmlrpclib.Fault, fault:
if fault.faultCode == FAULTCODE_AUTHERROR:
raise AuthenticationError
elif fault.faultCode == FAULTCODE_INVALIDREQUEST:
raise InvalidRequestError(fault.faultString)
elif fault.faultCode == FAULTCODE_NOTENOUGHCREDITS:
raise NotEnoughCreditsError(fault.faultString)
elif fault.faultCode == FAULTCODE_UNABLETOACQUIRE:
raise UnableToAcquireResourcesError(fault.faultString)
else:
raise InternalError(fault.faultString)
def acquire_lan_resources(self, count):
"""
<Purpose>
Acquire LAN vessels.
<Arguments>
count
The number of vessels to acquire.
<Exceptions>
The common exceptions described in the module comments, as well as:
SeattleClearinghouseNotEnoughCredits
If the account does not have enough available vessel credits to fulfill
the request.
<Side Effects>
If successful, 'count' LAN vessels have been acquired for the account.
<Returns>
A list of vessel handles of the acquired vessels.
"""
return self.acquire_resources('lan', count)
def acquire_wan_resources(self, count):
"""
<Purpose>
Acquire WAN vessels.
<Arguments>
count
The number of vessels to acquire.
<Exceptions>
The common exceptions described in the module comments, as well as:
SeattleClearinghouseNotEnoughCredits
If the account does not have enough available vessel credits to fulfill
the request.
<Side Effects>
If successful, 'count' WAN vessels have been acquired for the account.
<Returns>
A list of vessel handles of the acquired vessels.
"""
return self.acquire_resources('wan', count)
def acquire_nat_resources(self, count):
"""
<Purpose>
Acquire NAT vessels.
<Arguments>
count
The number of vessels to acquire.
<Exceptions>
The common exceptions described in the module comments, as well as:
SeattleClearinghouseNotEnoughCredits
If the account does not have enough available vessel credits to fulfill
the request.
<Side Effects>
If successful, 'count' NAT vessels have been acquired for the account.
<Returns>
A list of vessel handles of the acquired vessels.
"""
return self.acquire_resources('nat', count)
def acquire_random_resources(self, count):
"""
<Purpose>
Acquire vessels (they can be LAN, WAN, NAT, or any combination of these).
<Arguments>
count
The number of vessels to acquire.
<Exceptions>
The common exceptions described in the module comments, as well as:
SeattleClearinghouseNotEnoughCredits
If the account does not have enough available vessel credits to fulfill
the request.
<Side Effects>
If successful, 'count' vessels have been acquired for the account.
<Returns>
A list of vessel handles of the acquired vessels.
"""
return self.acquire_resources('random', count)
def acquire_resources(self, res_type, count):
"""
<Purpose>
Acquire vessels.
<Arguments>
res_type
A string describing the type of vessels to acquire.
count
The number of vessels to acquire.
<Exceptions>
The common exceptions described in the module comments, as well as:
SeattleClearinghouseNotEnoughCredits
If the account does not have enough available vessel credits to fulfill
the request.
<Side Effects>
If successful, 'count' vessels have been acquired for the account.
<Returns>
A list of vessel handles of the acquired vessels.
"""
if not isinstance(res_type, basestring):
raise TypeError("res_type must be a string")
if type(count) not in [int, long]:
raise TypeError("count must be an integer")
rspec = {'rspec_type':res_type, 'number_of_nodes':count}
return self._do_call(self.proxy.acquire_resources, rspec)
def acquire_specific_vessels(self, handlelist):
"""
<Purpose>
Attempt to acquire specific vessels.
<Arguments>
handlelist
A list of vessel handles.
<Exceptions>
The common exceptions described in the module comments, as well as:
SeattleClearinghouseNotEnoughCredits
If the account does not have enough available vessel credits to fulfill
the request.
<Side Effects>
If successful, zero or more vessels from handlelist have been acquired.
<Returns>
A list of vessel handles of the acquired vessels.
"""
_validate_handle_list(handlelist)
return self._do_call(self.proxy.acquire_specific_vessels, handlelist)
def release_resources(self, handlelist):
"""
<Purpose>
Release vessels.
<Arguments>
handlelist
A list of handles as returned by acquire_vessels() or found in the
'handle' key of the dictionaries returned by get_resource_info().
<Exceptions>
The common exceptions described in the module comments.
<Side Effects>
If successful, the vessels in handlelist have been released. If not
successful, it is possible that a partial set of the vessels was
released.
<Returns>
None
"""
_validate_handle_list(handlelist)
return self._do_call(self.proxy.release_resources, handlelist)
def renew_resources(self, handlelist):
"""
<Purpose>
Renew vessels.
<Arguments>
handlelist
A list of handles as returned by acquire_vessels() or found in the
'handle' key of the dictionaries returned by get_resource_info().
<Exceptions>
The common exceptions described in the module comments, as well as:
SeattleClearinghouseNotEnoughCredits
If the account is currently over its vessel credit limit, then vessels
cannot be renewed until the account is no longer over its credit limit.
<Side Effects>
If successful, the vessels in handlelist have been renewed. If not
successful, it is possible that a partial set of the vessels was
renewed.
<Returns>
None
"""
_validate_handle_list(handlelist)
return self._do_call(self.proxy.renew_resources, handlelist)
def get_resource_info(self):
"""
<Purpose>
Obtain information about acquired vessels.
<Arguments>
None
<Exceptions>
The common exceptions described in the module comments, as well as:
<Side Effects>
None
<Returns>
A list of dictionaries, where each dictionary describes a vessel that
is currently acquired by the account.
"""
return self._do_call(self.proxy.get_resource_info)
def get_account_info(self):
"""
<Purpose>
Obtain information about the account.
<Arguments>
None
<Exceptions>
The common exceptions described in the module comments, as well as:
<Side Effects>
None
<Returns>
A dictionary with information about the account.
"""
return self._do_call(self.proxy.get_account_info)
def get_public_key(self):
"""
<Purpose>
Obtain the public key of the account.
<Arguments>
None
<Exceptions>
The common exceptions described in the module comments, as well as:
None
<Side Effects>
None
<Returns>
A string containing the public key of the account.
"""
return self._do_call(self.proxy.get_public_key)
def set_public_key(self, password, pubkeystring):
"""
<Purpose>
Set the public key of the account.
<Arguments>
password
The account password. This is required because changing the public
key of the account cannot be done with just the api key.
pubkeystring
A string representing the new public key to be set for the account.
<Exceptions>
The common exceptions described in the module comments, as well as:
InvalidRequestError
If the pubkey is invalid.
<Side Effects>
The public key of the account is changed and will be updated on all
vessels the account has acquired.
<Returns>
None
"""
self._do_pwauth_call(self.proxy.set_public_key, password, pubkeystring)
def regenerate_api_key(self, password):
"""
<Purpose>
Generate a new API key for the account..
<Arguments>
password
The account password. This is required because changing the api
key of the account cannot be done with just the current api key.
<Exceptions>
The common exceptions described in the module comments, as well as:
None
<Side Effects>
The account's api key has been changed.
<Returns>
The new api key for the account.
"""
api_key = self._do_pwauth_call(self.proxy.regenerate_api_key, password)
self.auth['api_key'] = api_key
return api_key
def _validate_handle_list(handlelist):
"""
Raise a TypeError or ValueError if handlelist is not a non-empty list of
string.
"""
if not isinstance(handlelist, list):
raise TypeError("Invalid data type for handle list: " +
str(type(handlelist)))
for handle in handlelist:
if not isinstance(handle, basestring):
raise TypeError("Invalid data type for a handle in the handle list: " +
str(type(handle)))
if not handlelist:
raise ValueError("Given handlelist is empty.")
def _get_ssl_transport(ca_certs_file):
"""
Returns an object usable as the transport for an xmlrpclib proxy. This will
be an M2Crypto.m2xmlrpclib.SSL_Transport that has been configured with a
context that has the ca_certs_file loaded, will not allow SSLv2, and will
reject certificate names that don't match the hostname.
"""
try:
import M2Crypto
except ImportError, err:
raise ImportError("In order to use the SeattleClearinghouse XMLRPC client with " +
"allow_ssl_insecure=False, you need M2Crypto " +
"installed. " + str(err))
# We don't define this class until here because otherwise M2Crypto may not
# be available.
class M2CryptoSSLTransport(M2Crypto.m2xmlrpclib.SSL_Transport):
def request(self, host, handler, request_body, verbose=0):
if host.find(":") == -1:
host = host + ":443"
return M2Crypto.m2xmlrpclib.SSL_Transport.request(self, host, handler,
request_body, verbose)
ctx = M2Crypto.SSL.Context("sslv3")
ctx.set_verify(M2Crypto.SSL.verify_peer |
M2Crypto.SSL.verify_fail_if_no_peer_cert, depth=9)
if ctx.load_verify_locations(ca_certs_file) != 1:
raise SeattleClearinghouseError("No CA certs found in file: " + ca_certs_file)
return M2CryptoSSLTransport(ctx)
class SeattleClearinghouseError(Exception):
"""Base class for exceptions raised by the SeattleClearinghouseClient."""
class CommunicationError(SeattleClearinghouseError):
"""
Indicates that XMLRPC communication failed.
"""
class InternalError(SeattleClearinghouseError):
"""
Indicates an unexpected error occurred, probably either a bug in this
client or a bug in SeattleClearinghouse.
"""
class AuthenticationError(SeattleClearinghouseError):
"""Indicates an authentication failure (invalid username and/or API key)."""
def __init__(self, msg=None):
if msg is None:
msg = "Authentication failed. Invalid username and/or API key."
SeattleClearinghouseError.__init__(self, msg)
class InvalidRequestError(SeattleClearinghouseError):
"""Indicates that the request is invalid."""
class NotEnoughCreditsError(SeattleClearinghouseError):
"""
Indicates that the requested operation requires more vessel credits to
be available then the account currently has.
"""
class UnableToAcquireResourcesError(SeattleClearinghouseError):
"""
Indicates that the requested operation failed because SeattleClearinghouse was unable
to acquire the requested resources.
"""