Mercurial > repos > guerler > springsuite
comparison planemo/lib/python3.7/site-packages/boto/utils.py @ 0:d30785e31577 draft
"planemo upload commit 6eee67778febed82ddd413c3ca40b3183a3898f1"
author | guerler |
---|---|
date | Fri, 31 Jul 2020 00:18:57 -0400 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:d30785e31577 |
---|---|
1 # Copyright (c) 2006-2012 Mitch Garnaat http://garnaat.org/ | |
2 # Copyright (c) 2010, Eucalyptus Systems, Inc. | |
3 # Copyright (c) 2012 Amazon.com, Inc. or its affiliates. | |
4 # All rights reserved. | |
5 # | |
6 # Permission is hereby granted, free of charge, to any person obtaining a | |
7 # copy of this software and associated documentation files (the | |
8 # "Software"), to deal in the Software without restriction, including | |
9 # without limitation the rights to use, copy, modify, merge, publish, dis- | |
10 # tribute, sublicense, and/or sell copies of the Software, and to permit | |
11 # persons to whom the Software is furnished to do so, subject to the fol- | |
12 # lowing conditions: | |
13 # | |
14 # The above copyright notice and this permission notice shall be included | |
15 # in all copies or substantial portions of the Software. | |
16 # | |
17 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | |
18 # OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABIL- | |
19 # ITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT | |
20 # SHALL THE AUTHOR BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | |
21 # WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
22 # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS | |
23 # IN THE SOFTWARE. | |
24 | |
25 # | |
26 # Parts of this code were copied or derived from sample code supplied by AWS. | |
27 # The following notice applies to that code. | |
28 # | |
29 # This software code is made available "AS IS" without warranties of any | |
30 # kind. You may copy, display, modify and redistribute the software | |
31 # code either by itself or as incorporated into your code; provided that | |
32 # you do not remove any proprietary notices. Your use of this software | |
33 # code is at your own risk and you waive any claim against Amazon | |
34 # Digital Services, Inc. or its affiliates with respect to your use of | |
35 # this software code. (c) 2006 Amazon Digital Services, Inc. or its | |
36 # affiliates. | |
37 | |
38 """ | |
39 Some handy utility functions used by several classes. | |
40 """ | |
41 | |
42 import subprocess | |
43 import time | |
44 import logging.handlers | |
45 import boto | |
46 import boto.provider | |
47 import tempfile | |
48 import random | |
49 import smtplib | |
50 import datetime | |
51 import re | |
52 import email.mime.multipart | |
53 import email.mime.base | |
54 import email.mime.text | |
55 import email.utils | |
56 import email.encoders | |
57 import gzip | |
58 import threading | |
59 import locale | |
60 from boto.compat import six, StringIO, urllib, encodebytes | |
61 | |
62 from contextlib import contextmanager | |
63 | |
64 from hashlib import md5, sha512 | |
65 _hashfn = sha512 | |
66 | |
67 from boto.compat import json | |
68 | |
69 try: | |
70 from boto.compat.json import JSONDecodeError | |
71 except ImportError: | |
72 JSONDecodeError = ValueError | |
73 | |
74 # List of Query String Arguments of Interest | |
75 qsa_of_interest = ['acl', 'cors', 'defaultObjectAcl', 'location', 'logging', | |
76 'partNumber', 'policy', 'requestPayment', 'torrent', | |
77 'versioning', 'versionId', 'versions', 'website', | |
78 'uploads', 'uploadId', 'response-content-type', | |
79 'response-content-language', 'response-expires', | |
80 'response-cache-control', 'response-content-disposition', | |
81 'response-content-encoding', 'delete', 'lifecycle', | |
82 'tagging', 'restore', | |
83 # storageClass is a QSA for buckets in Google Cloud Storage. | |
84 # (StorageClass is associated to individual keys in S3, but | |
85 # having it listed here should cause no problems because | |
86 # GET bucket?storageClass is not part of the S3 API.) | |
87 'storageClass', | |
88 # websiteConfig is a QSA for buckets in Google Cloud | |
89 # Storage. | |
90 'websiteConfig', | |
91 # compose is a QSA for objects in Google Cloud Storage. | |
92 'compose', | |
93 # billing is a QSA for buckets in Google Cloud Storage. | |
94 'billing', | |
95 # userProject is a QSA for requests in Google Cloud Storage. | |
96 'userProject', | |
97 # encryptionConfig is a QSA for requests in Google Cloud | |
98 # Storage. | |
99 'encryptionConfig'] | |
100 | |
101 | |
102 _first_cap_regex = re.compile('(.)([A-Z][a-z]+)') | |
103 _number_cap_regex = re.compile('([a-z])([0-9]+)') | |
104 _end_cap_regex = re.compile('([a-z0-9])([A-Z])') | |
105 | |
106 | |
107 def unquote_v(nv): | |
108 if len(nv) == 1: | |
109 return nv | |
110 else: | |
111 return (nv[0], urllib.parse.unquote(nv[1])) | |
112 | |
113 | |
114 def canonical_string(method, path, headers, expires=None, | |
115 provider=None): | |
116 """ | |
117 Generates the aws canonical string for the given parameters | |
118 """ | |
119 if not provider: | |
120 provider = boto.provider.get_default() | |
121 interesting_headers = {} | |
122 for key in headers: | |
123 lk = key.lower() | |
124 if headers[key] is not None and \ | |
125 (lk in ['content-md5', 'content-type', 'date'] or | |
126 lk.startswith(provider.header_prefix)): | |
127 interesting_headers[lk] = str(headers[key]).strip() | |
128 | |
129 # these keys get empty strings if they don't exist | |
130 if 'content-type' not in interesting_headers: | |
131 interesting_headers['content-type'] = '' | |
132 if 'content-md5' not in interesting_headers: | |
133 interesting_headers['content-md5'] = '' | |
134 | |
135 # just in case someone used this. it's not necessary in this lib. | |
136 if provider.date_header in interesting_headers: | |
137 interesting_headers['date'] = '' | |
138 | |
139 # if you're using expires for query string auth, then it trumps date | |
140 # (and provider.date_header) | |
141 if expires: | |
142 interesting_headers['date'] = str(expires) | |
143 | |
144 sorted_header_keys = sorted(interesting_headers.keys()) | |
145 | |
146 buf = "%s\n" % method | |
147 for key in sorted_header_keys: | |
148 val = interesting_headers[key] | |
149 if key.startswith(provider.header_prefix): | |
150 buf += "%s:%s\n" % (key, val) | |
151 else: | |
152 buf += "%s\n" % val | |
153 | |
154 # don't include anything after the first ? in the resource... | |
155 # unless it is one of the QSA of interest, defined above | |
156 t = path.split('?') | |
157 buf += t[0] | |
158 | |
159 if len(t) > 1: | |
160 qsa = t[1].split('&') | |
161 qsa = [a.split('=', 1) for a in qsa] | |
162 qsa = [unquote_v(a) for a in qsa if a[0] in qsa_of_interest] | |
163 if len(qsa) > 0: | |
164 qsa.sort(key=lambda x: x[0]) | |
165 qsa = ['='.join(a) for a in qsa] | |
166 buf += '?' | |
167 buf += '&'.join(qsa) | |
168 | |
169 return buf | |
170 | |
171 | |
172 def merge_meta(headers, metadata, provider=None): | |
173 if not provider: | |
174 provider = boto.provider.get_default() | |
175 metadata_prefix = provider.metadata_prefix | |
176 final_headers = headers.copy() | |
177 for k in metadata.keys(): | |
178 if k.lower() in boto.s3.key.Key.base_user_settable_fields: | |
179 final_headers[k] = metadata[k] | |
180 else: | |
181 final_headers[metadata_prefix + k] = metadata[k] | |
182 | |
183 return final_headers | |
184 | |
185 | |
186 def get_aws_metadata(headers, provider=None): | |
187 if not provider: | |
188 provider = boto.provider.get_default() | |
189 metadata_prefix = provider.metadata_prefix | |
190 metadata = {} | |
191 for hkey in headers.keys(): | |
192 if hkey.lower().startswith(metadata_prefix): | |
193 val = urllib.parse.unquote(headers[hkey]) | |
194 if isinstance(val, bytes): | |
195 try: | |
196 val = val.decode('utf-8') | |
197 except UnicodeDecodeError: | |
198 # Just leave the value as-is | |
199 pass | |
200 metadata[hkey[len(metadata_prefix):]] = val | |
201 del headers[hkey] | |
202 return metadata | |
203 | |
204 | |
205 def retry_url(url, retry_on_404=True, num_retries=10, timeout=None): | |
206 """ | |
207 Retry a url. This is specifically used for accessing the metadata | |
208 service on an instance. Since this address should never be proxied | |
209 (for security reasons), we create a ProxyHandler with a NULL | |
210 dictionary to override any proxy settings in the environment. | |
211 """ | |
212 for i in range(0, num_retries): | |
213 try: | |
214 proxy_handler = urllib.request.ProxyHandler({}) | |
215 opener = urllib.request.build_opener(proxy_handler) | |
216 req = urllib.request.Request(url) | |
217 r = opener.open(req, timeout=timeout) | |
218 result = r.read() | |
219 | |
220 if(not isinstance(result, six.string_types) and | |
221 hasattr(result, 'decode')): | |
222 result = result.decode('utf-8') | |
223 | |
224 return result | |
225 except urllib.error.HTTPError as e: | |
226 code = e.getcode() | |
227 if code == 404 and not retry_on_404: | |
228 return '' | |
229 except Exception as e: | |
230 boto.log.exception('Caught exception reading instance data') | |
231 # If not on the last iteration of the loop then sleep. | |
232 if i + 1 != num_retries: | |
233 boto.log.debug('Sleeping before retrying') | |
234 time.sleep(min(2 ** i, | |
235 boto.config.get('Boto', 'max_retry_delay', 60))) | |
236 boto.log.error('Unable to read instance data, giving up') | |
237 return '' | |
238 | |
239 | |
240 def _get_instance_metadata(url, num_retries, timeout=None): | |
241 return LazyLoadMetadata(url, num_retries, timeout) | |
242 | |
243 | |
244 class LazyLoadMetadata(dict): | |
245 def __init__(self, url, num_retries, timeout=None): | |
246 self._url = url | |
247 self._num_retries = num_retries | |
248 self._leaves = {} | |
249 self._dicts = [] | |
250 self._timeout = timeout | |
251 data = boto.utils.retry_url(self._url, num_retries=self._num_retries, timeout=self._timeout) | |
252 if data: | |
253 fields = data.split('\n') | |
254 for field in fields: | |
255 if field.endswith('/'): | |
256 key = field[0:-1] | |
257 self._dicts.append(key) | |
258 else: | |
259 p = field.find('=') | |
260 if p > 0: | |
261 key = field[p + 1:] | |
262 resource = field[0:p] + '/openssh-key' | |
263 else: | |
264 key = resource = field | |
265 self._leaves[key] = resource | |
266 self[key] = None | |
267 | |
268 def _materialize(self): | |
269 for key in self: | |
270 self[key] | |
271 | |
272 def __getitem__(self, key): | |
273 if key not in self: | |
274 # allow dict to throw the KeyError | |
275 return super(LazyLoadMetadata, self).__getitem__(key) | |
276 | |
277 # already loaded | |
278 val = super(LazyLoadMetadata, self).__getitem__(key) | |
279 if val is not None: | |
280 return val | |
281 | |
282 if key in self._leaves: | |
283 resource = self._leaves[key] | |
284 last_exception = None | |
285 | |
286 for i in range(0, self._num_retries): | |
287 try: | |
288 val = boto.utils.retry_url( | |
289 self._url + urllib.parse.quote(resource, | |
290 safe="/:"), | |
291 num_retries=self._num_retries, | |
292 timeout=self._timeout) | |
293 if val and val[0] == '{': | |
294 val = json.loads(val) | |
295 break | |
296 else: | |
297 p = val.find('\n') | |
298 if p > 0: | |
299 val = val.split('\n') | |
300 break | |
301 | |
302 except JSONDecodeError as e: | |
303 boto.log.debug( | |
304 "encountered '%s' exception: %s" % ( | |
305 e.__class__.__name__, e)) | |
306 boto.log.debug( | |
307 'corrupted JSON data found: %s' % val) | |
308 last_exception = e | |
309 | |
310 except Exception as e: | |
311 boto.log.debug("encountered unretryable" + | |
312 " '%s' exception, re-raising" % ( | |
313 e.__class__.__name__)) | |
314 last_exception = e | |
315 raise | |
316 | |
317 boto.log.error("Caught exception reading meta data" + | |
318 " for the '%s' try" % (i + 1)) | |
319 | |
320 if i + 1 != self._num_retries: | |
321 next_sleep = min( | |
322 random.random() * 2 ** i, | |
323 boto.config.get('Boto', 'max_retry_delay', 60)) | |
324 time.sleep(next_sleep) | |
325 else: | |
326 boto.log.error('Unable to read meta data, giving up') | |
327 boto.log.error( | |
328 "encountered '%s' exception: %s" % ( | |
329 last_exception.__class__.__name__, last_exception)) | |
330 raise last_exception | |
331 | |
332 self[key] = val | |
333 elif key in self._dicts: | |
334 self[key] = LazyLoadMetadata(self._url + key + '/', | |
335 self._num_retries) | |
336 | |
337 return super(LazyLoadMetadata, self).__getitem__(key) | |
338 | |
339 def get(self, key, default=None): | |
340 try: | |
341 return self[key] | |
342 except KeyError: | |
343 return default | |
344 | |
345 def values(self): | |
346 self._materialize() | |
347 return super(LazyLoadMetadata, self).values() | |
348 | |
349 def items(self): | |
350 self._materialize() | |
351 return super(LazyLoadMetadata, self).items() | |
352 | |
353 def __str__(self): | |
354 self._materialize() | |
355 return super(LazyLoadMetadata, self).__str__() | |
356 | |
357 def __repr__(self): | |
358 self._materialize() | |
359 return super(LazyLoadMetadata, self).__repr__() | |
360 | |
361 | |
362 def _build_instance_metadata_url(url, version, path): | |
363 """ | |
364 Builds an EC2 metadata URL for fetching information about an instance. | |
365 | |
366 Example: | |
367 | |
368 >>> _build_instance_metadata_url('http://169.254.169.254', 'latest', 'meta-data/') | |
369 http://169.254.169.254/latest/meta-data/ | |
370 | |
371 :type url: string | |
372 :param url: URL to metadata service, e.g. 'http://169.254.169.254' | |
373 | |
374 :type version: string | |
375 :param version: Version of the metadata to get, e.g. 'latest' | |
376 | |
377 :type path: string | |
378 :param path: Path of the metadata to get, e.g. 'meta-data/'. If a trailing | |
379 slash is required it must be passed in with the path. | |
380 | |
381 :return: The full metadata URL | |
382 """ | |
383 return '%s/%s/%s' % (url, version, path) | |
384 | |
385 | |
386 def get_instance_metadata(version='latest', url='http://169.254.169.254', | |
387 data='meta-data/', timeout=None, num_retries=5): | |
388 """ | |
389 Returns the instance metadata as a nested Python dictionary. | |
390 Simple values (e.g. local_hostname, hostname, etc.) will be | |
391 stored as string values. Values such as ancestor-ami-ids will | |
392 be stored in the dict as a list of string values. More complex | |
393 fields such as public-keys and will be stored as nested dicts. | |
394 | |
395 If the timeout is specified, the connection to the specified url | |
396 will time out after the specified number of seconds. | |
397 | |
398 """ | |
399 try: | |
400 metadata_url = _build_instance_metadata_url(url, version, data) | |
401 return _get_instance_metadata(metadata_url, num_retries=num_retries, timeout=timeout) | |
402 except urllib.error.URLError: | |
403 boto.log.exception("Exception caught when trying to retrieve " | |
404 "instance metadata for: %s", data) | |
405 return None | |
406 | |
407 | |
408 def get_instance_identity(version='latest', url='http://169.254.169.254', | |
409 timeout=None, num_retries=5): | |
410 """ | |
411 Returns the instance identity as a nested Python dictionary. | |
412 """ | |
413 iid = {} | |
414 base_url = _build_instance_metadata_url(url, version, | |
415 'dynamic/instance-identity/') | |
416 try: | |
417 data = retry_url(base_url, num_retries=num_retries, timeout=timeout) | |
418 fields = data.split('\n') | |
419 for field in fields: | |
420 val = retry_url(base_url + '/' + field + '/', num_retries=num_retries, timeout=timeout) | |
421 if val[0] == '{': | |
422 val = json.loads(val) | |
423 if field: | |
424 iid[field] = val | |
425 return iid | |
426 except urllib.error.URLError: | |
427 return None | |
428 | |
429 | |
430 def get_instance_userdata(version='latest', sep=None, | |
431 url='http://169.254.169.254', timeout=None, num_retries=5): | |
432 ud_url = _build_instance_metadata_url(url, version, 'user-data') | |
433 user_data = retry_url(ud_url, retry_on_404=False, num_retries=num_retries, timeout=timeout) | |
434 if user_data: | |
435 if sep: | |
436 l = user_data.split(sep) | |
437 user_data = {} | |
438 for nvpair in l: | |
439 t = nvpair.split('=') | |
440 user_data[t[0].strip()] = t[1].strip() | |
441 return user_data | |
442 | |
443 ISO8601 = '%Y-%m-%dT%H:%M:%SZ' | |
444 ISO8601_MS = '%Y-%m-%dT%H:%M:%S.%fZ' | |
445 RFC1123 = '%a, %d %b %Y %H:%M:%S %Z' | |
446 LOCALE_LOCK = threading.Lock() | |
447 | |
448 | |
449 @contextmanager | |
450 def setlocale(name): | |
451 """ | |
452 A context manager to set the locale in a threadsafe manner. | |
453 """ | |
454 with LOCALE_LOCK: | |
455 saved = locale.setlocale(locale.LC_ALL) | |
456 | |
457 try: | |
458 yield locale.setlocale(locale.LC_ALL, name) | |
459 finally: | |
460 locale.setlocale(locale.LC_ALL, saved) | |
461 | |
462 | |
463 def get_ts(ts=None): | |
464 if not ts: | |
465 ts = time.gmtime() | |
466 return time.strftime(ISO8601, ts) | |
467 | |
468 | |
469 def parse_ts(ts): | |
470 with setlocale('C'): | |
471 ts = ts.strip() | |
472 try: | |
473 dt = datetime.datetime.strptime(ts, ISO8601) | |
474 return dt | |
475 except ValueError: | |
476 try: | |
477 dt = datetime.datetime.strptime(ts, ISO8601_MS) | |
478 return dt | |
479 except ValueError: | |
480 dt = datetime.datetime.strptime(ts, RFC1123) | |
481 return dt | |
482 | |
483 | |
484 def find_class(module_name, class_name=None): | |
485 if class_name: | |
486 module_name = "%s.%s" % (module_name, class_name) | |
487 modules = module_name.split('.') | |
488 c = None | |
489 | |
490 try: | |
491 for m in modules[1:]: | |
492 if c: | |
493 c = getattr(c, m) | |
494 else: | |
495 c = getattr(__import__(".".join(modules[0:-1])), m) | |
496 return c | |
497 except: | |
498 return None | |
499 | |
500 | |
501 def update_dme(username, password, dme_id, ip_address): | |
502 """ | |
503 Update your Dynamic DNS record with DNSMadeEasy.com | |
504 """ | |
505 dme_url = 'https://www.dnsmadeeasy.com/servlet/updateip' | |
506 dme_url += '?username=%s&password=%s&id=%s&ip=%s' | |
507 s = urllib.request.urlopen(dme_url % (username, password, dme_id, ip_address)) | |
508 return s.read() | |
509 | |
510 | |
511 def fetch_file(uri, file=None, username=None, password=None): | |
512 """ | |
513 Fetch a file based on the URI provided. | |
514 If you do not pass in a file pointer a tempfile.NamedTemporaryFile, | |
515 or None if the file could not be retrieved is returned. | |
516 The URI can be either an HTTP url, or "s3://bucket_name/key_name" | |
517 """ | |
518 boto.log.info('Fetching %s' % uri) | |
519 if file is None: | |
520 file = tempfile.NamedTemporaryFile() | |
521 try: | |
522 if uri.startswith('s3://'): | |
523 bucket_name, key_name = uri[len('s3://'):].split('/', 1) | |
524 c = boto.connect_s3(aws_access_key_id=username, | |
525 aws_secret_access_key=password) | |
526 bucket = c.get_bucket(bucket_name) | |
527 key = bucket.get_key(key_name) | |
528 key.get_contents_to_file(file) | |
529 else: | |
530 if username and password: | |
531 passman = urllib.request.HTTPPasswordMgrWithDefaultRealm() | |
532 passman.add_password(None, uri, username, password) | |
533 authhandler = urllib.request.HTTPBasicAuthHandler(passman) | |
534 opener = urllib.request.build_opener(authhandler) | |
535 urllib.request.install_opener(opener) | |
536 s = urllib.request.urlopen(uri) | |
537 file.write(s.read()) | |
538 file.seek(0) | |
539 except: | |
540 raise | |
541 boto.log.exception('Problem Retrieving file: %s' % uri) | |
542 file = None | |
543 return file | |
544 | |
545 | |
546 class ShellCommand(object): | |
547 | |
548 def __init__(self, command, wait=True, fail_fast=False, cwd=None): | |
549 self.exit_code = 0 | |
550 self.command = command | |
551 self.log_fp = StringIO() | |
552 self.wait = wait | |
553 self.fail_fast = fail_fast | |
554 self.run(cwd=cwd) | |
555 | |
556 def run(self, cwd=None): | |
557 boto.log.info('running:%s' % self.command) | |
558 self.process = subprocess.Popen(self.command, shell=True, | |
559 stdin=subprocess.PIPE, | |
560 stdout=subprocess.PIPE, | |
561 stderr=subprocess.PIPE, | |
562 cwd=cwd) | |
563 if(self.wait): | |
564 while self.process.poll() is None: | |
565 time.sleep(1) | |
566 t = self.process.communicate() | |
567 self.log_fp.write(t[0]) | |
568 self.log_fp.write(t[1]) | |
569 boto.log.info(self.log_fp.getvalue()) | |
570 self.exit_code = self.process.returncode | |
571 | |
572 if self.fail_fast and self.exit_code != 0: | |
573 raise Exception("Command " + self.command + | |
574 " failed with status " + self.exit_code) | |
575 | |
576 return self.exit_code | |
577 | |
578 def setReadOnly(self, value): | |
579 raise AttributeError | |
580 | |
581 def getStatus(self): | |
582 return self.exit_code | |
583 | |
584 status = property(getStatus, setReadOnly, None, | |
585 'The exit code for the command') | |
586 | |
587 def getOutput(self): | |
588 return self.log_fp.getvalue() | |
589 | |
590 output = property(getOutput, setReadOnly, None, | |
591 'The STDIN and STDERR output of the command') | |
592 | |
593 | |
594 class AuthSMTPHandler(logging.handlers.SMTPHandler): | |
595 """ | |
596 This class extends the SMTPHandler in the standard Python logging module | |
597 to accept a username and password on the constructor and to then use those | |
598 credentials to authenticate with the SMTP server. To use this, you could | |
599 add something like this in your boto config file: | |
600 | |
601 [handler_hand07] | |
602 class=boto.utils.AuthSMTPHandler | |
603 level=WARN | |
604 formatter=form07 | |
605 args=('localhost', 'username', 'password', 'from@abc', ['user1@abc', 'user2@xyz'], 'Logger Subject') | |
606 """ | |
607 | |
608 def __init__(self, mailhost, username, password, | |
609 fromaddr, toaddrs, subject): | |
610 """ | |
611 Initialize the handler. | |
612 | |
613 We have extended the constructor to accept a username/password | |
614 for SMTP authentication. | |
615 """ | |
616 super(AuthSMTPHandler, self).__init__(mailhost, fromaddr, | |
617 toaddrs, subject) | |
618 self.username = username | |
619 self.password = password | |
620 | |
621 def emit(self, record): | |
622 """ | |
623 Emit a record. | |
624 | |
625 Format the record and send it to the specified addressees. | |
626 It would be really nice if I could add authorization to this class | |
627 without having to resort to cut and paste inheritance but, no. | |
628 """ | |
629 try: | |
630 port = self.mailport | |
631 if not port: | |
632 port = smtplib.SMTP_PORT | |
633 smtp = smtplib.SMTP(self.mailhost, port) | |
634 smtp.login(self.username, self.password) | |
635 msg = self.format(record) | |
636 msg = "From: %s\r\nTo: %s\r\nSubject: %s\r\nDate: %s\r\n\r\n%s" % ( | |
637 self.fromaddr, | |
638 ','.join(self.toaddrs), | |
639 self.getSubject(record), | |
640 email.utils.formatdate(), msg) | |
641 smtp.sendmail(self.fromaddr, self.toaddrs, msg) | |
642 smtp.quit() | |
643 except (KeyboardInterrupt, SystemExit): | |
644 raise | |
645 except: | |
646 self.handleError(record) | |
647 | |
648 | |
649 class LRUCache(dict): | |
650 """A dictionary-like object that stores only a certain number of items, and | |
651 discards its least recently used item when full. | |
652 | |
653 >>> cache = LRUCache(3) | |
654 >>> cache['A'] = 0 | |
655 >>> cache['B'] = 1 | |
656 >>> cache['C'] = 2 | |
657 >>> len(cache) | |
658 3 | |
659 | |
660 >>> cache['A'] | |
661 0 | |
662 | |
663 Adding new items to the cache does not increase its size. Instead, the least | |
664 recently used item is dropped: | |
665 | |
666 >>> cache['D'] = 3 | |
667 >>> len(cache) | |
668 3 | |
669 >>> 'B' in cache | |
670 False | |
671 | |
672 Iterating over the cache returns the keys, starting with the most recently | |
673 used: | |
674 | |
675 >>> for key in cache: | |
676 ... print key | |
677 D | |
678 A | |
679 C | |
680 | |
681 This code is based on the LRUCache class from Genshi which is based on | |
682 `Myghty <http://www.myghty.org>`_'s LRUCache from ``myghtyutils.util``, | |
683 written by Mike Bayer and released under the MIT license (Genshi uses the | |
684 BSD License). | |
685 """ | |
686 | |
687 class _Item(object): | |
688 def __init__(self, key, value): | |
689 self.previous = self.next = None | |
690 self.key = key | |
691 self.value = value | |
692 | |
693 def __repr__(self): | |
694 return repr(self.value) | |
695 | |
696 def __init__(self, capacity): | |
697 self._dict = dict() | |
698 self.capacity = capacity | |
699 self.head = None | |
700 self.tail = None | |
701 | |
702 def __contains__(self, key): | |
703 return key in self._dict | |
704 | |
705 def __iter__(self): | |
706 cur = self.head | |
707 while cur: | |
708 yield cur.key | |
709 cur = cur.next | |
710 | |
711 def __len__(self): | |
712 return len(self._dict) | |
713 | |
714 def __getitem__(self, key): | |
715 item = self._dict[key] | |
716 self._update_item(item) | |
717 return item.value | |
718 | |
719 def __setitem__(self, key, value): | |
720 item = self._dict.get(key) | |
721 if item is None: | |
722 item = self._Item(key, value) | |
723 self._dict[key] = item | |
724 self._insert_item(item) | |
725 else: | |
726 item.value = value | |
727 self._update_item(item) | |
728 self._manage_size() | |
729 | |
730 def __repr__(self): | |
731 return repr(self._dict) | |
732 | |
733 def _insert_item(self, item): | |
734 item.previous = None | |
735 item.next = self.head | |
736 if self.head is not None: | |
737 self.head.previous = item | |
738 else: | |
739 self.tail = item | |
740 self.head = item | |
741 self._manage_size() | |
742 | |
743 def _manage_size(self): | |
744 while len(self._dict) > self.capacity: | |
745 del self._dict[self.tail.key] | |
746 if self.tail != self.head: | |
747 self.tail = self.tail.previous | |
748 self.tail.next = None | |
749 else: | |
750 self.head = self.tail = None | |
751 | |
752 def _update_item(self, item): | |
753 if self.head == item: | |
754 return | |
755 | |
756 previous = item.previous | |
757 previous.next = item.next | |
758 if item.next is not None: | |
759 item.next.previous = previous | |
760 else: | |
761 self.tail = previous | |
762 | |
763 item.previous = None | |
764 item.next = self.head | |
765 self.head.previous = self.head = item | |
766 | |
767 | |
768 class Password(object): | |
769 """ | |
770 Password object that stores itself as hashed. | |
771 Hash defaults to SHA512 if available, MD5 otherwise. | |
772 """ | |
773 hashfunc = _hashfn | |
774 | |
775 def __init__(self, str=None, hashfunc=None): | |
776 """ | |
777 Load the string from an initial value, this should be the | |
778 raw hashed password. | |
779 """ | |
780 self.str = str | |
781 if hashfunc: | |
782 self.hashfunc = hashfunc | |
783 | |
784 def set(self, value): | |
785 if not isinstance(value, bytes): | |
786 value = value.encode('utf-8') | |
787 self.str = self.hashfunc(value).hexdigest() | |
788 | |
789 def __str__(self): | |
790 return str(self.str) | |
791 | |
792 def __eq__(self, other): | |
793 if other is None: | |
794 return False | |
795 if not isinstance(other, bytes): | |
796 other = other.encode('utf-8') | |
797 return str(self.hashfunc(other).hexdigest()) == str(self.str) | |
798 | |
799 def __len__(self): | |
800 if self.str: | |
801 return len(self.str) | |
802 else: | |
803 return 0 | |
804 | |
805 | |
806 def notify(subject, body=None, html_body=None, to_string=None, | |
807 attachments=None, append_instance_id=True): | |
808 attachments = attachments or [] | |
809 if append_instance_id: | |
810 subject = "[%s] %s" % ( | |
811 boto.config.get_value("Instance", "instance-id"), subject) | |
812 if not to_string: | |
813 to_string = boto.config.get_value('Notification', 'smtp_to', None) | |
814 if to_string: | |
815 try: | |
816 from_string = boto.config.get_value('Notification', | |
817 'smtp_from', 'boto') | |
818 msg = email.mime.multipart.MIMEMultipart() | |
819 msg['From'] = from_string | |
820 msg['Reply-To'] = from_string | |
821 msg['To'] = to_string | |
822 msg['Date'] = email.utils.formatdate(localtime=True) | |
823 msg['Subject'] = subject | |
824 | |
825 if body: | |
826 msg.attach(email.mime.text.MIMEText(body)) | |
827 | |
828 if html_body: | |
829 part = email.mime.base.MIMEBase('text', 'html') | |
830 part.set_payload(html_body) | |
831 email.encoders.encode_base64(part) | |
832 msg.attach(part) | |
833 | |
834 for part in attachments: | |
835 msg.attach(part) | |
836 | |
837 smtp_host = boto.config.get_value('Notification', | |
838 'smtp_host', 'localhost') | |
839 | |
840 # Alternate port support | |
841 if boto.config.get_value("Notification", "smtp_port"): | |
842 server = smtplib.SMTP(smtp_host, int( | |
843 boto.config.get_value("Notification", "smtp_port"))) | |
844 else: | |
845 server = smtplib.SMTP(smtp_host) | |
846 | |
847 # TLS support | |
848 if boto.config.getbool("Notification", "smtp_tls"): | |
849 server.ehlo() | |
850 server.starttls() | |
851 server.ehlo() | |
852 smtp_user = boto.config.get_value('Notification', 'smtp_user', '') | |
853 smtp_pass = boto.config.get_value('Notification', 'smtp_pass', '') | |
854 if smtp_user: | |
855 server.login(smtp_user, smtp_pass) | |
856 server.sendmail(from_string, to_string, msg.as_string()) | |
857 server.quit() | |
858 except: | |
859 boto.log.exception('notify failed') | |
860 | |
861 | |
862 def get_utf8_value(value): | |
863 if not six.PY2 and isinstance(value, bytes): | |
864 return value | |
865 | |
866 if not isinstance(value, six.string_types): | |
867 value = six.text_type(value) | |
868 | |
869 if isinstance(value, six.text_type): | |
870 value = value.encode('utf-8') | |
871 | |
872 return value | |
873 | |
874 | |
875 def mklist(value): | |
876 if not isinstance(value, list): | |
877 if isinstance(value, tuple): | |
878 value = list(value) | |
879 else: | |
880 value = [value] | |
881 return value | |
882 | |
883 | |
884 def pythonize_name(name): | |
885 """Convert camel case to a "pythonic" name. | |
886 | |
887 Examples:: | |
888 | |
889 pythonize_name('CamelCase') -> 'camel_case' | |
890 pythonize_name('already_pythonized') -> 'already_pythonized' | |
891 pythonize_name('HTTPRequest') -> 'http_request' | |
892 pythonize_name('HTTPStatus200Ok') -> 'http_status_200_ok' | |
893 pythonize_name('UPPER') -> 'upper' | |
894 pythonize_name('') -> '' | |
895 | |
896 """ | |
897 s1 = _first_cap_regex.sub(r'\1_\2', name) | |
898 s2 = _number_cap_regex.sub(r'\1_\2', s1) | |
899 return _end_cap_regex.sub(r'\1_\2', s2).lower() | |
900 | |
901 | |
902 def write_mime_multipart(content, compress=False, deftype='text/plain', delimiter=':'): | |
903 """Description: | |
904 :param content: A list of tuples of name-content pairs. This is used | |
905 instead of a dict to ensure that scripts run in order | |
906 :type list of tuples: | |
907 | |
908 :param compress: Use gzip to compress the scripts, defaults to no compression | |
909 :type bool: | |
910 | |
911 :param deftype: The type that should be assumed if nothing else can be figured out | |
912 :type str: | |
913 | |
914 :param delimiter: mime delimiter | |
915 :type str: | |
916 | |
917 :return: Final mime multipart | |
918 :rtype: str: | |
919 """ | |
920 wrapper = email.mime.multipart.MIMEMultipart() | |
921 for name, con in content: | |
922 definite_type = guess_mime_type(con, deftype) | |
923 maintype, subtype = definite_type.split('/', 1) | |
924 if maintype == 'text': | |
925 mime_con = email.mime.text.MIMEText(con, _subtype=subtype) | |
926 else: | |
927 mime_con = email.mime.base.MIMEBase(maintype, subtype) | |
928 mime_con.set_payload(con) | |
929 # Encode the payload using Base64 | |
930 email.encoders.encode_base64(mime_con) | |
931 mime_con.add_header('Content-Disposition', 'attachment', filename=name) | |
932 wrapper.attach(mime_con) | |
933 rcontent = wrapper.as_string() | |
934 | |
935 if compress: | |
936 buf = StringIO() | |
937 gz = gzip.GzipFile(mode='wb', fileobj=buf) | |
938 try: | |
939 gz.write(rcontent) | |
940 finally: | |
941 gz.close() | |
942 rcontent = buf.getvalue() | |
943 | |
944 return rcontent | |
945 | |
946 | |
947 def guess_mime_type(content, deftype): | |
948 """Description: Guess the mime type of a block of text | |
949 :param content: content we're finding the type of | |
950 :type str: | |
951 | |
952 :param deftype: Default mime type | |
953 :type str: | |
954 | |
955 :rtype: <type>: | |
956 :return: <description> | |
957 """ | |
958 # Mappings recognized by cloudinit | |
959 starts_with_mappings = { | |
960 '#include': 'text/x-include-url', | |
961 '#!': 'text/x-shellscript', | |
962 '#cloud-config': 'text/cloud-config', | |
963 '#upstart-job': 'text/upstart-job', | |
964 '#part-handler': 'text/part-handler', | |
965 '#cloud-boothook': 'text/cloud-boothook' | |
966 } | |
967 rtype = deftype | |
968 for possible_type, mimetype in starts_with_mappings.items(): | |
969 if content.startswith(possible_type): | |
970 rtype = mimetype | |
971 break | |
972 return(rtype) | |
973 | |
974 | |
975 def compute_md5(fp, buf_size=8192, size=None): | |
976 """ | |
977 Compute MD5 hash on passed file and return results in a tuple of values. | |
978 | |
979 :type fp: file | |
980 :param fp: File pointer to the file to MD5 hash. The file pointer | |
981 will be reset to its current location before the | |
982 method returns. | |
983 | |
984 :type buf_size: integer | |
985 :param buf_size: Number of bytes per read request. | |
986 | |
987 :type size: int | |
988 :param size: (optional) The Maximum number of bytes to read from | |
989 the file pointer (fp). This is useful when uploading | |
990 a file in multiple parts where the file is being | |
991 split inplace into different parts. Less bytes may | |
992 be available. | |
993 | |
994 :rtype: tuple | |
995 :return: A tuple containing the hex digest version of the MD5 hash | |
996 as the first element, the base64 encoded version of the | |
997 plain digest as the second element and the data size as | |
998 the third element. | |
999 """ | |
1000 return compute_hash(fp, buf_size, size, hash_algorithm=md5) | |
1001 | |
1002 | |
1003 def compute_hash(fp, buf_size=8192, size=None, hash_algorithm=md5): | |
1004 hash_obj = hash_algorithm() | |
1005 spos = fp.tell() | |
1006 if size and size < buf_size: | |
1007 s = fp.read(size) | |
1008 else: | |
1009 s = fp.read(buf_size) | |
1010 while s: | |
1011 if not isinstance(s, bytes): | |
1012 s = s.encode('utf-8') | |
1013 hash_obj.update(s) | |
1014 if size: | |
1015 size -= len(s) | |
1016 if size <= 0: | |
1017 break | |
1018 if size and size < buf_size: | |
1019 s = fp.read(size) | |
1020 else: | |
1021 s = fp.read(buf_size) | |
1022 hex_digest = hash_obj.hexdigest() | |
1023 base64_digest = encodebytes(hash_obj.digest()).decode('utf-8') | |
1024 if base64_digest[-1] == '\n': | |
1025 base64_digest = base64_digest[0:-1] | |
1026 # data_size based on bytes read. | |
1027 data_size = fp.tell() - spos | |
1028 fp.seek(spos) | |
1029 return (hex_digest, base64_digest, data_size) | |
1030 | |
1031 | |
1032 def find_matching_headers(name, headers): | |
1033 """ | |
1034 Takes a specific header name and a dict of headers {"name": "value"}. | |
1035 Returns a list of matching header names, case-insensitive. | |
1036 | |
1037 """ | |
1038 return [h for h in headers if h.lower() == name.lower()] | |
1039 | |
1040 | |
1041 def merge_headers_by_name(name, headers): | |
1042 """ | |
1043 Takes a specific header name and a dict of headers {"name": "value"}. | |
1044 Returns a string of all header values, comma-separated, that match the | |
1045 input header name, case-insensitive. | |
1046 | |
1047 """ | |
1048 matching_headers = find_matching_headers(name, headers) | |
1049 return ','.join(str(headers[h]) for h in matching_headers | |
1050 if headers[h] is not None) | |
1051 | |
1052 | |
1053 class RequestHook(object): | |
1054 """ | |
1055 This can be extended and supplied to the connection object | |
1056 to gain access to request and response object after the request completes. | |
1057 One use for this would be to implement some specific request logging. | |
1058 """ | |
1059 def handle_request_data(self, request, response, error=False): | |
1060 pass | |
1061 | |
1062 | |
1063 def host_is_ipv6(hostname): | |
1064 """ | |
1065 Detect (naively) if the hostname is an IPV6 host. | |
1066 Return a boolean. | |
1067 """ | |
1068 # empty strings or anything that is not a string is automatically not an | |
1069 # IPV6 address | |
1070 if not hostname or not isinstance(hostname, str): | |
1071 return False | |
1072 | |
1073 if hostname.startswith('['): | |
1074 return True | |
1075 | |
1076 if len(hostname.split(':')) > 2: | |
1077 return True | |
1078 | |
1079 # Anything else that doesn't start with brackets or doesn't have more than | |
1080 # one ':' should not be an IPV6 address. This is very naive but the rest of | |
1081 # the connection chain should error accordingly for typos or ill formed | |
1082 # addresses | |
1083 return False | |
1084 | |
1085 | |
1086 def parse_host(hostname): | |
1087 """ | |
1088 Given a hostname that may have a port name, ensure that the port is trimmed | |
1089 returning only the host, including hostnames that are IPV6 and may include | |
1090 brackets. | |
1091 """ | |
1092 # ensure that hostname does not have any whitespaces | |
1093 hostname = hostname.strip() | |
1094 | |
1095 if host_is_ipv6(hostname): | |
1096 return hostname.split(']:', 1)[0].strip('[]') | |
1097 else: | |
1098 return hostname.split(':', 1)[0] |