This issue tracker has been migrated to GitHub ,
and is currently read-only.
For more information,
see the GitHub FAQs in the Python's Developer Guide.
Created on 2012年03月19日 17:51 by Yury.Selivanov, last changed 2022年04月11日 14:57 by admin.
| Files | ||||
|---|---|---|---|---|
| File name | Uploaded | Description | Edit | |
| writable_closure.patch | Yury.Selivanov, 2012年03月19日 19:34 | review | ||
| writable_closure_02.patch | Yury.Selivanov, 2012年03月19日 19:50 | review | ||
| writable_closure_03.patch | Yury.Selivanov, 2012年03月20日 14:35 | review | ||
| writable_closure_04.patch | Yury.Selivanov, 2012年03月20日 20:03 | review | ||
| writable_closure_with_checking.patch | sbt, 2012年04月25日 14:33 | review | ||
| Messages (14) | |||
|---|---|---|---|
| msg156350 - (view) | Author: Yury Selivanov (Yury.Selivanov) * | Date: 2012年03月19日 17:51 | |
__code__ and __closure__ are designed to work together. There is no point in allowing to completely substitute the __code__ object, but protecting the __closure__. |
|||
| msg156357 - (view) | Author: Yury Selivanov (Yury.Selivanov) * | Date: 2012年03月19日 19:50 | |
Updated patch as per Andrew's code review. Thank you. |
|||
| msg156404 - (view) | Author: Andrew Svetlov (asvetlov) * (Python committer) | Date: 2012年03月20日 13:37 | |
Please update the doc also. I think changing from 'Read-only' to 'Writable' in Doc/reference/datamodel.rst is enough. |
|||
| msg156412 - (view) | Author: Yury Selivanov (Yury.Selivanov) * | Date: 2012年03月20日 14:35 | |
> Please update the doc also. I think changing from 'Read-only' to 'Writable' in Doc/reference/datamodel.rst is enough. Updated in writable_closure_03.patch. Thanks. |
|||
| msg156452 - (view) | Author: Yury Selivanov (Yury.Selivanov) * | Date: 2012年03月20日 20:03 | |
Updated patch per Benjamin's review. See writable_closure_04.patch. |
|||
| msg159117 - (view) | Author: Alyssa Coghlan (ncoghlan) * (Python committer) | Date: 2012年04月24日 04:54 | |
Another use case for a writeable __closure__ attribute is to make it possible to manually break reference cycles: http://blog.ccpgames.com/kristjan/2012/04/23/reference-cycles-with-closures/ |
|||
| msg159182 - (view) | Author: Richard Oudkerk (sbt) * (Python committer) | Date: 2012年04月24日 18:19 | |
Shouldn't test___closure__() also test what happens when the closure is replaced with None, or a tuple which is too long or too short or contains non-cell objects? All of these things seem to be checked when you create a new function using types.FunctionType: >>> h = types.FunctionType(g.__code__, g.__globals__, "h", g.__defaults__, None) Traceback (most recent call last): File "<stdin>", line 1, in <module> TypeError: arg 5 (closure) must be tuple >>> h = types.FunctionType(g.__code__, g.__globals__, "h", g.__defaults__, ()) Traceback (most recent call last): File "<stdin>", line 1, in <module> ValueError: g requires closure of length 2, not 0 >>> h = types.FunctionType(g.__code__, g.__globals__, "h", g.__defaults__, (1,2)) Traceback (most recent call last): File "<stdin>", line 1, in <module> TypeError: arg 5 (closure) expected cell, found int I think the setter should make similar checks. Maybe there is C code which assumes "broken" closures never happen. |
|||
| msg159235 - (view) | Author: Richard Oudkerk (sbt) * (Python committer) | Date: 2012年04月24日 23:37 | |
The patch causes crashes. If I define def cell(o): def f(): o return f.__closure__[0] def f(): a = 1 b = 2 def g(): return a + b return g g = f() then I find g.__closure__ = None; g() -> crash g.__closure__ = (cell(3),); g() -> crash g.__closure__ = (1, 2); g() -> SystemError * g.__closure__ = (cell(3), cell(4), cell(5)); g() -> returns 7 * SystemError: ..\Objects\cellobject.c:24: bad argument to internal function |
|||
| msg159238 - (view) | Author: Yury Selivanov (Yury.Selivanov) * | Date: 2012年04月24日 23:52 | |
> The patch causes crashes. Yes, that's known. First, we need to check, that we can only write tuple of cell objects or None in __closure__ (that's easy to add). Secondly, perhaps, we can check __closure__ correctness each time we start evaluating a code object. The latter would offer us better stability, but may also introduce some slowdowns -- need to find some time to implement and benchmark this. |
|||
| msg159287 - (view) | Author: Richard Oudkerk (sbt) * (Python committer) | Date: 2012年04月25日 14:33 | |
Version of patch which checks invariants in the setter and adds tests. |
|||
| msg159519 - (view) | Author: Andrew Svetlov (asvetlov) * (Python committer) | Date: 2012年04月28日 10:56 | |
sbt, looks good for me. |
|||
| msg304056 - (view) | Author: Devin Bayer (akvadrako) * | Date: 2017年10月10日 15:50 | |
Any updates on this? I'm trying to implement hot module reloading using code from IPython, which tries to modify __closure__ in place. It fails silently and doesn't work, but indeed would be nice to have. |
|||
| msg304106 - (view) | Author: Alyssa Coghlan (ncoghlan) * (Python committer) | Date: 2017年10月11日 05:33 | |
This still seems like a reasonable enhancement, but would presumably need updates to apply against the 3.7 development branch. |
|||
| msg304199 - (view) | Author: Alyssa Coghlan (ncoghlan) * (Python committer) | Date: 2017年10月12日 02:46 | |
Thinking about the interaction between this idea and https://bugs.python.org/issue30744 made me realise that there's a subtlety here that would probably need to be spelled out more clearly in the docs for __closure__ than it is for __code__: any changes made to a function object (whether it's a synchronous function, a generator, or a coroutine) will only affect *future* function invocations, as execution frames capture references to both the code object and all the closure cells when they're created. (Thought prompted by asking myself "What would happen to existing generator-iterators if you rebound the closure on the generator function?". The answer is "Nothing", but I figure if I had to think about it, that answer likely isn't going to be obvious to folks that are less familiar with how the eval loop works in practice) |
|||
| History | |||
|---|---|---|---|
| Date | User | Action | Args |
| 2022年04月11日 14:57:28 | admin | set | github: 58577 |
| 2022年03月28日 16:35:30 | vstinner | set | title: make __closure__ writable -> make function __closure__ writable |
| 2017年10月12日 02:46:41 | ncoghlan | set | messages: + msg304199 |
| 2017年10月11日 06:26:40 | rhettinger | set | nosy:
+ rhettinger |
| 2017年10月11日 05:33:11 | ncoghlan | set | messages:
+ msg304106 versions: + Python 3.7, - Python 3.3 |
| 2017年10月10日 15:50:04 | akvadrako | set | messages: + msg304056 |
| 2017年10月10日 15:48:16 | akvadrako | set | nosy:
+ akvadrako |
| 2012年12月29日 03:40:25 | meador.inge | set | nosy:
+ meador.inge |
| 2012年11月13日 03:42:01 | eric.snow | set | nosy:
+ eric.snow |
| 2012年04月28日 10:56:50 | asvetlov | set | messages: + msg159519 |
| 2012年04月25日 14:33:37 | sbt | set | files:
+ writable_closure_with_checking.patch messages: + msg159287 |
| 2012年04月24日 23:52:18 | Yury.Selivanov | set | messages: + msg159238 |
| 2012年04月24日 23:37:56 | sbt | set | messages: + msg159235 |
| 2012年04月24日 18:19:50 | sbt | set | nosy:
+ sbt messages: + msg159182 |
| 2012年04月24日 04:54:27 | ncoghlan | set | messages: + msg159117 |
| 2012年03月21日 02:05:53 | Yury.Selivanov | set | nosy:
+ ncoghlan |
| 2012年03月20日 20:03:04 | Yury.Selivanov | set | files:
+ writable_closure_04.patch messages: + msg156452 |
| 2012年03月20日 19:59:29 | asvetlov | set | nosy:
+ benjamin.peterson |
| 2012年03月20日 19:58:34 | asvetlov | set | stage: patch review |
| 2012年03月20日 14:35:27 | Yury.Selivanov | set | files:
+ writable_closure_03.patch messages: + msg156412 |
| 2012年03月20日 13:37:27 | asvetlov | set | messages: + msg156404 |
| 2012年03月19日 19:50:58 | Yury.Selivanov | set | files:
+ writable_closure_02.patch messages: + msg156357 |
| 2012年03月19日 19:34:06 | Yury.Selivanov | set | files: + writable_closure.patch |
| 2012年03月19日 19:33:53 | Yury.Selivanov | set | files: - writable_closure.patch |
| 2012年03月19日 17:51:16 | Yury.Selivanov | create | |