28
28
29
29
from collections .abc import Callable
30
30
from types import FunctionType , NoneType
31
- from typing import Any , NamedTuple
31
+ from typing import Any , NamedTuple , NoReturn , Literal , overload
32
32
33
33
# TODO:
34
34
#
59
59
}
60
60
61
61
class Unspecified :
62
- def __repr__ (self ):
62
+ def __repr__ (self ) -> str :
63
63
return '<Unspecified>'
64
64
65
65
unspecified = Unspecified ()
66
66
67
67
68
68
class Null :
69
- def __repr__ (self ):
69
+ def __repr__ (self ) -> str :
70
70
return '<Null>'
71
71
72
72
NULL = Null ()
73
73
74
74
75
75
class Unknown :
76
- def __repr__ (self ):
76
+ def __repr__ (self ) -> str :
77
77
return '<Unknown>'
78
78
79
79
unknown = Unknown ()
80
80
81
81
sig_end_marker = '--'
82
82
83
83
Appender = Callable [[str ], None ]
84
- Outputter = Callable [[None ], str ]
84
+ Outputter = Callable [[], str ]
85
85
86
86
class _TextAccumulator (NamedTuple ):
87
87
text : list [str ]
88
88
append : Appender
89
89
output : Outputter
90
90
91
- def _text_accumulator ():
92
- text = []
91
+ def _text_accumulator () -> _TextAccumulator :
92
+ text : list [ str ] = []
93
93
def output ():
94
94
s = '' .join (text )
95
95
text .clear ()
@@ -98,10 +98,10 @@ def output():
98
98
99
99
100
100
class TextAccumulator (NamedTuple ):
101
- text : list [str ]
102
101
append : Appender
102
+ output : Outputter
103
103
104
- def text_accumulator ():
104
+ def text_accumulator () -> TextAccumulator :
105
105
"""
106
106
Creates a simple text accumulator / joiner.
107
107
@@ -115,8 +115,28 @@ def text_accumulator():
115
115
text , append , output = _text_accumulator ()
116
116
return TextAccumulator (append , output )
117
117
118
-
119
- def warn_or_fail (fail = False , * args , filename = None , line_number = None ):
118
+ @overload
119
+ def warn_or_fail (
120
+ * args : object ,
121
+ fail : Literal [True ],
122
+ filename : str | None = None ,
123
+ line_number : int | None = None ,
124
+ ) -> NoReturn : ...
125
+
126
+ @overload
127
+ def warn_or_fail (
128
+ * args : object ,
129
+ fail : Literal [False ] = False ,
130
+ filename : str | None = None ,
131
+ line_number : int | None = None ,
132
+ ) -> None : ...
133
+
134
+ def warn_or_fail (
135
+ * args : object ,
136
+ fail : bool = False ,
137
+ filename : str | None = None ,
138
+ line_number : int | None = None ,
139
+ ) -> None :
120
140
joined = " " .join ([str (a ) for a in args ])
121
141
add , output = text_accumulator ()
122
142
if fail :
@@ -139,14 +159,22 @@ def warn_or_fail(fail=False, *args, filename=None, line_number=None):
139
159
sys .exit (- 1 )
140
160
141
161
142
- def warn (* args , filename = None , line_number = None ):
143
- return warn_or_fail (False , * args , filename = filename , line_number = line_number )
162
+ def warn (
163
+ * args : object ,
164
+ filename : str | None = None ,
165
+ line_number : int | None = None ,
166
+ ) -> None :
167
+ return warn_or_fail (* args , filename = filename , line_number = line_number , fail = False )
144
168
145
- def fail (* args , filename = None , line_number = None ):
146
- return warn_or_fail (True , * args , filename = filename , line_number = line_number )
169
+ def fail (
170
+ * args : object ,
171
+ filename : str | None = None ,
172
+ line_number : int | None = None ,
173
+ ) -> NoReturn :
174
+ warn_or_fail (* args , filename = filename , line_number = line_number , fail = True )
147
175
148
176
149
- def quoted_for_c_string (s ) :
177
+ def quoted_for_c_string (s : str ) -> str :
150
178
for old , new in (
151
179
('\\ ' , '\\ \\ ' ), # must be first!
152
180
('"' , '\\ "' ),
@@ -155,13 +183,13 @@ def quoted_for_c_string(s):
155
183
s = s .replace (old , new )
156
184
return s
157
185
158
- def c_repr (s ) :
186
+ def c_repr (s : str ) -> str :
159
187
return '"' + s + '"'
160
188
161
189
162
190
is_legal_c_identifier = re .compile ('^[A-Za-z_][A-Za-z0-9_]*$' ).match
163
191
164
- def is_legal_py_identifier (s ) :
192
+ def is_legal_py_identifier (s : str ) -> bool :
165
193
return all (is_legal_c_identifier (field ) for field in s .split ('.' ))
166
194
167
195
# identifiers that are okay in Python but aren't a good idea in C.
@@ -174,7 +202,7 @@ def is_legal_py_identifier(s):
174
202
typedef typeof union unsigned void volatile while
175
203
""" .strip ().split ())
176
204
177
- def ensure_legal_c_identifier (s ) :
205
+ def ensure_legal_c_identifier (s : str ) -> str :
178
206
# for now, just complain if what we're given isn't legal
179
207
if not is_legal_c_identifier (s ):
180
208
fail ("Illegal C identifier: {}" .format (s ))
@@ -183,22 +211,22 @@ def ensure_legal_c_identifier(s):
183
211
return s + "_value"
184
212
return s
185
213
186
- def rstrip_lines (s ) :
214
+ def rstrip_lines (s : str ) -> str :
187
215
text , add , output = _text_accumulator ()
188
216
for line in s .split ('\n ' ):
189
217
add (line .rstrip ())
190
218
add ('\n ' )
191
219
text .pop ()
192
220
return output ()
193
221
194
- def format_escape (s ) :
222
+ def format_escape (s : str ) -> str :
195
223
# double up curly-braces, this string will be used
196
224
# as part of a format_map() template later
197
225
s = s .replace ('{' , '{{' )
198
226
s = s .replace ('}' , '}}' )
199
227
return s
200
228
201
- def linear_format (s , ** kwargs ) :
229
+ def linear_format (s : str , ** kwargs : str ) -> str :
202
230
"""
203
231
Perform str.format-like substitution, except:
204
232
* The strings substituted must be on lines by
@@ -242,7 +270,7 @@ def linear_format(s, **kwargs):
242
270
243
271
return output ()[:- 1 ]
244
272
245
- def indent_all_lines (s , prefix ) :
273
+ def indent_all_lines (s : str , prefix : str ) -> str :
246
274
"""
247
275
Returns 's', with 'prefix' prepended to all lines.
248
276
@@ -263,7 +291,7 @@ def indent_all_lines(s, prefix):
263
291
final .append (last )
264
292
return '' .join (final )
265
293
266
- def suffix_all_lines (s , suffix ) :
294
+ def suffix_all_lines (s : str , suffix : str ) -> str :
267
295
"""
268
296
Returns 's', with 'suffix' appended to all lines.
269
297
@@ -283,7 +311,7 @@ def suffix_all_lines(s, suffix):
283
311
return '' .join (final )
284
312
285
313
286
- def version_splitter (s ) :
314
+ def version_splitter (s : str ) -> tuple [ int , ...] :
287
315
"""Splits a version string into a tuple of integers.
288
316
289
317
The following ASCII characters are allowed, and employ
@@ -294,7 +322,7 @@ def version_splitter(s):
294
322
(This permits Python-style version strings such as "1.4b3".)
295
323
"""
296
324
version = []
297
- accumulator = []
325
+ accumulator : list [ str ] = []
298
326
def flush ():
299
327
if not accumulator :
300
328
raise ValueError ('Unsupported version string: ' + repr (s ))
@@ -314,7 +342,7 @@ def flush():
314
342
flush ()
315
343
return tuple (version )
316
344
317
- def version_comparitor (version1 , version2 ) :
345
+ def version_comparitor (version1 : str , version2 : str ) -> Literal [ - 1 , 0 , 1 ] :
318
346
iterator = itertools .zip_longest (version_splitter (version1 ), version_splitter (version2 ), fillvalue = 0 )
319
347
for i , (a , b ) in enumerate (iterator ):
320
348
if a < b :
0 commit comments