Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cpython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
cpython
Commits
eee72d47
Commit
eee72d47
authored
Feb 27, 2018
by
Tobotimus
Committed by
Serhiy Storchaka
Feb 27, 2018
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
bpo-32222: Fix pygettext skipping docstrings for funcs with arg typehints (GH-4745)
parent
3a087bed
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
102 additions
and
4 deletions
+102
-4
Lib/test/test_tools/test_i18n.py
Lib/test/test_tools/test_i18n.py
+88
-0
Misc/NEWS.d/next/Tools-Demos/2017-12-07-20-51-20.bpo-32222.hPBcGT.rst
...next/Tools-Demos/2017-12-07-20-51-20.bpo-32222.hPBcGT.rst
+3
-0
Tools/i18n/pygettext.py
Tools/i18n/pygettext.py
+11
-4
No files found.
Lib/test/test_tools/test_i18n.py
View file @
eee72d47
...
@@ -3,6 +3,7 @@
...
@@ -3,6 +3,7 @@
import
os
import
os
import
sys
import
sys
import
unittest
import
unittest
import
textwrap
from
test.support.script_helper
import
assert_python_ok
from
test.support.script_helper
import
assert_python_ok
from
test.test_tools
import
skip_if_missing
,
toolsdir
from
test.test_tools
import
skip_if_missing
,
toolsdir
...
@@ -28,6 +29,41 @@ class Test_pygettext(unittest.TestCase):
...
@@ -28,6 +29,41 @@ class Test_pygettext(unittest.TestCase):
headers
[
key
]
=
val
.
strip
()
headers
[
key
]
=
val
.
strip
()
return
headers
return
headers
def
get_msgids
(
self
,
data
):
""" utility: return all msgids in .po file as a list of strings """
msgids
=
[]
reading_msgid
=
False
cur_msgid
=
[]
for
line
in
data
.
split
(
'
\
n
'
):
if
reading_msgid
:
if
line
.
startswith
(
'"'
):
cur_msgid
.
append
(
line
.
strip
(
'"'
))
else
:
msgids
.
append
(
'
\
n
'
.
join
(
cur_msgid
))
cur_msgid
=
[]
reading_msgid
=
False
continue
if
line
.
startswith
(
'msgid '
):
line
=
line
[
len
(
'msgid '
):]
cur_msgid
.
append
(
line
.
strip
(
'"'
))
reading_msgid
=
True
else
:
if
reading_msgid
:
msgids
.
append
(
'
\
n
'
.
join
(
cur_msgid
))
return
msgids
def
extract_docstrings_from_str
(
self
,
module_content
):
""" utility: return all msgids extracted from module_content """
filename
=
'test_docstrings.py'
with
temp_cwd
(
None
)
as
cwd
:
with
open
(
filename
,
'w'
)
as
fp
:
fp
.
write
(
module_content
)
assert_python_ok
(
self
.
script
,
'-D'
,
filename
)
with
open
(
'messages.pot'
)
as
fp
:
data
=
fp
.
read
()
return
self
.
get_msgids
(
data
)
def
test_header
(
self
):
def
test_header
(
self
):
"""Make sure the required fields are in the header, according to:
"""Make sure the required fields are in the header, according to:
http://www.gnu.org/software/gettext/manual/gettext.html#Header-Entry
http://www.gnu.org/software/gettext/manual/gettext.html#Header-Entry
...
@@ -72,3 +108,55 @@ class Test_pygettext(unittest.TestCase):
...
@@ -72,3 +108,55 @@ class Test_pygettext(unittest.TestCase):
# This will raise if the date format does not exactly match.
# This will raise if the date format does not exactly match.
datetime
.
strptime
(
creationDate
,
'%Y-%m-%d %H:%M%z'
)
datetime
.
strptime
(
creationDate
,
'%Y-%m-%d %H:%M%z'
)
def
test_funcdocstring_annotated_args
(
self
):
""" Test docstrings for functions with annotated args """
msgids
=
self
.
extract_docstrings_from_str
(
textwrap
.
dedent
(
'''
\
def foo(bar: str):
"""doc"""
'''
))
self
.
assertIn
(
'doc'
,
msgids
)
def
test_funcdocstring_annotated_return
(
self
):
""" Test docstrings for functions with annotated return type """
msgids
=
self
.
extract_docstrings_from_str
(
textwrap
.
dedent
(
'''
\
def foo(bar) -> str:
"""doc"""
'''
))
self
.
assertIn
(
'doc'
,
msgids
)
def
test_funcdocstring_defvalue_args
(
self
):
""" Test docstring for functions with default arg values """
msgids
=
self
.
extract_docstrings_from_str
(
textwrap
.
dedent
(
'''
\
def foo(bar=()):
"""doc"""
'''
))
self
.
assertIn
(
'doc'
,
msgids
)
def
test_funcdocstring_multiple_funcs
(
self
):
""" Test docstring extraction for multiple functions combining
annotated args, annotated return types and default arg values
"""
msgids
=
self
.
extract_docstrings_from_str
(
textwrap
.
dedent
(
'''
\
def foo1(bar: tuple=()) -> str:
"""doc1"""
def foo2(bar: List[1:2]) -> (lambda x: x):
"""doc2"""
def foo3(bar: 'func'=lambda x: x) -> {1: 2}:
"""doc3"""
'''
))
self
.
assertIn
(
'doc1'
,
msgids
)
self
.
assertIn
(
'doc2'
,
msgids
)
self
.
assertIn
(
'doc3'
,
msgids
)
def
test_classdocstring_early_colon
(
self
):
""" Test docstring extraction for a class with colons occuring within
the parentheses.
"""
msgids
=
self
.
extract_docstrings_from_str
(
textwrap
.
dedent
(
'''
\
class D(L[1:2], F({1: 2}), metaclass=M(lambda x: x)):
"""doc"""
'''
))
self
.
assertIn
(
'doc'
,
msgids
)
Misc/NEWS.d/next/Tools-Demos/2017-12-07-20-51-20.bpo-32222.hPBcGT.rst
0 → 100644
View file @
eee72d47
Fix pygettext not extracting docstrings for functions with type annotated
arguments.
Patch by Toby Harradine.
Tools/i18n/pygettext.py
View file @
eee72d47
...
@@ -320,6 +320,7 @@ class TokenEater:
...
@@ -320,6 +320,7 @@ class TokenEater:
self
.
__lineno
=
-
1
self
.
__lineno
=
-
1
self
.
__freshmodule
=
1
self
.
__freshmodule
=
1
self
.
__curfile
=
None
self
.
__curfile
=
None
self
.
__enclosurecount
=
0
def
__call__
(
self
,
ttype
,
tstring
,
stup
,
etup
,
line
):
def
__call__
(
self
,
ttype
,
tstring
,
stup
,
etup
,
line
):
# dispatch
# dispatch
...
@@ -340,7 +341,7 @@ class TokenEater:
...
@@ -340,7 +341,7 @@ class TokenEater:
elif
ttype
not
in
(
tokenize
.
COMMENT
,
tokenize
.
NL
):
elif
ttype
not
in
(
tokenize
.
COMMENT
,
tokenize
.
NL
):
self
.
__freshmodule
=
0
self
.
__freshmodule
=
0
return
return
# class docstring?
# class
or func/method
docstring?
if
ttype
==
tokenize
.
NAME
and
tstring
in
(
'class'
,
'def'
):
if
ttype
==
tokenize
.
NAME
and
tstring
in
(
'class'
,
'def'
):
self
.
__state
=
self
.
__suiteseen
self
.
__state
=
self
.
__suiteseen
return
return
...
@@ -348,9 +349,15 @@ class TokenEater:
...
@@ -348,9 +349,15 @@ class TokenEater:
self
.
__state
=
self
.
__keywordseen
self
.
__state
=
self
.
__keywordseen
def
__suiteseen
(
self
,
ttype
,
tstring
,
lineno
):
def
__suiteseen
(
self
,
ttype
,
tstring
,
lineno
):
# ignore anything until we see the colon
# skip over any enclosure pairs until we see the colon
if
ttype
==
tokenize
.
OP
and
tstring
==
':'
:
if
ttype
==
tokenize
.
OP
:
if
tstring
==
':'
and
self
.
__enclosurecount
==
0
:
# we see a colon and we're not in an enclosure: end of def
self
.
__state
=
self
.
__suitedocstring
self
.
__state
=
self
.
__suitedocstring
elif
tstring
in
'([{'
:
self
.
__enclosurecount
+=
1
elif
tstring
in
')]}'
:
self
.
__enclosurecount
-=
1
def
__suitedocstring
(
self
,
ttype
,
tstring
,
lineno
):
def
__suitedocstring
(
self
,
ttype
,
tstring
,
lineno
):
# ignore any intervening noise
# ignore any intervening noise
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment