mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 03:04:41 +00:00 
			
		
		
		
	bpo-33064: lib2to3: support trailing comma after *args and **kwargs (#6096)
New tests also added. I also made the comments in line with the builtin Grammar/Grammar. PEP 306 was withdrawn, Kees Blom's railroad program has been lost to the sands of time for at least 16 years now (I found a python-dev post from people looking for it).
This commit is contained in:
		
							parent
							
								
									a34510a4c5
								
							
						
					
					
						commit
						b51f5de711
					
				
					 3 changed files with 39 additions and 26 deletions
				
			
		|  | @ -1,26 +1,7 @@ | ||||||
| # Grammar for 2to3. This grammar supports Python 2.x and 3.x. | # Grammar for 2to3. This grammar supports Python 2.x and 3.x. | ||||||
| 
 | 
 | ||||||
| # Note:  Changing the grammar specified in this file will most likely | # NOTE WELL: You should also follow all the steps listed at | ||||||
| #        require corresponding changes in the parser module | # https://devguide.python.org/grammar/ | ||||||
| #        (../Modules/parsermodule.c).  If you can't make the changes to |  | ||||||
| #        that module yourself, please co-ordinate the required changes |  | ||||||
| #        with someone who can; ask around on python-dev for help.  Fred |  | ||||||
| #        Drake <fdrake@acm.org> will probably be listening there. |  | ||||||
| 
 |  | ||||||
| # NOTE WELL: You should also follow all the steps listed in PEP 306, |  | ||||||
| # "How to Change Python's Grammar" |  | ||||||
| 
 |  | ||||||
| # Commands for Kees Blom's railroad program |  | ||||||
| #diagram:token NAME |  | ||||||
| #diagram:token NUMBER |  | ||||||
| #diagram:token STRING |  | ||||||
| #diagram:token NEWLINE |  | ||||||
| #diagram:token ENDMARKER |  | ||||||
| #diagram:token INDENT |  | ||||||
| #diagram:output\input python.bla |  | ||||||
| #diagram:token DEDENT |  | ||||||
| #diagram:output\textwidth 20.04cm\oddsidemargin  0.0cm\evensidemargin 0.0cm |  | ||||||
| #diagram:rules |  | ||||||
| 
 | 
 | ||||||
| # Start symbols for the grammar: | # Start symbols for the grammar: | ||||||
| #	file_input is a module or sequence of commands read from an input file; | #	file_input is a module or sequence of commands read from an input file; | ||||||
|  | @ -38,13 +19,13 @@ async_funcdef: 'async' funcdef | ||||||
| funcdef: 'def' NAME parameters ['->' test] ':' suite | funcdef: 'def' NAME parameters ['->' test] ':' suite | ||||||
| parameters: '(' [typedargslist] ')' | parameters: '(' [typedargslist] ')' | ||||||
| typedargslist: ((tfpdef ['=' test] ',')* | typedargslist: ((tfpdef ['=' test] ',')* | ||||||
|                 ('*' [tname] (',' tname ['=' test])* [',' '**' tname] | '**' tname) |                 ('*' [tname] (',' tname ['=' test])* [',' ['**' tname [',']]] | '**' tname [',']) | ||||||
|                 | tfpdef ['=' test] (',' tfpdef ['=' test])* [',']) |                 | tfpdef ['=' test] (',' tfpdef ['=' test])* [',']) | ||||||
| tname: NAME [':' test] | tname: NAME [':' test] | ||||||
| tfpdef: tname | '(' tfplist ')' | tfpdef: tname | '(' tfplist ')' | ||||||
| tfplist: tfpdef (',' tfpdef)* [','] | tfplist: tfpdef (',' tfpdef)* [','] | ||||||
| varargslist: ((vfpdef ['=' test] ',')* | varargslist: ((vfpdef ['=' test] ',')* | ||||||
|               ('*' [vname] (',' vname ['=' test])*  [',' '**' vname] | '**' vname) |               ('*' [vname] (',' vname ['=' test])*  [',' ['**' vname [',']]] | '**' vname [',']) | ||||||
|               | vfpdef ['=' test] (',' vfpdef ['=' test])* [',']) |               | vfpdef ['=' test] (',' vfpdef ['=' test])* [',']) | ||||||
| vname: NAME | vname: NAME | ||||||
| vfpdef: vname | '(' vfplist ')' | vfpdef: vname | '(' vfplist ')' | ||||||
|  |  | ||||||
|  | @ -9,7 +9,6 @@ | ||||||
| # Testing imports | # Testing imports | ||||||
| from . import support | from . import support | ||||||
| from .support import driver, driver_no_print_statement | from .support import driver, driver_no_print_statement | ||||||
| from test.support import verbose |  | ||||||
| 
 | 
 | ||||||
| # Python imports | # Python imports | ||||||
| import difflib | import difflib | ||||||
|  | @ -22,7 +21,6 @@ | ||||||
| import sys | import sys | ||||||
| import tempfile | import tempfile | ||||||
| import unittest | import unittest | ||||||
| import warnings |  | ||||||
| 
 | 
 | ||||||
| # Local imports | # Local imports | ||||||
| from lib2to3.pgen2 import driver as pgen2_driver | from lib2to3.pgen2 import driver as pgen2_driver | ||||||
|  | @ -305,6 +303,38 @@ def test_8(self): | ||||||
|                         *g:6, h:7, i=8, j:9=10, **k:11) -> 12: pass""" |                         *g:6, h:7, i=8, j:9=10, **k:11) -> 12: pass""" | ||||||
|         self.validate(s) |         self.validate(s) | ||||||
| 
 | 
 | ||||||
|  |     def test_9(self): | ||||||
|  |         s = """def f( | ||||||
|  |           a: str, | ||||||
|  |           b: int, | ||||||
|  |           *, | ||||||
|  |           c: bool = False, | ||||||
|  |           **kwargs, | ||||||
|  |         ) -> None: | ||||||
|  |             call(c=c, **kwargs,)""" | ||||||
|  |         self.validate(s) | ||||||
|  | 
 | ||||||
|  |     def test_10(self): | ||||||
|  |         s = """def f( | ||||||
|  |           a: str, | ||||||
|  |         ) -> None: | ||||||
|  |             call(a,)""" | ||||||
|  |         self.validate(s) | ||||||
|  | 
 | ||||||
|  |     def test_11(self): | ||||||
|  |         s = """def f( | ||||||
|  |           a: str = '', | ||||||
|  |         ) -> None: | ||||||
|  |             call(a=a,)""" | ||||||
|  |         self.validate(s) | ||||||
|  | 
 | ||||||
|  |     def test_12(self): | ||||||
|  |         s = """def f( | ||||||
|  |           *args: str, | ||||||
|  |         ) -> None: | ||||||
|  |             call(*args,)""" | ||||||
|  |         self.validate(s) | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| # Adapted from Python 3's Lib/test/test_grammar.py:GrammarTests.test_var_annot | # Adapted from Python 3's Lib/test/test_grammar.py:GrammarTests.test_var_annot | ||||||
| class TestVarAnnotations(GrammarTest): | class TestVarAnnotations(GrammarTest): | ||||||
|  | @ -407,7 +437,7 @@ def test_new_syntax(self): | ||||||
|         self.validate("class B(t, *args): pass") |         self.validate("class B(t, *args): pass") | ||||||
|         self.validate("class B(t, **kwargs): pass") |         self.validate("class B(t, **kwargs): pass") | ||||||
|         self.validate("class B(t, *args, **kwargs): pass") |         self.validate("class B(t, *args, **kwargs): pass") | ||||||
|         self.validate("class B(t, y=9, *args, **kwargs): pass") |         self.validate("class B(t, y=9, *args, **kwargs,): pass") | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class TestParserIdempotency(support.TestCase): | class TestParserIdempotency(support.TestCase): | ||||||
|  |  | ||||||
|  | @ -0,0 +1,2 @@ | ||||||
|  | lib2to3 now properly supports trailing commas after ``*args`` and | ||||||
|  | ``**kwargs`` in function signatures. | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Łukasz Langa
						Łukasz Langa