- 
          
- 
                Notifications
    You must be signed in to change notification settings 
- Fork 33.2k
gh-108512: Add and use new replacements for PySys_GetObject() #111035
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from 5 commits
4d0f508
              eb42b39
              2d4588d
              2eb9533
              9fc2f3d
              65713ce
              e6ecf11
              8a0f5f2
              516829e
              9503aaf
              e2857ef
              dc26ec2
              104dcc2
              cf75fc3
              9434659
              56639d8
              b40a665
              03b9c0a
              5d793c5
              09869ed
              439bc3c
              93ab31b
              154a82a
              81c7605
              File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|  | @@ -222,10 +222,55 @@ These are utility functions that make functionality from the :mod:`sys` module | |||||||||||||||||||||||||
| accessible to C code. They all work with the current interpreter thread's | ||||||||||||||||||||||||||
| :mod:`sys` module's dict, which is contained in the internal thread state structure. | ||||||||||||||||||||||||||
|  | ||||||||||||||||||||||||||
| .. c:function:: PyObject *PySys_GetAttr(PyObject *name) | ||||||||||||||||||||||||||
|  | ||||||||||||||||||||||||||
| Get the attribute *name* of the :mod:`sys` module. Return a :term:`strong reference`. | ||||||||||||||||||||||||||
| Raise :exc:`RuntimeError` and return ``NULL`` if it does not exist. | ||||||||||||||||||||||||||
|          | ||||||||||||||||||||||||||
| Raise :exc:`RuntimeError` and return ``NULL`` if it does not exist. | |
| Raise :exc:`RuntimeError` and return ``NULL`` if it does not exist or if the :mod:`sys` module cannot be found. | 
        
          
              
                Outdated
          
        
      There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I suggest to use a list and start with the return value, so it's easier to see the 3 cases:
| If the object exists, set *\*result* to a new :term:`strong reference` | |
| to the object and return ``1``. | |
| If the object does not exist, set *\*result* to ``NULL`` and return ``0``, | |
| without setting an exception. | |
| If other error occurred, set an exception, set *\*result* to ``NULL`` and | |
| return ``-1``. | |
| * Return ``1`` and set *\*result* to a new :term:`strong reference` | |
| to the object if the attribute exists. | |
| * Return ``0`` without setting an exception and set *\*result* to ``NULL`` | |
| if the attribute does not exist. | |
| * Return ``-1``, set an exception and set *\*result* to ``NULL`` | |
| if an error occurred. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"Return and then set exception and variable" looks like a wrong sequence to me. It cannot do anything after returning.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just say it in the opposite order in this case:
Set an exception, set *\*result* to ``NULL``, and return ``-1``, if an error occurred.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
|  | @@ -1098,6 +1098,55 @@ class Data(_testcapi.ObjExtraData): | |
| del d.extra | ||
| self.assertIsNone(d.extra) | ||
|  | ||
| def test_sys_getattr(self): | ||
| sys_getattr = _testcapi.sys_getattr | ||
|  | ||
| self.assertIs(sys_getattr('stdout'), sys.stdout) | ||
| with support.swap_attr(sys, '\U0001f40d', 42): | ||
| self.assertEqual(sys_getattr('\U0001f40d'), 42) | ||
|  | ||
| with self.assertRaisesRegex(RuntimeError, r'lost sys\.nonexisting'): | ||
|          | ||
| sys_getattr('nonexisting') | ||
| with self.assertRaisesRegex(RuntimeError, r'lost sys\.1'): | ||
| sys_getattr(1) | ||
| self.assertRaises(TypeError, sys_getattr, []) | ||
| # CRASHES sys_getattr(NULL) | ||
|  | ||
| def test_sys_getattrstring(self): | ||
| sys_getattr = _testcapi.sys_getattrstring | ||
|  | ||
| self.assertIs(sys_getattr(b'stdout'), sys.stdout) | ||
| with support.swap_attr(sys, '\U0001f40d', 42): | ||
| self.assertEqual(sys_getattr('\U0001f40d'.encode()), 42) | ||
|  | ||
| with self.assertRaisesRegex(RuntimeError, r'lost sys\.nonexisting'): | ||
| sys_getattr(b'nonexisting') | ||
| self.assertRaises(UnicodeDecodeError, sys_getattr, b'\xff') | ||
| # CRASHES sys_getattr(NULL) | ||
|  | ||
| def test_sys_getoptionalattr(self): | ||
| sys_getattr = _testcapi.sys_getoptionalattr | ||
|          | ||
|  | ||
| self.assertIs(sys_getattr('stdout'), sys.stdout) | ||
| with support.swap_attr(sys, '\U0001f40d', 42): | ||
| self.assertEqual(sys_getattr('\U0001f40d'), 42) | ||
|  | ||
| self.assertIs(sys_getattr('nonexisting'), AttributeError) | ||
| self.assertIs(sys_getattr(1), AttributeError) | ||
| self.assertRaises(TypeError, sys_getattr, []) | ||
| # CRASHES sys_getattr(NULL) | ||
|  | ||
| def test_sys_getoptionalattrstring(self): | ||
| sys_getattr = _testcapi.sys_getoptionalattrstring | ||
|  | ||
| self.assertIs(sys_getattr(b'stdout'), sys.stdout) | ||
| with support.swap_attr(sys, '\U0001f40d', 42): | ||
| self.assertEqual(sys_getattr('\U0001f40d'.encode()), 42) | ||
|  | ||
| self.assertIs(sys_getattr(b'nonexisting'), AttributeError) | ||
| self.assertRaises(UnicodeDecodeError, sys_getattr, b'\xff') | ||
|          | ||
| # CRASHES sys_getattr(NULL) | ||
|  | ||
| def test_sys_getobject(self): | ||
| getobject = _testcapi.sys_getobject | ||
|  | ||
|  | ||
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
| @@ -0,0 +1,2 @@ | ||
| Add functions :c:func:`PySys_GetAttr`, :c:func:`PySys_GetAttrString`, | ||
| :c:func:`PySys_GetOptionalAttr` and :c:func:`PySys_GetOptionalAttrString`. | 
Uh oh!
There was an error while loading. Please reload this page.