mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-05-22 21:32:31 +00:00
python-3.6.zip added from Github
README.cosmo contains the necessary links.
This commit is contained in:
parent
75fc601ff5
commit
0c4c56ff39
4219 changed files with 1968626 additions and 0 deletions
618
third_party/python/Lib/test/test_syntax.py
vendored
Normal file
618
third_party/python/Lib/test/test_syntax.py
vendored
Normal file
|
@ -0,0 +1,618 @@
|
|||
"""This module tests SyntaxErrors.
|
||||
|
||||
Here's an example of the sort of thing that is tested.
|
||||
|
||||
>>> def f(x):
|
||||
... global x
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: name 'x' is parameter and global
|
||||
|
||||
The tests are all raise SyntaxErrors. They were created by checking
|
||||
each C call that raises SyntaxError. There are several modules that
|
||||
raise these exceptions-- ast.c, compile.c, future.c, pythonrun.c, and
|
||||
symtable.c.
|
||||
|
||||
The parser itself outlaws a lot of invalid syntax. None of these
|
||||
errors are tested here at the moment. We should add some tests; since
|
||||
there are infinitely many programs with invalid syntax, we would need
|
||||
to be judicious in selecting some.
|
||||
|
||||
The compiler generates a synthetic module name for code executed by
|
||||
doctest. Since all the code comes from the same module, a suffix like
|
||||
[1] is appended to the module name, As a consequence, changing the
|
||||
order of tests in this module means renumbering all the errors after
|
||||
it. (Maybe we should enable the ellipsis option for these tests.)
|
||||
|
||||
In ast.c, syntax errors are raised by calling ast_error().
|
||||
|
||||
Errors from set_context():
|
||||
|
||||
>>> obj.None = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
>>> None = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to keyword
|
||||
|
||||
>>> f() = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to function call
|
||||
|
||||
>>> del f()
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't delete function call
|
||||
|
||||
>>> a + 1 = 2
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to operator
|
||||
|
||||
>>> (x for x in x) = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to generator expression
|
||||
|
||||
>>> 1 = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to literal
|
||||
|
||||
>>> "abc" = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to literal
|
||||
|
||||
>>> b"" = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to literal
|
||||
|
||||
>>> `1` = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
If the left-hand side of an assignment is a list or tuple, an illegal
|
||||
expression inside that contain should still cause a syntax error.
|
||||
This test just checks a couple of cases rather than enumerating all of
|
||||
them.
|
||||
|
||||
>>> (a, "b", c) = (1, 2, 3)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to literal
|
||||
|
||||
>>> [a, b, c + 1] = [1, 2, 3]
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to operator
|
||||
|
||||
>>> a if 1 else b = 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to conditional expression
|
||||
|
||||
From compiler_complex_args():
|
||||
|
||||
>>> def f(None=1):
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
|
||||
From ast_for_arguments():
|
||||
|
||||
>>> def f(x, y=1, z):
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: non-default argument follows default argument
|
||||
|
||||
>>> def f(x, None):
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
>>> def f(*None):
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
>>> def f(**None):
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
|
||||
From ast_for_funcdef():
|
||||
|
||||
>>> def None(x):
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
|
||||
From ast_for_call():
|
||||
|
||||
>>> def f(it, *varargs):
|
||||
... return list(it)
|
||||
>>> L = range(10)
|
||||
>>> f(x for x in L)
|
||||
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
|
||||
>>> f(x for x in L, 1)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: Generator expression must be parenthesized if not sole argument
|
||||
>>> f(x for x in L, y for y in L)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: Generator expression must be parenthesized if not sole argument
|
||||
>>> f((x for x in L), 1)
|
||||
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
|
||||
|
||||
>>> f(i0, i1, i2, i3, i4, i5, i6, i7, i8, i9, i10, i11,
|
||||
... i12, i13, i14, i15, i16, i17, i18, i19, i20, i21, i22,
|
||||
... i23, i24, i25, i26, i27, i28, i29, i30, i31, i32, i33,
|
||||
... i34, i35, i36, i37, i38, i39, i40, i41, i42, i43, i44,
|
||||
... i45, i46, i47, i48, i49, i50, i51, i52, i53, i54, i55,
|
||||
... i56, i57, i58, i59, i60, i61, i62, i63, i64, i65, i66,
|
||||
... i67, i68, i69, i70, i71, i72, i73, i74, i75, i76, i77,
|
||||
... i78, i79, i80, i81, i82, i83, i84, i85, i86, i87, i88,
|
||||
... i89, i90, i91, i92, i93, i94, i95, i96, i97, i98, i99,
|
||||
... i100, i101, i102, i103, i104, i105, i106, i107, i108,
|
||||
... i109, i110, i111, i112, i113, i114, i115, i116, i117,
|
||||
... i118, i119, i120, i121, i122, i123, i124, i125, i126,
|
||||
... i127, i128, i129, i130, i131, i132, i133, i134, i135,
|
||||
... i136, i137, i138, i139, i140, i141, i142, i143, i144,
|
||||
... i145, i146, i147, i148, i149, i150, i151, i152, i153,
|
||||
... i154, i155, i156, i157, i158, i159, i160, i161, i162,
|
||||
... i163, i164, i165, i166, i167, i168, i169, i170, i171,
|
||||
... i172, i173, i174, i175, i176, i177, i178, i179, i180,
|
||||
... i181, i182, i183, i184, i185, i186, i187, i188, i189,
|
||||
... i190, i191, i192, i193, i194, i195, i196, i197, i198,
|
||||
... i199, i200, i201, i202, i203, i204, i205, i206, i207,
|
||||
... i208, i209, i210, i211, i212, i213, i214, i215, i216,
|
||||
... i217, i218, i219, i220, i221, i222, i223, i224, i225,
|
||||
... i226, i227, i228, i229, i230, i231, i232, i233, i234,
|
||||
... i235, i236, i237, i238, i239, i240, i241, i242, i243,
|
||||
... i244, i245, i246, i247, i248, i249, i250, i251, i252,
|
||||
... i253, i254, i255)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: more than 255 arguments
|
||||
|
||||
The actual error cases counts positional arguments, keyword arguments,
|
||||
and generator expression arguments separately. This test combines the
|
||||
three.
|
||||
|
||||
>>> f(i0, i1, i2, i3, i4, i5, i6, i7, i8, i9, i10, i11,
|
||||
... i12, i13, i14, i15, i16, i17, i18, i19, i20, i21, i22,
|
||||
... i23, i24, i25, i26, i27, i28, i29, i30, i31, i32, i33,
|
||||
... i34, i35, i36, i37, i38, i39, i40, i41, i42, i43, i44,
|
||||
... i45, i46, i47, i48, i49, i50, i51, i52, i53, i54, i55,
|
||||
... i56, i57, i58, i59, i60, i61, i62, i63, i64, i65, i66,
|
||||
... i67, i68, i69, i70, i71, i72, i73, i74, i75, i76, i77,
|
||||
... i78, i79, i80, i81, i82, i83, i84, i85, i86, i87, i88,
|
||||
... i89, i90, i91, i92, i93, i94, i95, i96, i97, i98, i99,
|
||||
... i100, i101, i102, i103, i104, i105, i106, i107, i108,
|
||||
... i109, i110, i111, i112, i113, i114, i115, i116, i117,
|
||||
... i118, i119, i120, i121, i122, i123, i124, i125, i126,
|
||||
... i127, i128, i129, i130, i131, i132, i133, i134, i135,
|
||||
... i136, i137, i138, i139, i140, i141, i142, i143, i144,
|
||||
... i145, i146, i147, i148, i149, i150, i151, i152, i153,
|
||||
... i154, i155, i156, i157, i158, i159, i160, i161, i162,
|
||||
... i163, i164, i165, i166, i167, i168, i169, i170, i171,
|
||||
... i172, i173, i174, i175, i176, i177, i178, i179, i180,
|
||||
... i181, i182, i183, i184, i185, i186, i187, i188, i189,
|
||||
... i190, i191, i192, i193, i194, i195, i196, i197, i198,
|
||||
... i199, i200, i201, i202, i203, i204, i205, i206, i207,
|
||||
... i208, i209, i210, i211, i212, i213, i214, i215, i216,
|
||||
... i217, i218, i219, i220, i221, i222, i223, i224, i225,
|
||||
... i226, i227, i228, i229, i230, i231, i232, i233, i234,
|
||||
... i235, i236, i237, i238, i239, i240, i241, i242, i243,
|
||||
... (x for x in i244), i245, i246, i247, i248, i249, i250, i251,
|
||||
... i252=1, i253=1, i254=1, i255=1)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: more than 255 arguments
|
||||
|
||||
>>> f(lambda x: x[0] = 3)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: lambda cannot contain assignment
|
||||
|
||||
The grammar accepts any test (basically, any expression) in the
|
||||
keyword slot of a call site. Test a few different options.
|
||||
|
||||
>>> f(x()=2)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: keyword can't be an expression
|
||||
>>> f(a or b=1)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: keyword can't be an expression
|
||||
>>> f(x.y=1)
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: keyword can't be an expression
|
||||
|
||||
|
||||
More set_context():
|
||||
|
||||
>>> (x for x in x) += 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to generator expression
|
||||
>>> None += 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to keyword
|
||||
>>> f() += 1
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to function call
|
||||
|
||||
|
||||
Test continue in finally in weird combinations.
|
||||
|
||||
continue in for loop under finally should be ok.
|
||||
|
||||
>>> def test():
|
||||
... try:
|
||||
... pass
|
||||
... finally:
|
||||
... for abc in range(10):
|
||||
... continue
|
||||
... print(abc)
|
||||
>>> test()
|
||||
9
|
||||
|
||||
Start simple, a continue in a finally should not be allowed.
|
||||
|
||||
>>> def test():
|
||||
... for abc in range(10):
|
||||
... try:
|
||||
... pass
|
||||
... finally:
|
||||
... continue
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: 'continue' not supported inside 'finally' clause
|
||||
|
||||
This is essentially a continue in a finally which should not be allowed.
|
||||
|
||||
>>> def test():
|
||||
... for abc in range(10):
|
||||
... try:
|
||||
... pass
|
||||
... finally:
|
||||
... try:
|
||||
... continue
|
||||
... except:
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: 'continue' not supported inside 'finally' clause
|
||||
|
||||
>>> def foo():
|
||||
... try:
|
||||
... pass
|
||||
... finally:
|
||||
... continue
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: 'continue' not supported inside 'finally' clause
|
||||
|
||||
>>> def foo():
|
||||
... for a in ():
|
||||
... try:
|
||||
... pass
|
||||
... finally:
|
||||
... continue
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: 'continue' not supported inside 'finally' clause
|
||||
|
||||
>>> def foo():
|
||||
... for a in ():
|
||||
... try:
|
||||
... pass
|
||||
... finally:
|
||||
... try:
|
||||
... continue
|
||||
... finally:
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: 'continue' not supported inside 'finally' clause
|
||||
|
||||
>>> def foo():
|
||||
... for a in ():
|
||||
... try: pass
|
||||
... finally:
|
||||
... try:
|
||||
... pass
|
||||
... except:
|
||||
... continue
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: 'continue' not supported inside 'finally' clause
|
||||
|
||||
There is one test for a break that is not in a loop. The compiler
|
||||
uses a single data structure to keep track of try-finally and loops,
|
||||
so we need to be sure that a break is actually inside a loop. If it
|
||||
isn't, there should be a syntax error.
|
||||
|
||||
>>> try:
|
||||
... print(1)
|
||||
... break
|
||||
... print(2)
|
||||
... finally:
|
||||
... print(3)
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: 'break' outside loop
|
||||
|
||||
This raises a SyntaxError, it used to raise a SystemError.
|
||||
Context for this change can be found on issue #27514
|
||||
|
||||
In 2.5 there was a missing exception and an assert was triggered in a debug
|
||||
build. The number of blocks must be greater than CO_MAXBLOCKS. SF #1565514
|
||||
|
||||
>>> while 1:
|
||||
... while 2:
|
||||
... while 3:
|
||||
... while 4:
|
||||
... while 5:
|
||||
... while 6:
|
||||
... while 8:
|
||||
... while 9:
|
||||
... while 10:
|
||||
... while 11:
|
||||
... while 12:
|
||||
... while 13:
|
||||
... while 14:
|
||||
... while 15:
|
||||
... while 16:
|
||||
... while 17:
|
||||
... while 18:
|
||||
... while 19:
|
||||
... while 20:
|
||||
... while 21:
|
||||
... while 22:
|
||||
... break
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: too many statically nested blocks
|
||||
|
||||
Misuse of the nonlocal and global statement can lead to a few unique syntax errors.
|
||||
|
||||
>>> def f():
|
||||
... x = 1
|
||||
... global x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: name 'x' is assigned to before global declaration
|
||||
|
||||
>>> def f():
|
||||
... x = 1
|
||||
... def g():
|
||||
... print(x)
|
||||
... nonlocal x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: name 'x' is used prior to nonlocal declaration
|
||||
|
||||
>>> def f():
|
||||
... x = 1
|
||||
... def g():
|
||||
... x = 2
|
||||
... nonlocal x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: name 'x' is assigned to before nonlocal declaration
|
||||
|
||||
>>> def f(x):
|
||||
... nonlocal x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: name 'x' is parameter and nonlocal
|
||||
|
||||
>>> def f():
|
||||
... global x
|
||||
... nonlocal x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: name 'x' is nonlocal and global
|
||||
|
||||
>>> def f():
|
||||
... nonlocal x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: no binding for nonlocal 'x' found
|
||||
|
||||
From SF bug #1705365
|
||||
>>> nonlocal x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: nonlocal declaration not allowed at module level
|
||||
|
||||
From https://bugs.python.org/issue25973
|
||||
>>> class A:
|
||||
... def f(self):
|
||||
... nonlocal __x
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: no binding for nonlocal '_A__x' found
|
||||
|
||||
|
||||
This tests assignment-context; there was a bug in Python 2.5 where compiling
|
||||
a complex 'if' (one with 'elif') would fail to notice an invalid suite,
|
||||
leading to spurious errors.
|
||||
|
||||
>>> if 1:
|
||||
... x() = 1
|
||||
... elif 1:
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: can't assign to function call
|
||||
|
||||
>>> if 1:
|
||||
... pass
|
||||
... elif 1:
|
||||
... x() = 1
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: can't assign to function call
|
||||
|
||||
>>> if 1:
|
||||
... x() = 1
|
||||
... elif 1:
|
||||
... pass
|
||||
... else:
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: can't assign to function call
|
||||
|
||||
>>> if 1:
|
||||
... pass
|
||||
... elif 1:
|
||||
... x() = 1
|
||||
... else:
|
||||
... pass
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: can't assign to function call
|
||||
|
||||
>>> if 1:
|
||||
... pass
|
||||
... elif 1:
|
||||
... pass
|
||||
... else:
|
||||
... x() = 1
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: can't assign to function call
|
||||
|
||||
Make sure that the old "raise X, Y[, Z]" form is gone:
|
||||
>>> raise X, Y
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: invalid syntax
|
||||
>>> raise X, Y, Z
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: invalid syntax
|
||||
|
||||
|
||||
>>> f(a=23, a=234)
|
||||
Traceback (most recent call last):
|
||||
...
|
||||
SyntaxError: keyword argument repeated
|
||||
|
||||
>>> {1, 2, 3} = 42
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: can't assign to literal
|
||||
|
||||
Corner-cases that used to fail to raise the correct error:
|
||||
|
||||
>>> def f(*, x=lambda __debug__:0): pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: assignment to keyword
|
||||
|
||||
>>> def f(*args:(lambda __debug__:0)): pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: assignment to keyword
|
||||
|
||||
>>> def f(**kwargs:(lambda __debug__:0)): pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: assignment to keyword
|
||||
|
||||
>>> with (lambda *:0): pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: named arguments must follow bare *
|
||||
|
||||
Corner-cases that used to crash:
|
||||
|
||||
>>> def f(**__debug__): pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: assignment to keyword
|
||||
|
||||
>>> def f(*xx, __debug__): pass
|
||||
Traceback (most recent call last):
|
||||
SyntaxError: assignment to keyword
|
||||
|
||||
"""
|
||||
|
||||
import re
|
||||
import unittest
|
||||
import warnings
|
||||
|
||||
from test import support
|
||||
|
||||
class SyntaxTestCase(unittest.TestCase):
|
||||
|
||||
def _check_error(self, code, errtext,
|
||||
filename="<testcase>", mode="exec", subclass=None, lineno=None, offset=None):
|
||||
"""Check that compiling code raises SyntaxError with errtext.
|
||||
|
||||
errtest is a regular expression that must be present in the
|
||||
test of the exception raised. If subclass is specified it
|
||||
is the expected subclass of SyntaxError (e.g. IndentationError).
|
||||
"""
|
||||
try:
|
||||
compile(code, filename, mode)
|
||||
except SyntaxError as err:
|
||||
if subclass and not isinstance(err, subclass):
|
||||
self.fail("SyntaxError is not a %s" % subclass.__name__)
|
||||
mo = re.search(errtext, str(err))
|
||||
if mo is None:
|
||||
self.fail("SyntaxError did not contain '%r'" % (errtext,))
|
||||
self.assertEqual(err.filename, filename)
|
||||
if lineno is not None:
|
||||
self.assertEqual(err.lineno, lineno)
|
||||
if offset is not None:
|
||||
self.assertEqual(err.offset, offset)
|
||||
else:
|
||||
self.fail("compile() did not raise SyntaxError")
|
||||
|
||||
def test_assign_call(self):
|
||||
self._check_error("f() = 1", "assign")
|
||||
|
||||
def test_assign_del(self):
|
||||
self._check_error("del f()", "delete")
|
||||
|
||||
def test_global_err_then_warn(self):
|
||||
# Bug #763201: The SyntaxError raised for one global statement
|
||||
# shouldn't be clobbered by a SyntaxWarning issued for a later one.
|
||||
source = """if 1:
|
||||
def error(a):
|
||||
global a # SyntaxError
|
||||
def warning():
|
||||
b = 1
|
||||
global b # SyntaxWarning
|
||||
"""
|
||||
warnings.filterwarnings(action='ignore', category=SyntaxWarning)
|
||||
self._check_error(source, "global")
|
||||
warnings.filters.pop(0)
|
||||
|
||||
def test_break_outside_loop(self):
|
||||
self._check_error("break", "outside loop")
|
||||
|
||||
def test_unexpected_indent(self):
|
||||
self._check_error("foo()\n bar()\n", "unexpected indent",
|
||||
subclass=IndentationError)
|
||||
|
||||
def test_no_indent(self):
|
||||
self._check_error("if 1:\nfoo()", "expected an indented block",
|
||||
subclass=IndentationError)
|
||||
|
||||
def test_bad_outdent(self):
|
||||
self._check_error("if 1:\n foo()\n bar()",
|
||||
"unindent does not match .* level",
|
||||
subclass=IndentationError)
|
||||
|
||||
def test_kwargs_last(self):
|
||||
self._check_error("int(base=10, '2')",
|
||||
"positional argument follows keyword argument")
|
||||
|
||||
def test_kwargs_last2(self):
|
||||
self._check_error("int(**{'base': 10}, '2')",
|
||||
"positional argument follows "
|
||||
"keyword argument unpacking")
|
||||
|
||||
def test_kwargs_last3(self):
|
||||
self._check_error("int(**{'base': 10}, *['2'])",
|
||||
"iterable argument unpacking follows "
|
||||
"keyword argument unpacking")
|
||||
|
||||
def test_main():
|
||||
support.run_unittest(SyntaxTestCase)
|
||||
from test import test_syntax
|
||||
support.run_doctest(test_syntax, verbosity=True)
|
||||
|
||||
if __name__ == "__main__":
|
||||
test_main()
|
Loading…
Add table
Add a link
Reference in a new issue