-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathQtrac.py
98 lines (86 loc) · 3.45 KB
/
Qtrac.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
#!/usr/bin/env python3
# Copyright © 2012-13 Qtrac Ltd. All rights reserved.
# This program or module is free software: you can redistribute it
# and/or modify it under the terms of the GNU General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version. It is provided for
# educational purposes and is distributed in the hope that it will be
# useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# General Public License for more details.
import abc
import collections
import errno
import functools
import os
import sys
def coroutine(function):
@functools.wraps(function)
def wrapper(*args, **kwargs):
generator = function(*args, **kwargs)
next(generator)
return generator
return wrapper
if sys.version_info[:2] < (3, 3):
def remove_if_exists(filename):
try:
os.remove(filename)
except OSError as err:
if err.errno != errno.ENOENT:
raise
else:
def remove_if_exists(filename):
try:
os.remove(filename)
except FileNotFoundError:
pass # All other exceptions are passed to the caller
# Thanks to Nick Coghlan for these!
if sys.version_info[:2] >= (3, 3):
def has_methods(*methods):
def decorator(Base):
def __subclasshook__(Class, Subclass):
if Class is Base:
attributes = collections.ChainMap(*(Superclass.__dict__
for Superclass in Subclass.__mro__))
if all(method in attributes for method in methods):
return True
return NotImplemented
Base.__subclasshook__ = classmethod(__subclasshook__)
return Base
return decorator
else:
def has_methods(*methods):
def decorator(Base):
def __subclasshook__(Class, Subclass):
if Class is Base:
needed = set(methods)
for Superclass in Subclass.__mro__:
for meth in needed.copy():
if meth in Superclass.__dict__:
needed.discard(meth)
if not needed:
return True
return NotImplemented
Base.__subclasshook__ = classmethod(__subclasshook__)
return Base
return decorator
# Thanks to Nick Coghlan for this!
class Requirer(metaclass=abc.ABCMeta):
# Since we have rules for adding new expected attributes, we *do*
# perform the check for subclasses
@classmethod
def __subclasshook__(Class, Subclass):
methods = set()
for Superclass in Subclass.__mro__:
if hasattr(Superclass, "required_methods"):
methods |= set(Superclass.required_methods)
attributes = collections.ChainMap(*(Superclass.__dict__
for Superclass in Class.__mro__))
if all(method in attributes for method in methods):
return True
return NotImplemented
def report(message="", error=False):
if len(message) >= 70 and not error:
message = message[:67] + "..."
sys.stdout.write("\r{:70}{}".format(message, "\n" if error else ""))
sys.stdout.flush()