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.

classification
Title: Do not always use exceptions to return result from coroutine
Type: performance Stage: resolved
Components: Versions: Python 3.10
process
Status: closed Resolution: fixed
Dependencies: Superseder:
Assigned To: Nosy List: Mark.Shannon, asvetlov, lukasz.langa, petr.viktorin, scoder, serhiy.storchaka, v2m, yselivanov
Priority: normal Keywords: patch

Created on 2020-09-11 01:10 by v2m, last changed 2022-04-11 14:59 by admin. This issue is now closed.

Pull Requests
URL Status Linked Edit
PR 22196 merged v2m, 2020-09-11 01:11
PR 22330 merged serhiy.storchaka, 2020-09-20 11:19
PR 22443 merged v2m, 2020-09-28 21:39
PR 22663 merged v2m, 2020-10-12 04:54
PR 22677 merged v2m, 2020-10-12 23:22
Messages (49)
msg376698 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-11 01:10
Currently async functions are more expensive to use comparing to their sync counterparts. A simple microbenchmark shows that difference could be quite significant:
```
import time

def f(a):
    if a == 0:
        return 0
    return f(a - 1)

async def g(a):
    if a == 0:
        return 0
    return await g(a - 1)

N = 100000
C = 200

t0 = time.time()
for _ in range(N):
    f(C)
t1 = time.time()
for _ in range(N):
    try:
        g(C).send(None)
    except StopIteration:
        pass
t2 = time.time()

print(f"Sync functions: {t1 - t0} s")
print(f"Coroutines: {t2 - t1} s")
```
Results from master on my machine:

Sync functions: 2.8642687797546387 s
Coroutines: 9.172159910202026 s

NOTE: Due to viral nature of async functions their number in codebase could become quite significant so having hundreds of them in a single call stack is not something uncommon.

One of reasons of such performance gap is that async functions always return its results via raising StopIteration exception which is not cheap. This can be avoided if in addition to `_PyGen_Send` always return result via exception we could have another function that will allow us to distinguish whether value that was returned from generator is a final result (return case) of whether this is yielded value.
In linked PR I've added function `_PyGen_SendNoStopIteration` with this behavior and updated ceval.c and _asynciomodule.c to use it instead of `_PyGen_Send` which resulted in a measurable difference:

Sync functions: 2.8861589431762695 s
Coroutines: 5.730362176895142 s
msg376701 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-11 02:07
Big +1 from me. This is something I always wanted to do myself (since the time of PEP 492 & 525 implementations) and I think this is a necessary change. It's great that this isn't just a C API UX improvement but also yields a big perf improvement.
msg376703 - (view) Author: Stefan Behnel (scoder) * (Python committer) Date: 2020-09-11 08:09
Big +1 from me, too, for the same reasons Yury gave.
msg376721 - (view) Author: Stefan Behnel (scoder) * (Python committer) Date: 2020-09-11 10:57
Copying some of the design discussion from the PR here (https://github.com/python/cpython/pull/22196/files#r486730457), because it belongs into the ticket.

Yury Selivanov proposed to add a new C-API function for this (naming changes by me):

    typedef enum {PYGEN_RETURN, PYGEN_ERROR, PYGEN_NEXT} PyGenSendStatus;

    PyGenSendStatus PyGen_Send(PyGenObject *gen, PyObject *arg, PyObject **result);

Mark Shannon and I agreed that the status code should be the return value, with some confusion whether "PyGen_" or "PyCoro_" would be appropriate prefixes.

Mark Shannon wrote: I don't think [the C-API function] should be public, as a possible further improvement is to stop passing exceptions through a side channel, but in result. Maybe we don't want to do that, but lets' not add to the (already rather large) C-API.

However, I think this will be demanded and used by extensions, including Cython implemented ones, so it seems better to make them use a public function than a private one.

Let's continue these lines of discussion here.
msg376735 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-11 17:19
If I understand proposed shape of API correctly - it was not supposed to return exception via "result" so contract for new `PyGen_Send` function is something like:

Return value | result   | Comment
-----------------------------------------------------
PYGEN_RETURN | not NULL | Returned value
PYGEN_NEXT   | not NULL | Yielded value
PYGEN_ERROR  | NULL     | Regular PyErr_* functions should be used to work with error case
msg376737 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-11 18:31
@Mark
> Mark Shannon wrote: I don't think [the C-API function] should be public, as a possible further improvement is to stop passing exceptions through a side channel, but in result. Maybe we don't want to do that, but lets' not add to the (already rather large) C-API.

Yeah, we can add it as a "private" function, I'm not entirely opposed to that. But... it would be great if Cython and C code could still depend on it and use it. And then... why should it be private? The corresponding Python API "gen.send()" and "gen.throw()" is public, why can't the C API be public too?

We will not fundamentally change generators (it would be a major backwards incompatible change), so committing to a good C API sounds reasonable.

@Mark
> Remember that PyIter_Next() is pretty much the same, though, and it has the standard "return PyObject*" interface. These two would diverge then.

Maybe we should call it `_PyIter_Send()`?  While `.send()` is mostly about coroutines, regular generators have the method too, and it would be weird to call `_PyCoro_Send` on a generator object.

@Vladimir
> PYGEN_ERROR  | NULL     | Regular PyErr_* functions should be used to work with error case

Correct.
msg377001 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-16 16:09
Mark, Stefan,

I don't want this to be stale so I propose to move with my suggestions:

1. We make the new API public. Mark, if you have objections to that - please elaborate with some details. IMO, the corresponding Python API is long public and there's no harm in exposing a C version of it. Especially given the fact that uvloop, cython, and even asyncio itself will be relying on that API.

2. I propose to name the new API `PyIter_Send`. Motivation: it will work with both generators and coroutines and plays nicely with `PyIter_Next`.
msg377002 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-16 17:01
As for returned value, I propose to return -1 in case of error, 1 for yielded value and 0 for returned value (i.e. define PYGEN_RETURN = 0, PYGEN_YIELD = 1 and PYGEN_ERROR = -1, but without exposing public names).

It would be uniform with other C API: many functions return -1 on error (if they return int and can fail), and PyDict_Next() and _PySet_NextEntry() return 1 for every yielded item, and 0 if the iteration has been finished.
msg377003 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-16 17:05
> As for returned value, I propose to return -1 in case of error, 1 for yielded value and 0 for returned value (i.e. define PYGEN_RETURN = 0, PYGEN_YIELD = 1 and PYGEN_ERROR = -1, but without exposing public names).

Sure, that works.
msg377006 - (view) Author: Stefan Behnel (scoder) * (Python committer) Date: 2020-09-16 18:15
I'm happy to see this moving forward.

Not convinved of the "PyIter_Send()" name, though. I don't consider this part of the iterator protocol. It's specific to generators and coroutines. Cython would probably guard its usage by "PyGen_CheckExact()" or "PyCoro_CheckExact()", and not use it for arbitrary iterators.

Since coroutines inherit the generator protocol more or less, I think "PyGen_Send()" is a more suitable name, better than "PyCoro_Send()".
msg377007 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-16 18:22
Also should it be specific to generators/coroutines and accept PyGenObject* or should it try to handle multiple cases and expose the result for them in uniform way, i.e.
```
if (PyGen_CheckExact(gen) || PyCoro_CheckExact(gen)) {
   // use coroutine/generator specific code that avoids raising exceptions
   *result = ...
   return PYGEN_RETURN;
}
PyObject *ret;
if (arg == Py_None) {
  ret = Py_TYPE(gen)->tp_iternext(gen);
}
else {
  ret = _PyObject_CallMethodIdOneArg(coro, &PyId_send, arg);
}
if (ret != NULL) {
  *result = ret;
  return PYGEN_YIELD;
}
if (_PyGen_FetchStopIterationValue(result) == 0) {
  return PYGEN_RETURN;
}
return PYGEN_ERROR;
```
msg377008 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-16 18:27
> Since coroutines inherit the generator protocol more or less, I think "PyGen_Send()" is a more suitable name, better than "PyCoro_Send()".

OK, +1. PyGen_Send it is.

> Also should it be specific to generators/coroutines and accept PyGenObject*

I think it should be specific to generators and coroutines. Calling `PyObject_CallMethodIdOneArg(coro, &PyId_send, arg);` and interpreting exceptions to emulate the low level API seems a bit too much.
msg377023 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-16 20:44
> I think it should be specific to generators and coroutines. Calling `PyObject_CallMethodIdOneArg(coro, &PyId_send, arg);` and interpreting exceptions to emulate the low level API seems a bit too much.

To add to my point: typically higher-level APIs go under the `PyObject_*` namespace, whereas `Py{Type}_*` is more concrete. So I'd make `PyGen_Send` to only work with `PyGen` and `PyCoro`.
msg377025 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-16 21:14
There are other abstract object APIs: PyNumber, PySequence, PyMapping, etc. In particularly PyIter_Next() works with the iterator protocol, there is no single iterator class. Seems PyGen_* API is related to concrete class, but we can introduce new namespace for the generator protocol.
msg377051 - (view) Author: Mark Shannon (Mark.Shannon) * (Python committer) Date: 2020-09-17 10:03
I agree with Serhiy, that `PyGen_` is a bad prefix.

Unless a function takes generator objects and *only* generators objects, then it shouldn't have a `PyGen` prefix.

The API function is the C equivalent of obj.send(val).
The first parameter is an object.

Coroutines do not inherit from generators.
That the the C implementations are so coupled is an unfortunate historical accident, and may well be changed.

Regardless of how this is implemented internally, any API function's signature should reflect the equivalent Python code.
And please use an enum, not an int, as the return value. It's a huge boon for readability.

I would suggest:
`PySendResult PyIter_Send(PyObject *obj, PyObject *arg, PyObject **result);`
msg377063 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-17 19:17
I guess `PyIter_Send` would imply that this function should work for all inputs (like in https://bugs.python.org/msg377007) which also sounds reasonable.
msg377066 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-17 21:11
> I guess `PyIter_Send` would imply that this function should work for all inputs (like in https://bugs.python.org/msg377007) which also sounds reasonable.

I like `PyIter_Send` (and why I initially proposed it myself) because it will also work with the "send" slot defined on some types if we end up adding one.
msg377067 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-17 21:21
so to summarize:

Proposed function signature:
```
PySendResult PyIter_Send(PyObject *obj, PyObject *arg, PyObject **result);
```

For generators/coroutines function will delegate to specialized implementation that does not raise StopIteration exception
For types that provide `tp_iternext` if arg is Py_None function call invoke `Py_TYPE(obj)->tp_iternext(obj)`
For all other cases function will try to call `send` method

Regarding of the case function will not raise StopIteration and will always return pair status/result.

Does it sound correct?
msg377077 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-18 00:09
> Does it sound correct?

It does, but given the amount of back and forth on this, I'd wait for Serhiy and Stefan to confirm if they're OK.

IMO the `PyIter_Send` name is OK (given how generic the implementation will be) and returning an enum, while a tad unconventional for the C API is way more convenient for the developer.
msg377092 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-18 07:36
I would be fine even with a generator-specific API as a first step, for simplicity. But the end goal is to support all generator-like objects. It is much more useful for end users and does not have drawbacks.

Enum for result seems not necessary (other functions with three-state result just return -1, 0 or 1), but if other core developers want it -- OK.
msg377097 - (view) Author: Stefan Behnel (scoder) * (Python committer) Date: 2020-09-18 08:58
I would also have preferred a more type specific function, but yeah, as long as the types for which the function would normally be used are special cased early enough in the implementation, it makes no big difference.

Fine with me, too.
msg377098 - (view) Author: Stefan Behnel (scoder) * (Python committer) Date: 2020-09-18 09:11
BTW, just to give this a house number, I remember having measured a performance improvement of up to 70% at some point when switching from "generators always raise a StopIteration at the end" to "generators just return NULL" in Cython. For short-running generators and coroutines, this can make a big difference.
msg377109 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-18 11:40
We introduced _PyObject_LookupAttr() and _PyObject_GetMethod() for similar purposes. And they have similar signatures. Although they all are private.
msg377128 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-18 16:36
> I would also have preferred a more type specific function, but yeah, as long as the types for which the function would normally be used are special cased early enough in the implementation, it makes no big difference.

Maybe add two API funcs: PyGen_Send (specific to generators & coroutines) and PyIter_Send?
msg377132 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-18 17:55
Sounds like a good middleground to start: add ``PySendResult `` and `PySendResult PyGen_Send(PyGenObject*, PyObject* PyObject**)` specific to generators and coroutines. Subsequent changes could introduce `PySendResult PyIter_Send(PyObject*, PyObject*, PyObject**)` that would be more generic (call tp_next, invoke "send" or maybe in the future use dedicated slot for "send" operator so i.e. asyncio.Future or Cython coroutines could benefit from the same optimization)
msg377136 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-18 20:35
> Sounds like a good middleground to start: add ``PySendResult `` and `PySendResult PyGen_Send(PyGenObject*, PyObject* PyObject**)` specific to generators and coroutines. 

Yes, it seems that everybody agreed on that. I can give the PR another review -- is it ready?
msg377137 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-18 20:38
Yes, it should be
msg377146 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-19 01:38
New changeset 2b05361bf7cbbd76035206fd9befe87f37489f1e by Vladimir Matveev in branch 'master':
bpo-41756: Introduce PyGen_Send C API (GH-22196)
https://github.com/python/cpython/commit/2b05361bf7cbbd76035206fd9befe87f37489f1e
msg377147 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-19 01:40
Thanks, Vladimir! Also huge thanks to Mark, Serhiy, and Stefan. If there are any nits to fix let's do that in follow up PRs.
msg377215 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-20 11:21
PR 22330 refactors gen_send_ex(), making it similar to the new PyGen_Send().
msg377219 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-20 16:03
Vladimir, could you please submit a PR to update 3.10/whatsnew? Need to mention both the new C API and the new perf boost in relevant sections.
msg377249 - (view) Author: Mark Shannon (Mark.Shannon) * (Python committer) Date: 2020-09-21 10:50
Yury,

Why was the PR merged with a new API function `PyGen_Send`?

I explicitly said that any new API function should *not* start with `PyGen`, nor should any function rely on generators and async "coroutines" sharing the same memory layout.

If you disagree with me, please say why, don't just merge the PR.

The name `PyGen` is misleading as it can handle coroutines as well as generators.
There is no performance advantage to only handling these two types.
Worse, it requires that a `PyCoroObject` can always be cast to a `PyGenObject`, preventing better layout of either object in the future.


Would you revert the PR, please.
msg377266 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-09-21 18:36
> If you disagree with me, please say why, don't just merge the PR.

Apologies, Mark. I didn't intend to merge something bypassing your opinion; just missed your comment between reviewing multiple PRs in a few unrelated repos. I'm sorry.

On the actual naming subject, you proposed:

> `PySendResult PyIter_Send(PyObject *obj, PyObject *arg, PyObject **result);`

The problem with using this name is that ideally we should also support non-native coroutine and generator implementations (i.e. resolve the "send" attribute and call it using Python calling convention). Ideally we should have two C APIs: one low-level supporting only native objects and a high level one, supporting all kinds of them.

Can we perhaps add both `PyGen_Send()` and `PyCoro_Send()` for now that would only accept generators and coroutines respectively? After that we can discuss adding a more generic `PyIter_Send`?


> Would you revert the PR, please.

Since this is in 3.10/master that nobody uses right now except us (Python core devs), can we just issue a follow up PR to fix whatever is there to fix? I'd like to avoid the churn of reverting, and again, I apologize for pushing this a bit hastily.  Let me know if you actually want a revert and I'll do that.
msg377298 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-22 05:09
New changeset 6c33385e3a1dce31d7b9037eebfc584075795dba by Serhiy Storchaka in branch 'master':
bpo-41756: Refactor gen_send_ex(). (GH-22330)
https://github.com/python/cpython/commit/6c33385e3a1dce31d7b9037eebfc584075795dba
msg377299 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-22 05:31
I agree that we should add PyIter_Send() (the name is discussable, but PyIter_Send LGTM) which should support iterators and objects with the send() method. It would be much more useful for user, and can replace PyGen_Send() in the interpreter core code. But merging the PR with PyGen_Send() was not a mistake. It was good to split the changes on smaller steps easier to review.

Vladimir, do you mind to create a new PR for PyIter_Send()? I seen its implementation in one of intermediate versions of your PR.
msg377582 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-27 23:02
Serhiy, AFAIR PyIter_Send in my PR appear only as a rename from placeholder `Name_TBD` and it still was specific to PyGenObjects. Do you mean something that was listed in https://bugs.python.org/msg377007 ?
msg377587 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-09-28 07:38
No, I meant a function which combines PyGen_Send, tp_iternext and _PyGen_FetchStopIterationValue. Was not it in your PR?
msg377608 - (view) Author: Vladimir Matveev (v2m) * Date: 2020-09-28 17:38
No, I don't think so but I can definitely make one. A few questions first:
- having PySendResult as a result type of PyIterSend seems ok, however prefix for each concrete value (PYGEN_*) is not aligned with the prefix of the function itself (PyIter_)
- should it also deal with tstate->c_tracefunc (probably not) or just be something like
```
PySendResult
PyIter_Send(PyObject *iter, PyObject *arg, PyObject **result)
{
    _Py_IDENTIFIER(send);
    assert(result != NULL);

    if (PyGen_CheckExact(iter) || PyCoro_CheckExact(iter)) {
        return PyGen_Send((PyGenObject *)iter, arg, result);
    }

    if (arg == Py_None && Py_TYPE(iter)->tp_iternext != NULL) {
        *result = Py_TYPE(iter)->tp_iternext(iter);
    }
    else {
        *result = _PyObject_CallMethodIdOneArg(iter, &PyId_send, arg);
    }

    if (*result == NULL) {
        if (_PyGen_FetchStopIterationValue(result) == 0) {
            return PYGEN_RETURN;
        }
        return PYGEN_ERROR;
    }
    return PYGEN_NEXT;
}
```
msg378357 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-10-10 00:15
New changeset 037245c5ac46c3436f617a1f5d965929754be239 by Vladimir Matveev in branch 'master':
bpo-41756: Add PyIter_Send function (#22443)
https://github.com/python/cpython/commit/037245c5ac46c3436f617a1f5d965929754be239
msg378379 - (view) Author: Mark Shannon (Mark.Shannon) * (Python committer) Date: 2020-10-10 10:37
Vladimir,
Thanks for adding PyIter_Send().

Don't forget to remove PyGen_Send() :)
msg378425 - (view) Author: Stefan Behnel (scoder) * (Python committer) Date: 2020-10-11 08:47
> Don't forget to remove PyGen_Send()

That's the function that allows sending data into a generator. It's also used internally by "PyIter_Send()". Are you really suggesting to remove it, or to make it underscore-private again? (As it was before.)
msg378432 - (view) Author: Mark Shannon (Mark.Shannon) * (Python committer) Date: 2020-10-11 12:36
What's the difference between removing it and making it properly private (i.e. static)?

It's an irrelevant detail whether the code is inlined or in a helper function.
msg378515 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-10-12 19:11
With the latest PR now merged this issue can be closed. Please reopen if there are any other action items left.
msg378518 - (view) Author: Serhiy Storchaka (serhiy.storchaka) * (Python committer) Date: 2020-10-12 19:22
Few things I forget about. The new C API function should be exported in PC/python3dll.c.

Also, in Include/abstract.h it should only be available for limited C API >= 3.10:

#if !defined(Py_LIMITED_API) || Py_LIMITED_API+0 >= 0x030A0000
...
#endif

We now should discuss the behavior of PySend_Iter() if value is NULL. It is not documented, the current behavior differs from the behavior for non-NULL value, and it is used in the ceval loop. We should document this case explicitly and maybe change the behavior if it would be more appropriate.
msg378527 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-10-12 21:28
> Also, in Include/abstract.h it should only be available for limited C API >= 3.10:

Vladimir, could you please submit a PR?

> We now should discuss the behavior of PySend_Iter() if value is NULL. It is not documented, the current behavior differs from the behavior for non-NULL value, and it is used in the ceval loop.

IMO: I'd keep the behavior and just document it.
msg378570 - (view) Author: Yury Selivanov (yselivanov) * (Python committer) Date: 2020-10-13 17:27
New changeset cfb0f57ff876ab3d04ff144f19eda58844981643 by Vladimir Matveev in branch 'master':
bpo-41756: Export PyGen_Send and wrap it in if-defs (#22677)
https://github.com/python/cpython/commit/cfb0f57ff876ab3d04ff144f19eda58844981643
msg382003 - (view) Author: Andrew Svetlov (asvetlov) * (Python committer) Date: 2020-11-28 14:42
Can we close the issue?
msg398074 - (view) Author: Petr Viktorin (petr.viktorin) * (Python committer) Date: 2021-07-23 15:56
Hello!
This change added an enum to the stable ABI:
    typedef enum {
        PYGEN_RETURN = 0,
        PYGEN_ERROR = -1,
        PYGEN_NEXT = 1,
    } PySendResult;

Adding new values to enums might break the stable ABI in some (admittedly rare) cases; so usually it's better to avoid enums and stick with int and #ifdef'd values.
This particular one looks like it won't be extended and won't cause problems, but still, switching to int would make stable ABI easier to work with.
Is anyone against switching to int?

See pbo-44727 for details.
msg398075 - (view) Author: Petr Viktorin (petr.viktorin) * (Python committer) Date: 2021-07-23 16:04
that is, bpo-44727
History
Date User Action Args
2022-04-11 14:59:35adminsetgithub: 85922
2021-07-23 16:04:57petr.viktorinsetmessages: + msg398075
2021-07-23 15:56:58petr.viktorinsetnosy: + petr.viktorin
messages: + msg398074
2021-05-05 10:13:43asvetlovsetstatus: open -> closed
stage: patch review -> resolved
2020-11-28 14:42:34asvetlovsetnosy: + asvetlov
messages: + msg382003
2020-10-13 17:27:00yselivanovsetmessages: + msg378570
2020-10-12 23:22:24v2msetstage: resolved -> patch review
pull_requests: + pull_request21649
2020-10-12 21:28:06yselivanovsetmessages: + msg378527
2020-10-12 21:21:44yselivanovsetstatus: closed -> open
2020-10-12 19:22:29serhiy.storchakasetmessages: + msg378518
2020-10-12 19:11:55yselivanovsetstatus: open -> closed

messages: + msg378515
stage: patch review -> resolved
2020-10-12 04:54:32v2msetpull_requests: + pull_request21639
2020-10-11 12:36:50Mark.Shannonsetmessages: + msg378432
2020-10-11 08:47:17scodersetmessages: + msg378425
2020-10-10 10:37:47Mark.Shannonsetmessages: + msg378379
2020-10-10 00:15:23yselivanovsetmessages: + msg378357
2020-09-28 21:39:08v2msetstage: resolved -> patch review
pull_requests: + pull_request21473
2020-09-28 17:38:21v2msetmessages: + msg377608
2020-09-28 07:38:03serhiy.storchakasetmessages: + msg377587
2020-09-27 23:02:39v2msetmessages: + msg377582
2020-09-22 05:31:09serhiy.storchakasetmessages: + msg377299
2020-09-22 05:09:04serhiy.storchakasetmessages: + msg377298
2020-09-21 18:36:08yselivanovsetmessages: + msg377266
2020-09-21 10:50:10Mark.Shannonsetmessages: + msg377249
2020-09-20 16:03:50yselivanovsetmessages: + msg377219
2020-09-20 11:21:43serhiy.storchakasetstatus: closed -> open

messages: + msg377215
2020-09-20 11:19:19serhiy.storchakasetpull_requests: + pull_request21374
2020-09-19 01:40:34yselivanovsetstatus: open -> closed
resolution: fixed
messages: + msg377147

stage: patch review -> resolved
2020-09-19 01:38:46yselivanovsetmessages: + msg377146
2020-09-18 20:38:41v2msetmessages: + msg377137
2020-09-18 20:35:05yselivanovsetmessages: + msg377136
2020-09-18 17:55:09v2msetmessages: + msg377132
2020-09-18 16:36:49yselivanovsetmessages: + msg377128
2020-09-18 11:40:54serhiy.storchakasetmessages: + msg377109
2020-09-18 09:11:10scodersetmessages: + msg377098
2020-09-18 08:58:18scodersetmessages: + msg377097
2020-09-18 07:36:31serhiy.storchakasetmessages: + msg377092
2020-09-18 00:09:08yselivanovsetmessages: + msg377077
2020-09-17 21:21:23v2msetmessages: + msg377067
2020-09-17 21:11:47yselivanovsetmessages: + msg377066
2020-09-17 19:17:57v2msetmessages: + msg377063
2020-09-17 10:03:19Mark.Shannonsetmessages: + msg377051
2020-09-16 21:14:53serhiy.storchakasetmessages: + msg377025
2020-09-16 20:44:00yselivanovsetmessages: + msg377023
2020-09-16 18:27:41yselivanovsetmessages: + msg377008
2020-09-16 18:22:32v2msetmessages: + msg377007
2020-09-16 18:15:38scodersetmessages: + msg377006
2020-09-16 17:05:40yselivanovsetmessages: + msg377003
2020-09-16 17:01:40serhiy.storchakasetmessages: + msg377002
2020-09-16 16:09:28yselivanovsetmessages: + msg377001
2020-09-12 07:00:58vstinnersetnosy: - vstinner
2020-09-11 18:31:13yselivanovsetmessages: + msg376737
2020-09-11 17:19:55v2msetmessages: + msg376735
2020-09-11 10:57:11scodersetmessages: + msg376721
2020-09-11 08:26:43serhiy.storchakasetnosy: + serhiy.storchaka
2020-09-11 08:09:02scodersetnosy: + scoder
messages: + msg376703
2020-09-11 02:07:38yselivanovsetnosy: + vstinner, lukasz.langa, Mark.Shannon
messages: + msg376701
2020-09-11 01:11:41v2msetkeywords: + patch
stage: patch review
pull_requests: + pull_request21255
2020-09-11 01:10:36v2mcreate