| #include "Python.h" |
| #include "structmember.h" |
| |
| |
| /*[clinic input] |
| module _asyncio |
| [clinic start generated code]*/ |
| /*[clinic end generated code: output=da39a3ee5e6b4b0d input=8fd17862aa989c69]*/ |
| |
| |
| /* identifiers used from some functions */ |
| _Py_IDENTIFIER(add_done_callback); |
| _Py_IDENTIFIER(call_soon); |
| _Py_IDENTIFIER(cancel); |
| _Py_IDENTIFIER(send); |
| _Py_IDENTIFIER(throw); |
| _Py_IDENTIFIER(_step); |
| _Py_IDENTIFIER(_schedule_callbacks); |
| _Py_IDENTIFIER(_wakeup); |
| |
| |
| /* State of the _asyncio module */ |
| static PyObject *all_tasks; |
| static PyObject *current_tasks; |
| static PyObject *traceback_extract_stack; |
| static PyObject *asyncio_get_event_loop; |
| static PyObject *asyncio_future_repr_info_func; |
| static PyObject *asyncio_task_repr_info_func; |
| static PyObject *asyncio_task_get_stack_func; |
| static PyObject *asyncio_task_print_stack_func; |
| static PyObject *asyncio_InvalidStateError; |
| static PyObject *asyncio_CancelledError; |
| static PyObject *inspect_isgenerator; |
| |
| |
| typedef enum { |
| STATE_PENDING, |
| STATE_CANCELLED, |
| STATE_FINISHED |
| } fut_state; |
| |
| #define FutureObj_HEAD(prefix) \ |
| PyObject_HEAD \ |
| PyObject *prefix##_loop; \ |
| PyObject *prefix##_callbacks; \ |
| PyObject *prefix##_exception; \ |
| PyObject *prefix##_result; \ |
| PyObject *prefix##_source_tb; \ |
| fut_state prefix##_state; \ |
| int prefix##_log_tb; \ |
| int prefix##_blocking; \ |
| PyObject *dict; \ |
| PyObject *prefix##_weakreflist; |
| |
| typedef struct { |
| FutureObj_HEAD(fut) |
| } FutureObj; |
| |
| typedef struct { |
| FutureObj_HEAD(task) |
| PyObject *task_fut_waiter; |
| PyObject *task_coro; |
| int task_must_cancel; |
| int task_log_destroy_pending; |
| } TaskObj; |
| |
| typedef struct { |
| PyObject_HEAD |
| TaskObj *sw_task; |
| PyObject *sw_arg; |
| } TaskSendMethWrapper; |
| |
| typedef struct { |
| PyObject_HEAD |
| TaskObj *ww_task; |
| } TaskWakeupMethWrapper; |
| |
| |
| #include "clinic/_asynciomodule.c.h" |
| |
| |
| /*[clinic input] |
| class _asyncio.Future "FutureObj *" "&Future_Type" |
| [clinic start generated code]*/ |
| /*[clinic end generated code: output=da39a3ee5e6b4b0d input=00d3e4abca711e0f]*/ |
| |
| /* Get FutureIter from Future */ |
| static PyObject* future_new_iter(PyObject *); |
| static inline int future_call_schedule_callbacks(FutureObj *); |
| |
| static int |
| future_schedule_callbacks(FutureObj *fut) |
| { |
| Py_ssize_t len; |
| PyObject* iters; |
| int i; |
| |
| if (fut->fut_callbacks == NULL) { |
| PyErr_SetString(PyExc_RuntimeError, "NULL callbacks"); |
| return -1; |
| } |
| |
| len = PyList_GET_SIZE(fut->fut_callbacks); |
| if (len == 0) { |
| return 0; |
| } |
| |
| iters = PyList_GetSlice(fut->fut_callbacks, 0, len); |
| if (iters == NULL) { |
| return -1; |
| } |
| if (PyList_SetSlice(fut->fut_callbacks, 0, len, NULL) < 0) { |
| Py_DECREF(iters); |
| return -1; |
| } |
| |
| for (i = 0; i < len; i++) { |
| PyObject *handle = NULL; |
| PyObject *cb = PyList_GET_ITEM(iters, i); |
| |
| handle = _PyObject_CallMethodId( |
| fut->fut_loop, &PyId_call_soon, "OO", cb, fut, NULL); |
| |
| if (handle == NULL) { |
| Py_DECREF(iters); |
| return -1; |
| } |
| else { |
| Py_DECREF(handle); |
| } |
| } |
| |
| Py_DECREF(iters); |
| return 0; |
| } |
| |
| static int |
| future_init(FutureObj *fut, PyObject *loop) |
| { |
| PyObject *res = NULL; |
| _Py_IDENTIFIER(get_debug); |
| |
| if (loop == NULL || loop == Py_None) { |
| loop = PyObject_CallObject(asyncio_get_event_loop, NULL); |
| if (loop == NULL) { |
| return -1; |
| } |
| } |
| else { |
| Py_INCREF(loop); |
| } |
| Py_CLEAR(fut->fut_loop); |
| fut->fut_loop = loop; |
| |
| res = _PyObject_CallMethodId(fut->fut_loop, &PyId_get_debug, NULL); |
| if (res == NULL) { |
| return -1; |
| } |
| if (PyObject_IsTrue(res)) { |
| Py_CLEAR(res); |
| fut->fut_source_tb = PyObject_CallObject(traceback_extract_stack, NULL); |
| if (fut->fut_source_tb == NULL) { |
| return -1; |
| } |
| } |
| else { |
| Py_CLEAR(res); |
| } |
| |
| fut->fut_callbacks = PyList_New(0); |
| if (fut->fut_callbacks == NULL) { |
| return -1; |
| } |
| |
| return 0; |
| } |
| |
| static PyObject * |
| future_set_result(FutureObj *fut, PyObject *res) |
| { |
| if (fut->fut_state != STATE_PENDING) { |
| PyErr_SetString(asyncio_InvalidStateError, "invalid state"); |
| return NULL; |
| } |
| |
| Py_INCREF(res); |
| fut->fut_result = res; |
| fut->fut_state = STATE_FINISHED; |
| |
| if (future_call_schedule_callbacks(fut) == -1) { |
| return NULL; |
| } |
| Py_RETURN_NONE; |
| } |
| |
| static PyObject * |
| future_set_exception(FutureObj *fut, PyObject *exc) |
| { |
| PyObject *exc_val = NULL; |
| |
| if (fut->fut_state != STATE_PENDING) { |
| PyErr_SetString(asyncio_InvalidStateError, "invalid state"); |
| return NULL; |
| } |
| |
| if (PyExceptionClass_Check(exc)) { |
| exc_val = PyObject_CallObject(exc, NULL); |
| if (exc_val == NULL) { |
| return NULL; |
| } |
| } |
| else { |
| exc_val = exc; |
| Py_INCREF(exc_val); |
| } |
| if (!PyExceptionInstance_Check(exc_val)) { |
| Py_DECREF(exc_val); |
| PyErr_SetString(PyExc_TypeError, "invalid exception object"); |
| return NULL; |
| } |
| if ((PyObject*)Py_TYPE(exc_val) == PyExc_StopIteration) { |
| Py_DECREF(exc_val); |
| PyErr_SetString(PyExc_TypeError, |
| "StopIteration interacts badly with generators " |
| "and cannot be raised into a Future"); |
| return NULL; |
| } |
| |
| fut->fut_exception = exc_val; |
| fut->fut_state = STATE_FINISHED; |
| |
| if (future_call_schedule_callbacks(fut) == -1) { |
| return NULL; |
| } |
| |
| fut->fut_log_tb = 1; |
| Py_RETURN_NONE; |
| } |
| |
| static int |
| future_get_result(FutureObj *fut, PyObject **result) |
| { |
| PyObject *exc; |
| |
| if (fut->fut_state == STATE_CANCELLED) { |
| exc = _PyObject_CallNoArg(asyncio_CancelledError); |
| if (exc == NULL) { |
| return -1; |
| } |
| *result = exc; |
| return 1; |
| } |
| |
| if (fut->fut_state != STATE_FINISHED) { |
| PyObject *msg = PyUnicode_FromString("Result is not ready."); |
| if (msg == NULL) { |
| return -1; |
| } |
| |
| exc = _PyObject_CallArg1(asyncio_InvalidStateError, msg); |
| Py_DECREF(msg); |
| if (exc == NULL) { |
| return -1; |
| } |
| |
| *result = exc; |
| return 1; |
| } |
| |
| fut->fut_log_tb = 0; |
| if (fut->fut_exception != NULL) { |
| Py_INCREF(fut->fut_exception); |
| *result = fut->fut_exception; |
| return 1; |
| } |
| |
| Py_INCREF(fut->fut_result); |
| *result = fut->fut_result; |
| return 0; |
| } |
| |
| static PyObject * |
| future_add_done_callback(FutureObj *fut, PyObject *arg) |
| { |
| if (fut->fut_state != STATE_PENDING) { |
| PyObject *handle = _PyObject_CallMethodId( |
| fut->fut_loop, &PyId_call_soon, "OO", arg, fut, NULL); |
| |
| if (handle == NULL) { |
| return NULL; |
| } |
| else { |
| Py_DECREF(handle); |
| } |
| } |
| else { |
| int err = PyList_Append(fut->fut_callbacks, arg); |
| if (err != 0) { |
| return NULL; |
| } |
| } |
| Py_RETURN_NONE; |
| } |
| |
| static PyObject * |
| future_cancel(FutureObj *fut) |
| { |
| if (fut->fut_state != STATE_PENDING) { |
| Py_RETURN_FALSE; |
| } |
| fut->fut_state = STATE_CANCELLED; |
| |
| if (future_call_schedule_callbacks(fut) == -1) { |
| return NULL; |
| } |
| |
| Py_RETURN_TRUE; |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.__init__ |
| |
| * |
| loop: 'O' = NULL |
| |
| This class is *almost* compatible with concurrent.futures.Future. |
| |
| Differences: |
| |
| - result() and exception() do not take a timeout argument and |
| raise an exception when the future isn't done yet. |
| |
| - Callbacks registered with add_done_callback() are always called |
| via the event loop's call_soon_threadsafe(). |
| |
| - This class is not compatible with the wait() and as_completed() |
| methods in the concurrent.futures package. |
| [clinic start generated code]*/ |
| |
| static int |
| _asyncio_Future___init___impl(FutureObj *self, PyObject *loop) |
| /*[clinic end generated code: output=9ed75799eaccb5d6 input=8e1681f23605be2d]*/ |
| |
| { |
| return future_init(self, loop); |
| } |
| |
| static int |
| FutureObj_clear(FutureObj *fut) |
| { |
| Py_CLEAR(fut->fut_loop); |
| Py_CLEAR(fut->fut_callbacks); |
| Py_CLEAR(fut->fut_result); |
| Py_CLEAR(fut->fut_exception); |
| Py_CLEAR(fut->fut_source_tb); |
| Py_CLEAR(fut->dict); |
| return 0; |
| } |
| |
| static int |
| FutureObj_traverse(FutureObj *fut, visitproc visit, void *arg) |
| { |
| Py_VISIT(fut->fut_loop); |
| Py_VISIT(fut->fut_callbacks); |
| Py_VISIT(fut->fut_result); |
| Py_VISIT(fut->fut_exception); |
| Py_VISIT(fut->fut_source_tb); |
| Py_VISIT(fut->dict); |
| return 0; |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.result |
| |
| Return the result this future represents. |
| |
| If the future has been cancelled, raises CancelledError. If the |
| future's result isn't yet available, raises InvalidStateError. If |
| the future is done and has an exception set, this exception is raised. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_result_impl(FutureObj *self) |
| /*[clinic end generated code: output=f35f940936a4b1e5 input=49ecf9cf5ec50dc5]*/ |
| { |
| PyObject *result; |
| int res = future_get_result(self, &result); |
| |
| if (res == -1) { |
| return NULL; |
| } |
| |
| if (res == 0) { |
| return result; |
| } |
| |
| assert(res == 1); |
| |
| PyErr_SetObject(PyExceptionInstance_Class(result), result); |
| Py_DECREF(result); |
| return NULL; |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.exception |
| |
| Return the exception that was set on this future. |
| |
| The exception (or None if no exception was set) is returned only if |
| the future is done. If the future has been cancelled, raises |
| CancelledError. If the future isn't done yet, raises |
| InvalidStateError. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_exception_impl(FutureObj *self) |
| /*[clinic end generated code: output=88b20d4f855e0710 input=733547a70c841c68]*/ |
| { |
| if (self->fut_state == STATE_CANCELLED) { |
| PyErr_SetString(asyncio_CancelledError, ""); |
| return NULL; |
| } |
| |
| if (self->fut_state != STATE_FINISHED) { |
| PyErr_SetString(asyncio_InvalidStateError, "Result is not ready."); |
| return NULL; |
| } |
| |
| if (self->fut_exception != NULL) { |
| self->fut_log_tb = 0; |
| Py_INCREF(self->fut_exception); |
| return self->fut_exception; |
| } |
| |
| Py_RETURN_NONE; |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.set_result |
| |
| res: 'O' |
| / |
| |
| Mark the future done and set its result. |
| |
| If the future is already done when this method is called, raises |
| InvalidStateError. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_set_result(FutureObj *self, PyObject *res) |
| /*[clinic end generated code: output=a620abfc2796bfb6 input=8619565e0503357e]*/ |
| { |
| return future_set_result(self, res); |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.set_exception |
| |
| exception: 'O' |
| / |
| |
| Mark the future done and set an exception. |
| |
| If the future is already done when this method is called, raises |
| InvalidStateError. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_set_exception(FutureObj *self, PyObject *exception) |
| /*[clinic end generated code: output=f1c1b0cd321be360 input=1377dbe15e6ea186]*/ |
| { |
| return future_set_exception(self, exception); |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.add_done_callback |
| |
| fn: 'O' |
| / |
| |
| Add a callback to be run when the future becomes done. |
| |
| The callback is called with a single argument - the future object. If |
| the future is already done when this is called, the callback is |
| scheduled with call_soon. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_add_done_callback(FutureObj *self, PyObject *fn) |
| /*[clinic end generated code: output=819e09629b2ec2b5 input=8cce187e32cec6a8]*/ |
| { |
| return future_add_done_callback(self, fn); |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.remove_done_callback |
| |
| fn: 'O' |
| / |
| |
| Remove all instances of a callback from the "call when done" list. |
| |
| Returns the number of callbacks removed. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_remove_done_callback(FutureObj *self, PyObject *fn) |
| /*[clinic end generated code: output=5ab1fb52b24ef31f input=3fedb73e1409c31c]*/ |
| { |
| PyObject *newlist; |
| Py_ssize_t len, i, j=0; |
| |
| len = PyList_GET_SIZE(self->fut_callbacks); |
| if (len == 0) { |
| return PyLong_FromSsize_t(0); |
| } |
| |
| newlist = PyList_New(len); |
| if (newlist == NULL) { |
| return NULL; |
| } |
| |
| for (i = 0; i < len; i++) { |
| int ret; |
| PyObject *item = PyList_GET_ITEM(self->fut_callbacks, i); |
| |
| if ((ret = PyObject_RichCompareBool(fn, item, Py_EQ)) < 0) { |
| goto fail; |
| } |
| if (ret == 0) { |
| Py_INCREF(item); |
| PyList_SET_ITEM(newlist, j, item); |
| j++; |
| } |
| } |
| |
| if (PyList_SetSlice(newlist, j, len, NULL) < 0) { |
| goto fail; |
| } |
| if (PyList_SetSlice(self->fut_callbacks, 0, len, newlist) < 0) { |
| goto fail; |
| } |
| Py_DECREF(newlist); |
| return PyLong_FromSsize_t(len - j); |
| |
| fail: |
| Py_DECREF(newlist); |
| return NULL; |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.cancel |
| |
| Cancel the future and schedule callbacks. |
| |
| If the future is already done or cancelled, return False. Otherwise, |
| change the future's state to cancelled, schedule the callbacks and |
| return True. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_cancel_impl(FutureObj *self) |
| /*[clinic end generated code: output=e45b932ba8bd68a1 input=515709a127995109]*/ |
| { |
| return future_cancel(self); |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.cancelled |
| |
| Return True if the future was cancelled. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_cancelled_impl(FutureObj *self) |
| /*[clinic end generated code: output=145197ced586357d input=943ab8b7b7b17e45]*/ |
| { |
| if (self->fut_state == STATE_CANCELLED) { |
| Py_RETURN_TRUE; |
| } |
| else { |
| Py_RETURN_FALSE; |
| } |
| } |
| |
| /*[clinic input] |
| _asyncio.Future.done |
| |
| Return True if the future is done. |
| |
| Done means either that a result / exception are available, or that the |
| future was cancelled. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future_done_impl(FutureObj *self) |
| /*[clinic end generated code: output=244c5ac351145096 input=28d7b23fdb65d2ac]*/ |
| { |
| if (self->fut_state == STATE_PENDING) { |
| Py_RETURN_FALSE; |
| } |
| else { |
| Py_RETURN_TRUE; |
| } |
| } |
| |
| static PyObject * |
| FutureObj_get_blocking(FutureObj *fut) |
| { |
| if (fut->fut_blocking) { |
| Py_RETURN_TRUE; |
| } |
| else { |
| Py_RETURN_FALSE; |
| } |
| } |
| |
| static int |
| FutureObj_set_blocking(FutureObj *fut, PyObject *val) |
| { |
| int is_true = PyObject_IsTrue(val); |
| if (is_true < 0) { |
| return -1; |
| } |
| fut->fut_blocking = is_true; |
| return 0; |
| } |
| |
| static PyObject * |
| FutureObj_get_log_traceback(FutureObj *fut) |
| { |
| if (fut->fut_log_tb) { |
| Py_RETURN_TRUE; |
| } |
| else { |
| Py_RETURN_FALSE; |
| } |
| } |
| |
| static PyObject * |
| FutureObj_get_loop(FutureObj *fut) |
| { |
| if (fut->fut_loop == NULL) { |
| Py_RETURN_NONE; |
| } |
| Py_INCREF(fut->fut_loop); |
| return fut->fut_loop; |
| } |
| |
| static PyObject * |
| FutureObj_get_callbacks(FutureObj *fut) |
| { |
| if (fut->fut_callbacks == NULL) { |
| Py_RETURN_NONE; |
| } |
| Py_INCREF(fut->fut_callbacks); |
| return fut->fut_callbacks; |
| } |
| |
| static PyObject * |
| FutureObj_get_result(FutureObj *fut) |
| { |
| if (fut->fut_result == NULL) { |
| Py_RETURN_NONE; |
| } |
| Py_INCREF(fut->fut_result); |
| return fut->fut_result; |
| } |
| |
| static PyObject * |
| FutureObj_get_exception(FutureObj *fut) |
| { |
| if (fut->fut_exception == NULL) { |
| Py_RETURN_NONE; |
| } |
| Py_INCREF(fut->fut_exception); |
| return fut->fut_exception; |
| } |
| |
| static PyObject * |
| FutureObj_get_source_traceback(FutureObj *fut) |
| { |
| if (fut->fut_source_tb == NULL) { |
| Py_RETURN_NONE; |
| } |
| Py_INCREF(fut->fut_source_tb); |
| return fut->fut_source_tb; |
| } |
| |
| static PyObject * |
| FutureObj_get_state(FutureObj *fut) |
| { |
| _Py_IDENTIFIER(PENDING); |
| _Py_IDENTIFIER(CANCELLED); |
| _Py_IDENTIFIER(FINISHED); |
| PyObject *ret = NULL; |
| |
| switch (fut->fut_state) { |
| case STATE_PENDING: |
| ret = _PyUnicode_FromId(&PyId_PENDING); |
| break; |
| case STATE_CANCELLED: |
| ret = _PyUnicode_FromId(&PyId_CANCELLED); |
| break; |
| case STATE_FINISHED: |
| ret = _PyUnicode_FromId(&PyId_FINISHED); |
| break; |
| default: |
| assert (0); |
| } |
| Py_INCREF(ret); |
| return ret; |
| } |
| |
| /*[clinic input] |
| _asyncio.Future._repr_info |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future__repr_info_impl(FutureObj *self) |
| /*[clinic end generated code: output=fa69e901bd176cfb input=f21504d8e2ae1ca2]*/ |
| { |
| return PyObject_CallFunctionObjArgs( |
| asyncio_future_repr_info_func, self, NULL); |
| } |
| |
| /*[clinic input] |
| _asyncio.Future._schedule_callbacks |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Future__schedule_callbacks_impl(FutureObj *self) |
| /*[clinic end generated code: output=5e8958d89ea1c5dc input=4f5f295f263f4a88]*/ |
| { |
| int ret = future_schedule_callbacks(self); |
| if (ret == -1) { |
| return NULL; |
| } |
| Py_RETURN_NONE; |
| } |
| |
| static PyObject * |
| FutureObj_repr(FutureObj *fut) |
| { |
| _Py_IDENTIFIER(_repr_info); |
| |
| PyObject *_repr_info = _PyUnicode_FromId(&PyId__repr_info); // borrowed |
| if (_repr_info == NULL) { |
| return NULL; |
| } |
| |
| PyObject *rinfo = PyObject_CallMethodObjArgs((PyObject*)fut, _repr_info, |
| NULL); |
| if (rinfo == NULL) { |
| return NULL; |
| } |
| |
| PyObject *sp = PyUnicode_FromString(" "); |
| if (sp == NULL) { |
| Py_DECREF(rinfo); |
| return NULL; |
| } |
| |
| PyObject *rinfo_s = PyUnicode_Join(sp, rinfo); |
| Py_DECREF(sp); |
| Py_DECREF(rinfo); |
| if (rinfo_s == NULL) { |
| return NULL; |
| } |
| |
| PyObject *rstr = NULL; |
| PyObject *type_name = PyObject_GetAttrString((PyObject*)Py_TYPE(fut), |
| "__name__"); |
| if (type_name != NULL) { |
| rstr = PyUnicode_FromFormat("<%S %S>", type_name, rinfo_s); |
| Py_DECREF(type_name); |
| } |
| Py_DECREF(rinfo_s); |
| return rstr; |
| } |
| |
| static void |
| FutureObj_finalize(FutureObj *fut) |
| { |
| _Py_IDENTIFIER(call_exception_handler); |
| _Py_IDENTIFIER(message); |
| _Py_IDENTIFIER(exception); |
| _Py_IDENTIFIER(future); |
| _Py_IDENTIFIER(source_traceback); |
| |
| if (!fut->fut_log_tb) { |
| return; |
| } |
| assert(fut->fut_exception != NULL); |
| fut->fut_log_tb = 0;; |
| |
| PyObject *error_type, *error_value, *error_traceback; |
| /* Save the current exception, if any. */ |
| PyErr_Fetch(&error_type, &error_value, &error_traceback); |
| |
| PyObject *context = NULL; |
| PyObject *type_name = NULL; |
| PyObject *message = NULL; |
| PyObject *func = NULL; |
| PyObject *res = NULL; |
| |
| context = PyDict_New(); |
| if (context == NULL) { |
| goto finally; |
| } |
| |
| type_name = PyObject_GetAttrString((PyObject*)Py_TYPE(fut), "__name__"); |
| if (type_name == NULL) { |
| goto finally; |
| } |
| |
| message = PyUnicode_FromFormat( |
| "%S exception was never retrieved", type_name); |
| if (message == NULL) { |
| goto finally; |
| } |
| |
| if (_PyDict_SetItemId(context, &PyId_message, message) < 0 || |
| _PyDict_SetItemId(context, &PyId_exception, fut->fut_exception) < 0 || |
| _PyDict_SetItemId(context, &PyId_future, (PyObject*)fut) < 0) { |
| goto finally; |
| } |
| if (fut->fut_source_tb != NULL) { |
| if (_PyDict_SetItemId(context, &PyId_source_traceback, |
| fut->fut_source_tb) < 0) { |
| goto finally; |
| } |
| } |
| |
| func = _PyObject_GetAttrId(fut->fut_loop, &PyId_call_exception_handler); |
| if (func != NULL) { |
| res = _PyObject_CallArg1(func, context); |
| if (res == NULL) { |
| PyErr_WriteUnraisable(func); |
| } |
| } |
| |
| finally: |
| Py_CLEAR(context); |
| Py_CLEAR(type_name); |
| Py_CLEAR(message); |
| Py_CLEAR(func); |
| Py_CLEAR(res); |
| |
| /* Restore the saved exception. */ |
| PyErr_Restore(error_type, error_value, error_traceback); |
| } |
| |
| |
| static PyAsyncMethods FutureType_as_async = { |
| (unaryfunc)future_new_iter, /* am_await */ |
| 0, /* am_aiter */ |
| 0 /* am_anext */ |
| }; |
| |
| static PyMethodDef FutureType_methods[] = { |
| _ASYNCIO_FUTURE_RESULT_METHODDEF |
| _ASYNCIO_FUTURE_EXCEPTION_METHODDEF |
| _ASYNCIO_FUTURE_SET_RESULT_METHODDEF |
| _ASYNCIO_FUTURE_SET_EXCEPTION_METHODDEF |
| _ASYNCIO_FUTURE_ADD_DONE_CALLBACK_METHODDEF |
| _ASYNCIO_FUTURE_REMOVE_DONE_CALLBACK_METHODDEF |
| _ASYNCIO_FUTURE_CANCEL_METHODDEF |
| _ASYNCIO_FUTURE_CANCELLED_METHODDEF |
| _ASYNCIO_FUTURE_DONE_METHODDEF |
| _ASYNCIO_FUTURE__REPR_INFO_METHODDEF |
| _ASYNCIO_FUTURE__SCHEDULE_CALLBACKS_METHODDEF |
| {NULL, NULL} /* Sentinel */ |
| }; |
| |
| #define FUTURE_COMMON_GETSETLIST \ |
| {"_state", (getter)FutureObj_get_state, NULL, NULL}, \ |
| {"_asyncio_future_blocking", (getter)FutureObj_get_blocking, \ |
| (setter)FutureObj_set_blocking, NULL}, \ |
| {"_loop", (getter)FutureObj_get_loop, NULL, NULL}, \ |
| {"_callbacks", (getter)FutureObj_get_callbacks, NULL, NULL}, \ |
| {"_result", (getter)FutureObj_get_result, NULL, NULL}, \ |
| {"_exception", (getter)FutureObj_get_exception, NULL, NULL}, \ |
| {"_log_traceback", (getter)FutureObj_get_log_traceback, NULL, NULL}, \ |
| {"_source_traceback", (getter)FutureObj_get_source_traceback, NULL, NULL}, |
| |
| static PyGetSetDef FutureType_getsetlist[] = { |
| FUTURE_COMMON_GETSETLIST |
| {NULL} /* Sentinel */ |
| }; |
| |
| static void FutureObj_dealloc(PyObject *self); |
| |
| static PyTypeObject FutureType = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_asyncio.Future", |
| sizeof(FutureObj), /* tp_basicsize */ |
| .tp_dealloc = FutureObj_dealloc, |
| .tp_as_async = &FutureType_as_async, |
| .tp_repr = (reprfunc)FutureObj_repr, |
| .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | Py_TPFLAGS_BASETYPE |
| | Py_TPFLAGS_HAVE_FINALIZE, |
| .tp_doc = _asyncio_Future___init____doc__, |
| .tp_traverse = (traverseproc)FutureObj_traverse, |
| .tp_clear = (inquiry)FutureObj_clear, |
| .tp_weaklistoffset = offsetof(FutureObj, fut_weakreflist), |
| .tp_iter = (getiterfunc)future_new_iter, |
| .tp_methods = FutureType_methods, |
| .tp_getset = FutureType_getsetlist, |
| .tp_dictoffset = offsetof(FutureObj, dict), |
| .tp_init = (initproc)_asyncio_Future___init__, |
| .tp_new = PyType_GenericNew, |
| .tp_finalize = (destructor)FutureObj_finalize, |
| }; |
| |
| #define Future_CheckExact(obj) (Py_TYPE(obj) == &FutureType) |
| |
| static inline int |
| future_call_schedule_callbacks(FutureObj *fut) |
| { |
| if (Future_CheckExact(fut)) { |
| return future_schedule_callbacks(fut); |
| } |
| else { |
| /* `fut` is a subclass of Future */ |
| PyObject *ret = _PyObject_CallMethodId( |
| (PyObject*)fut, &PyId__schedule_callbacks, NULL); |
| if (ret == NULL) { |
| return -1; |
| } |
| |
| Py_DECREF(ret); |
| return 0; |
| } |
| } |
| |
| static void |
| FutureObj_dealloc(PyObject *self) |
| { |
| FutureObj *fut = (FutureObj *)self; |
| |
| if (Future_CheckExact(fut)) { |
| /* When fut is subclass of Future, finalizer is called from |
| * subtype_dealloc. |
| */ |
| if (PyObject_CallFinalizerFromDealloc(self) < 0) { |
| // resurrected. |
| return; |
| } |
| } |
| |
| if (fut->fut_weakreflist != NULL) { |
| PyObject_ClearWeakRefs(self); |
| } |
| |
| (void)FutureObj_clear(fut); |
| Py_TYPE(fut)->tp_free(fut); |
| } |
| |
| |
| /*********************** Future Iterator **************************/ |
| |
| typedef struct { |
| PyObject_HEAD |
| FutureObj *future; |
| } futureiterobject; |
| |
| static void |
| FutureIter_dealloc(futureiterobject *it) |
| { |
| PyObject_GC_UnTrack(it); |
| Py_XDECREF(it->future); |
| PyObject_GC_Del(it); |
| } |
| |
| static PyObject * |
| FutureIter_iternext(futureiterobject *it) |
| { |
| PyObject *res; |
| FutureObj *fut = it->future; |
| |
| if (fut == NULL) { |
| return NULL; |
| } |
| |
| if (fut->fut_state == STATE_PENDING) { |
| if (!fut->fut_blocking) { |
| fut->fut_blocking = 1; |
| Py_INCREF(fut); |
| return (PyObject *)fut; |
| } |
| PyErr_Format(PyExc_AssertionError, |
| "yield from wasn't used with future"); |
| return NULL; |
| } |
| |
| res = _asyncio_Future_result_impl(fut); |
| if (res != NULL) { |
| /* The result of the Future is not an exception. */ |
| if (_PyGen_SetStopIterationValue(res) < 0) { |
| Py_DECREF(res); |
| return NULL; |
| } |
| Py_DECREF(res); |
| } |
| |
| it->future = NULL; |
| Py_DECREF(fut); |
| return NULL; |
| } |
| |
| static PyObject * |
| FutureIter_send(futureiterobject *self, PyObject *unused) |
| { |
| /* Future.__iter__ doesn't care about values that are pushed to the |
| * generator, it just returns "self.result(). |
| */ |
| return FutureIter_iternext(self); |
| } |
| |
| static PyObject * |
| FutureIter_throw(futureiterobject *self, PyObject *args) |
| { |
| PyObject *type=NULL, *val=NULL, *tb=NULL; |
| if (!PyArg_ParseTuple(args, "O|OO", &type, &val, &tb)) |
| return NULL; |
| |
| if (val == Py_None) { |
| val = NULL; |
| } |
| if (tb == Py_None) { |
| tb = NULL; |
| } else if (tb != NULL && !PyTraceBack_Check(tb)) { |
| PyErr_SetString(PyExc_TypeError, "throw() third argument must be a traceback"); |
| return NULL; |
| } |
| |
| Py_INCREF(type); |
| Py_XINCREF(val); |
| Py_XINCREF(tb); |
| |
| if (PyExceptionClass_Check(type)) { |
| PyErr_NormalizeException(&type, &val, &tb); |
| /* No need to call PyException_SetTraceback since we'll be calling |
| PyErr_Restore for `type`, `val`, and `tb`. */ |
| } else if (PyExceptionInstance_Check(type)) { |
| if (val) { |
| PyErr_SetString(PyExc_TypeError, |
| "instance exception may not have a separate value"); |
| goto fail; |
| } |
| val = type; |
| type = PyExceptionInstance_Class(type); |
| Py_INCREF(type); |
| if (tb == NULL) |
| tb = PyException_GetTraceback(val); |
| } else { |
| PyErr_SetString(PyExc_TypeError, |
| "exceptions must be classes deriving BaseException or " |
| "instances of such a class"); |
| goto fail; |
| } |
| |
| Py_CLEAR(self->future); |
| |
| PyErr_Restore(type, val, tb); |
| |
| return FutureIter_iternext(self); |
| |
| fail: |
| Py_DECREF(type); |
| Py_XDECREF(val); |
| Py_XDECREF(tb); |
| return NULL; |
| } |
| |
| static PyObject * |
| FutureIter_close(futureiterobject *self, PyObject *arg) |
| { |
| Py_CLEAR(self->future); |
| Py_RETURN_NONE; |
| } |
| |
| static int |
| FutureIter_traverse(futureiterobject *it, visitproc visit, void *arg) |
| { |
| Py_VISIT(it->future); |
| return 0; |
| } |
| |
| static PyMethodDef FutureIter_methods[] = { |
| {"send", (PyCFunction)FutureIter_send, METH_O, NULL}, |
| {"throw", (PyCFunction)FutureIter_throw, METH_VARARGS, NULL}, |
| {"close", (PyCFunction)FutureIter_close, METH_NOARGS, NULL}, |
| {NULL, NULL} /* Sentinel */ |
| }; |
| |
| static PyTypeObject FutureIterType = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_asyncio.FutureIter", |
| .tp_basicsize = sizeof(futureiterobject), |
| .tp_itemsize = 0, |
| .tp_dealloc = (destructor)FutureIter_dealloc, |
| .tp_getattro = PyObject_GenericGetAttr, |
| .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, |
| .tp_traverse = (traverseproc)FutureIter_traverse, |
| .tp_iter = PyObject_SelfIter, |
| .tp_iternext = (iternextfunc)FutureIter_iternext, |
| .tp_methods = FutureIter_methods, |
| }; |
| |
| static PyObject * |
| future_new_iter(PyObject *fut) |
| { |
| futureiterobject *it; |
| |
| if (!PyObject_TypeCheck(fut, &FutureType)) { |
| PyErr_BadInternalCall(); |
| return NULL; |
| } |
| it = PyObject_GC_New(futureiterobject, &FutureIterType); |
| if (it == NULL) { |
| return NULL; |
| } |
| Py_INCREF(fut); |
| it->future = (FutureObj*)fut; |
| PyObject_GC_Track(it); |
| return (PyObject*)it; |
| } |
| |
| |
| /*********************** Task **************************/ |
| |
| |
| /*[clinic input] |
| class _asyncio.Task "TaskObj *" "&Task_Type" |
| [clinic start generated code]*/ |
| /*[clinic end generated code: output=da39a3ee5e6b4b0d input=719dcef0fcc03b37]*/ |
| |
| static int task_call_step_soon(TaskObj *, PyObject *); |
| static inline PyObject * task_call_wakeup(TaskObj *, PyObject *); |
| static inline PyObject * task_call_step(TaskObj *, PyObject *); |
| static PyObject * task_wakeup(TaskObj *, PyObject *); |
| static PyObject * task_step(TaskObj *, PyObject *); |
| |
| /* ----- Task._step wrapper */ |
| |
| static int |
| TaskSendMethWrapper_clear(TaskSendMethWrapper *o) |
| { |
| Py_CLEAR(o->sw_task); |
| Py_CLEAR(o->sw_arg); |
| return 0; |
| } |
| |
| static void |
| TaskSendMethWrapper_dealloc(TaskSendMethWrapper *o) |
| { |
| PyObject_GC_UnTrack(o); |
| (void)TaskSendMethWrapper_clear(o); |
| Py_TYPE(o)->tp_free(o); |
| } |
| |
| static PyObject * |
| TaskSendMethWrapper_call(TaskSendMethWrapper *o, |
| PyObject *args, PyObject *kwds) |
| { |
| return task_call_step(o->sw_task, o->sw_arg); |
| } |
| |
| static int |
| TaskSendMethWrapper_traverse(TaskSendMethWrapper *o, |
| visitproc visit, void *arg) |
| { |
| Py_VISIT(o->sw_task); |
| Py_VISIT(o->sw_arg); |
| return 0; |
| } |
| |
| static PyObject * |
| TaskSendMethWrapper_get___self__(TaskSendMethWrapper *o) |
| { |
| if (o->sw_task) { |
| Py_INCREF(o->sw_task); |
| return (PyObject*)o->sw_task; |
| } |
| Py_RETURN_NONE; |
| } |
| |
| static PyGetSetDef TaskSendMethWrapper_getsetlist[] = { |
| {"__self__", (getter)TaskSendMethWrapper_get___self__, NULL, NULL}, |
| {NULL} /* Sentinel */ |
| }; |
| |
| PyTypeObject TaskSendMethWrapper_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "TaskSendMethWrapper", |
| .tp_basicsize = sizeof(TaskSendMethWrapper), |
| .tp_itemsize = 0, |
| .tp_getset = TaskSendMethWrapper_getsetlist, |
| .tp_dealloc = (destructor)TaskSendMethWrapper_dealloc, |
| .tp_call = (ternaryfunc)TaskSendMethWrapper_call, |
| .tp_getattro = PyObject_GenericGetAttr, |
| .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, |
| .tp_traverse = (traverseproc)TaskSendMethWrapper_traverse, |
| .tp_clear = (inquiry)TaskSendMethWrapper_clear, |
| }; |
| |
| static PyObject * |
| TaskSendMethWrapper_new(TaskObj *task, PyObject *arg) |
| { |
| TaskSendMethWrapper *o; |
| o = PyObject_GC_New(TaskSendMethWrapper, &TaskSendMethWrapper_Type); |
| if (o == NULL) { |
| return NULL; |
| } |
| |
| Py_INCREF(task); |
| o->sw_task = task; |
| |
| Py_XINCREF(arg); |
| o->sw_arg = arg; |
| |
| PyObject_GC_Track(o); |
| return (PyObject*) o; |
| } |
| |
| /* ----- Task._wakeup wrapper */ |
| |
| static PyObject * |
| TaskWakeupMethWrapper_call(TaskWakeupMethWrapper *o, |
| PyObject *args, PyObject *kwds) |
| { |
| PyObject *fut; |
| |
| if (!PyArg_ParseTuple(args, "O|", &fut)) { |
| return NULL; |
| } |
| |
| return task_call_wakeup(o->ww_task, fut); |
| } |
| |
| static int |
| TaskWakeupMethWrapper_clear(TaskWakeupMethWrapper *o) |
| { |
| Py_CLEAR(o->ww_task); |
| return 0; |
| } |
| |
| static int |
| TaskWakeupMethWrapper_traverse(TaskWakeupMethWrapper *o, |
| visitproc visit, void *arg) |
| { |
| Py_VISIT(o->ww_task); |
| return 0; |
| } |
| |
| static void |
| TaskWakeupMethWrapper_dealloc(TaskWakeupMethWrapper *o) |
| { |
| PyObject_GC_UnTrack(o); |
| (void)TaskWakeupMethWrapper_clear(o); |
| Py_TYPE(o)->tp_free(o); |
| } |
| |
| PyTypeObject TaskWakeupMethWrapper_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "TaskWakeupMethWrapper", |
| .tp_basicsize = sizeof(TaskWakeupMethWrapper), |
| .tp_itemsize = 0, |
| .tp_dealloc = (destructor)TaskWakeupMethWrapper_dealloc, |
| .tp_call = (ternaryfunc)TaskWakeupMethWrapper_call, |
| .tp_getattro = PyObject_GenericGetAttr, |
| .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, |
| .tp_traverse = (traverseproc)TaskWakeupMethWrapper_traverse, |
| .tp_clear = (inquiry)TaskWakeupMethWrapper_clear, |
| }; |
| |
| static PyObject * |
| TaskWakeupMethWrapper_new(TaskObj *task) |
| { |
| TaskWakeupMethWrapper *o; |
| o = PyObject_GC_New(TaskWakeupMethWrapper, &TaskWakeupMethWrapper_Type); |
| if (o == NULL) { |
| return NULL; |
| } |
| |
| Py_INCREF(task); |
| o->ww_task = task; |
| |
| PyObject_GC_Track(o); |
| return (PyObject*) o; |
| } |
| |
| /* ----- Task */ |
| |
| /*[clinic input] |
| _asyncio.Task.__init__ |
| |
| coro: 'O' |
| * |
| loop: 'O' = NULL |
| |
| A coroutine wrapped in a Future. |
| [clinic start generated code]*/ |
| |
| static int |
| _asyncio_Task___init___impl(TaskObj *self, PyObject *coro, PyObject *loop) |
| /*[clinic end generated code: output=9f24774c2287fc2f input=71d8d28c201a18cd]*/ |
| { |
| PyObject *res; |
| _Py_IDENTIFIER(add); |
| |
| if (future_init((FutureObj*)self, loop)) { |
| return -1; |
| } |
| |
| self->task_fut_waiter = NULL; |
| self->task_must_cancel = 0; |
| self->task_log_destroy_pending = 1; |
| |
| Py_INCREF(coro); |
| self->task_coro = coro; |
| |
| if (task_call_step_soon(self, NULL)) { |
| return -1; |
| } |
| |
| res = _PyObject_CallMethodIdObjArgs(all_tasks, &PyId_add, self, NULL); |
| if (res == NULL) { |
| return -1; |
| } |
| Py_DECREF(res); |
| |
| return 0; |
| } |
| |
| static int |
| TaskObj_clear(TaskObj *task) |
| { |
| (void)FutureObj_clear((FutureObj*) task); |
| Py_CLEAR(task->task_coro); |
| Py_CLEAR(task->task_fut_waiter); |
| return 0; |
| } |
| |
| static int |
| TaskObj_traverse(TaskObj *task, visitproc visit, void *arg) |
| { |
| Py_VISIT(task->task_coro); |
| Py_VISIT(task->task_fut_waiter); |
| (void)FutureObj_traverse((FutureObj*) task, visit, arg); |
| return 0; |
| } |
| |
| static PyObject * |
| TaskObj_get_log_destroy_pending(TaskObj *task) |
| { |
| if (task->task_log_destroy_pending) { |
| Py_RETURN_TRUE; |
| } |
| else { |
| Py_RETURN_FALSE; |
| } |
| } |
| |
| static int |
| TaskObj_set_log_destroy_pending(TaskObj *task, PyObject *val) |
| { |
| int is_true = PyObject_IsTrue(val); |
| if (is_true < 0) { |
| return -1; |
| } |
| task->task_log_destroy_pending = is_true; |
| return 0; |
| } |
| |
| static PyObject * |
| TaskObj_get_must_cancel(TaskObj *task) |
| { |
| if (task->task_must_cancel) { |
| Py_RETURN_TRUE; |
| } |
| else { |
| Py_RETURN_FALSE; |
| } |
| } |
| |
| static PyObject * |
| TaskObj_get_coro(TaskObj *task) |
| { |
| if (task->task_coro) { |
| Py_INCREF(task->task_coro); |
| return task->task_coro; |
| } |
| |
| Py_RETURN_NONE; |
| } |
| |
| static PyObject * |
| TaskObj_get_fut_waiter(TaskObj *task) |
| { |
| if (task->task_fut_waiter) { |
| Py_INCREF(task->task_fut_waiter); |
| return task->task_fut_waiter; |
| } |
| |
| Py_RETURN_NONE; |
| } |
| |
| /*[clinic input] |
| @classmethod |
| _asyncio.Task.current_task |
| |
| loop: 'O' = NULL |
| |
| Return the currently running task in an event loop or None. |
| |
| By default the current task for the current event loop is returned. |
| |
| None is returned when called not in the context of a Task. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task_current_task_impl(PyTypeObject *type, PyObject *loop) |
| /*[clinic end generated code: output=99fbe7332c516e03 input=cd784537f02cf833]*/ |
| { |
| PyObject *res; |
| |
| if (loop == NULL) { |
| loop = PyObject_CallObject(asyncio_get_event_loop, NULL); |
| if (loop == NULL) { |
| return NULL; |
| } |
| |
| res = PyDict_GetItem(current_tasks, loop); |
| Py_DECREF(loop); |
| } |
| else { |
| res = PyDict_GetItem(current_tasks, loop); |
| } |
| |
| if (res == NULL) { |
| Py_RETURN_NONE; |
| } |
| else { |
| Py_INCREF(res); |
| return res; |
| } |
| } |
| |
| static PyObject * |
| task_all_tasks(PyObject *loop) |
| { |
| PyObject *task; |
| PyObject *task_loop; |
| PyObject *set; |
| PyObject *iter; |
| |
| assert(loop != NULL); |
| |
| set = PySet_New(NULL); |
| if (set == NULL) { |
| return NULL; |
| } |
| |
| iter = PyObject_GetIter(all_tasks); |
| if (iter == NULL) { |
| goto fail; |
| } |
| |
| while ((task = PyIter_Next(iter))) { |
| task_loop = PyObject_GetAttrString(task, "_loop"); |
| if (task_loop == NULL) { |
| Py_DECREF(task); |
| goto fail; |
| } |
| if (task_loop == loop) { |
| if (PySet_Add(set, task) == -1) { |
| Py_DECREF(task_loop); |
| Py_DECREF(task); |
| goto fail; |
| } |
| } |
| Py_DECREF(task_loop); |
| Py_DECREF(task); |
| } |
| |
| Py_DECREF(iter); |
| return set; |
| |
| fail: |
| Py_XDECREF(set); |
| Py_XDECREF(iter); |
| return NULL; |
| } |
| |
| /*[clinic input] |
| @classmethod |
| _asyncio.Task.all_tasks |
| |
| loop: 'O' = NULL |
| |
| Return a set of all tasks for an event loop. |
| |
| By default all tasks for the current event loop are returned. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task_all_tasks_impl(PyTypeObject *type, PyObject *loop) |
| /*[clinic end generated code: output=11f9b20749ccca5d input=cd64aa5f88bd5c49]*/ |
| { |
| PyObject *res; |
| |
| if (loop == NULL) { |
| loop = PyObject_CallObject(asyncio_get_event_loop, NULL); |
| if (loop == NULL) { |
| return NULL; |
| } |
| |
| res = task_all_tasks(loop); |
| Py_DECREF(loop); |
| } |
| else { |
| res = task_all_tasks(loop); |
| } |
| |
| return res; |
| } |
| |
| /*[clinic input] |
| _asyncio.Task._repr_info |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task__repr_info_impl(TaskObj *self) |
| /*[clinic end generated code: output=6a490eb66d5ba34b input=3c6d051ed3ddec8b]*/ |
| { |
| return PyObject_CallFunctionObjArgs( |
| asyncio_task_repr_info_func, self, NULL); |
| } |
| |
| /*[clinic input] |
| _asyncio.Task.cancel |
| |
| Request that this task cancel itself. |
| |
| This arranges for a CancelledError to be thrown into the |
| wrapped coroutine on the next cycle through the event loop. |
| The coroutine then has a chance to clean up or even deny |
| the request using try/except/finally. |
| |
| Unlike Future.cancel, this does not guarantee that the |
| task will be cancelled: the exception might be caught and |
| acted upon, delaying cancellation of the task or preventing |
| cancellation completely. The task may also return a value or |
| raise a different exception. |
| |
| Immediately after this method is called, Task.cancelled() will |
| not return True (unless the task was already cancelled). A |
| task will be marked as cancelled when the wrapped coroutine |
| terminates with a CancelledError exception (even if cancel() |
| was not called). |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task_cancel_impl(TaskObj *self) |
| /*[clinic end generated code: output=6bfc0479da9d5757 input=13f9bf496695cb52]*/ |
| { |
| if (self->task_state != STATE_PENDING) { |
| Py_RETURN_FALSE; |
| } |
| |
| if (self->task_fut_waiter) { |
| PyObject *res; |
| int is_true; |
| |
| res = _PyObject_CallMethodId( |
| self->task_fut_waiter, &PyId_cancel, NULL); |
| if (res == NULL) { |
| return NULL; |
| } |
| |
| is_true = PyObject_IsTrue(res); |
| Py_DECREF(res); |
| if (is_true < 0) { |
| return NULL; |
| } |
| |
| if (is_true) { |
| Py_RETURN_TRUE; |
| } |
| } |
| |
| self->task_must_cancel = 1; |
| Py_RETURN_TRUE; |
| } |
| |
| /*[clinic input] |
| _asyncio.Task.get_stack |
| |
| * |
| limit: 'O' = None |
| |
| Return the list of stack frames for this task's coroutine. |
| |
| If the coroutine is not done, this returns the stack where it is |
| suspended. If the coroutine has completed successfully or was |
| cancelled, this returns an empty list. If the coroutine was |
| terminated by an exception, this returns the list of traceback |
| frames. |
| |
| The frames are always ordered from oldest to newest. |
| |
| The optional limit gives the maximum number of frames to |
| return; by default all available frames are returned. Its |
| meaning differs depending on whether a stack or a traceback is |
| returned: the newest frames of a stack are returned, but the |
| oldest frames of a traceback are returned. (This matches the |
| behavior of the traceback module.) |
| |
| For reasons beyond our control, only one stack frame is |
| returned for a suspended coroutine. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task_get_stack_impl(TaskObj *self, PyObject *limit) |
| /*[clinic end generated code: output=c9aeeeebd1e18118 input=b1920230a766d17a]*/ |
| { |
| return PyObject_CallFunctionObjArgs( |
| asyncio_task_get_stack_func, self, limit, NULL); |
| } |
| |
| /*[clinic input] |
| _asyncio.Task.print_stack |
| |
| * |
| limit: 'O' = None |
| file: 'O' = None |
| |
| Print the stack or traceback for this task's coroutine. |
| |
| This produces output similar to that of the traceback module, |
| for the frames retrieved by get_stack(). The limit argument |
| is passed to get_stack(). The file argument is an I/O stream |
| to which the output is written; by default output is written |
| to sys.stderr. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task_print_stack_impl(TaskObj *self, PyObject *limit, |
| PyObject *file) |
| /*[clinic end generated code: output=7339e10314cd3f4d input=19f1e99ab5400bc3]*/ |
| { |
| return PyObject_CallFunctionObjArgs( |
| asyncio_task_print_stack_func, self, limit, file, NULL); |
| } |
| |
| /*[clinic input] |
| _asyncio.Task._step |
| |
| exc: 'O' = NULL |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task__step_impl(TaskObj *self, PyObject *exc) |
| /*[clinic end generated code: output=7ed23f0cefd5ae42 input=ada4b2324e5370af]*/ |
| { |
| return task_step(self, exc == Py_None ? NULL : exc); |
| } |
| |
| /*[clinic input] |
| _asyncio.Task._wakeup |
| |
| fut: 'O' |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _asyncio_Task__wakeup_impl(TaskObj *self, PyObject *fut) |
| /*[clinic end generated code: output=75cb341c760fd071 input=11ee4918a5bdbf21]*/ |
| { |
| return task_wakeup(self, fut); |
| } |
| |
| static void |
| TaskObj_finalize(TaskObj *task) |
| { |
| _Py_IDENTIFIER(call_exception_handler); |
| _Py_IDENTIFIER(task); |
| _Py_IDENTIFIER(message); |
| _Py_IDENTIFIER(source_traceback); |
| |
| PyObject *message = NULL; |
| PyObject *context = NULL; |
| PyObject *func = NULL; |
| PyObject *res = NULL; |
| |
| PyObject *error_type, *error_value, *error_traceback; |
| |
| if (task->task_state != STATE_PENDING || !task->task_log_destroy_pending) { |
| goto done; |
| } |
| |
| /* Save the current exception, if any. */ |
| PyErr_Fetch(&error_type, &error_value, &error_traceback); |
| |
| context = PyDict_New(); |
| if (context == NULL) { |
| goto finally; |
| } |
| |
| message = PyUnicode_FromString("Task was destroyed but it is pending!"); |
| if (message == NULL) { |
| goto finally; |
| } |
| |
| if (_PyDict_SetItemId(context, &PyId_message, message) < 0 || |
| _PyDict_SetItemId(context, &PyId_task, (PyObject*)task) < 0) |
| { |
| goto finally; |
| } |
| |
| if (task->task_source_tb != NULL) { |
| if (_PyDict_SetItemId(context, &PyId_source_traceback, |
| task->task_source_tb) < 0) |
| { |
| goto finally; |
| } |
| } |
| |
| func = _PyObject_GetAttrId(task->task_loop, &PyId_call_exception_handler); |
| if (func != NULL) { |
| res = _PyObject_CallArg1(func, context); |
| if (res == NULL) { |
| PyErr_WriteUnraisable(func); |
| } |
| } |
| |
| finally: |
| Py_CLEAR(context); |
| Py_CLEAR(message); |
| Py_CLEAR(func); |
| Py_CLEAR(res); |
| |
| /* Restore the saved exception. */ |
| PyErr_Restore(error_type, error_value, error_traceback); |
| |
| done: |
| FutureObj_finalize((FutureObj*)task); |
| } |
| |
| static void TaskObj_dealloc(PyObject *); /* Needs Task_CheckExact */ |
| |
| static PyMethodDef TaskType_methods[] = { |
| _ASYNCIO_FUTURE_RESULT_METHODDEF |
| _ASYNCIO_FUTURE_EXCEPTION_METHODDEF |
| _ASYNCIO_FUTURE_SET_RESULT_METHODDEF |
| _ASYNCIO_FUTURE_SET_EXCEPTION_METHODDEF |
| _ASYNCIO_FUTURE_ADD_DONE_CALLBACK_METHODDEF |
| _ASYNCIO_FUTURE_REMOVE_DONE_CALLBACK_METHODDEF |
| _ASYNCIO_FUTURE_CANCELLED_METHODDEF |
| _ASYNCIO_FUTURE_DONE_METHODDEF |
| _ASYNCIO_TASK_CURRENT_TASK_METHODDEF |
| _ASYNCIO_TASK_ALL_TASKS_METHODDEF |
| _ASYNCIO_TASK_CANCEL_METHODDEF |
| _ASYNCIO_TASK_GET_STACK_METHODDEF |
| _ASYNCIO_TASK_PRINT_STACK_METHODDEF |
| _ASYNCIO_TASK__WAKEUP_METHODDEF |
| _ASYNCIO_TASK__STEP_METHODDEF |
| _ASYNCIO_TASK__REPR_INFO_METHODDEF |
| {NULL, NULL} /* Sentinel */ |
| }; |
| |
| static PyGetSetDef TaskType_getsetlist[] = { |
| FUTURE_COMMON_GETSETLIST |
| {"_log_destroy_pending", (getter)TaskObj_get_log_destroy_pending, |
| (setter)TaskObj_set_log_destroy_pending, NULL}, |
| {"_must_cancel", (getter)TaskObj_get_must_cancel, NULL, NULL}, |
| {"_coro", (getter)TaskObj_get_coro, NULL, NULL}, |
| {"_fut_waiter", (getter)TaskObj_get_fut_waiter, NULL, NULL}, |
| {NULL} /* Sentinel */ |
| }; |
| |
| static PyTypeObject TaskType = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_asyncio.Task", |
| sizeof(TaskObj), /* tp_basicsize */ |
| .tp_base = &FutureType, |
| .tp_dealloc = TaskObj_dealloc, |
| .tp_as_async = &FutureType_as_async, |
| .tp_repr = (reprfunc)FutureObj_repr, |
| .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | Py_TPFLAGS_BASETYPE |
| | Py_TPFLAGS_HAVE_FINALIZE, |
| .tp_doc = _asyncio_Task___init____doc__, |
| .tp_traverse = (traverseproc)TaskObj_traverse, |
| .tp_clear = (inquiry)TaskObj_clear, |
| .tp_weaklistoffset = offsetof(TaskObj, task_weakreflist), |
| .tp_iter = (getiterfunc)future_new_iter, |
| .tp_methods = TaskType_methods, |
| .tp_getset = TaskType_getsetlist, |
| .tp_dictoffset = offsetof(TaskObj, dict), |
| .tp_init = (initproc)_asyncio_Task___init__, |
| .tp_new = PyType_GenericNew, |
| .tp_finalize = (destructor)TaskObj_finalize, |
| }; |
| |
| #define Task_CheckExact(obj) (Py_TYPE(obj) == &TaskType) |
| |
| static void |
| TaskObj_dealloc(PyObject *self) |
| { |
| TaskObj *task = (TaskObj *)self; |
| |
| if (Task_CheckExact(self)) { |
| /* When fut is subclass of Task, finalizer is called from |
| * subtype_dealloc. |
| */ |
| if (PyObject_CallFinalizerFromDealloc(self) < 0) { |
| // resurrected. |
| return; |
| } |
| } |
| |
| if (task->task_weakreflist != NULL) { |
| PyObject_ClearWeakRefs(self); |
| } |
| |
| (void)TaskObj_clear(task); |
| Py_TYPE(task)->tp_free(task); |
| } |
| |
| static inline PyObject * |
| task_call_wakeup(TaskObj *task, PyObject *fut) |
| { |
| if (Task_CheckExact(task)) { |
| return task_wakeup(task, fut); |
| } |
| else { |
| /* `task` is a subclass of Task */ |
| return _PyObject_CallMethodIdObjArgs((PyObject*)task, &PyId__wakeup, |
| fut, NULL); |
| } |
| } |
| |
| static inline PyObject * |
| task_call_step(TaskObj *task, PyObject *arg) |
| { |
| if (Task_CheckExact(task)) { |
| return task_step(task, arg); |
| } |
| else { |
| /* `task` is a subclass of Task */ |
| if (arg == NULL) { |
| arg = Py_None; |
| } |
| return _PyObject_CallMethodIdObjArgs((PyObject*)task, &PyId__step, |
| arg, NULL); |
| } |
| } |
| |
| static int |
| task_call_step_soon(TaskObj *task, PyObject *arg) |
| { |
| PyObject *handle; |
| |
| PyObject *cb = TaskSendMethWrapper_new(task, arg); |
| if (cb == NULL) { |
| return -1; |
| } |
| |
| handle = _PyObject_CallMethodIdObjArgs(task->task_loop, &PyId_call_soon, |
| cb, NULL); |
| Py_DECREF(cb); |
| if (handle == NULL) { |
| return -1; |
| } |
| |
| Py_DECREF(handle); |
| return 0; |
| } |
| |
| static PyObject * |
| task_set_error_soon(TaskObj *task, PyObject *et, const char *format, ...) |
| { |
| PyObject* msg; |
| |
| va_list vargs; |
| #ifdef HAVE_STDARG_PROTOTYPES |
| va_start(vargs, format); |
| #else |
| va_start(vargs); |
| #endif |
| msg = PyUnicode_FromFormatV(format, vargs); |
| va_end(vargs); |
| |
| if (msg == NULL) { |
| return NULL; |
| } |
| |
| PyObject *e = PyObject_CallFunctionObjArgs(et, msg, NULL); |
| Py_DECREF(msg); |
| if (e == NULL) { |
| return NULL; |
| } |
| |
| if (task_call_step_soon(task, e) == -1) { |
| Py_DECREF(e); |
| return NULL; |
| } |
| |
| Py_DECREF(e); |
| Py_RETURN_NONE; |
| } |
| |
| static PyObject * |
| task_step_impl(TaskObj *task, PyObject *exc) |
| { |
| int res; |
| int clear_exc = 0; |
| PyObject *result = NULL; |
| PyObject *coro = task->task_coro; |
| PyObject *o; |
| |
| if (task->task_state != STATE_PENDING) { |
| PyErr_Format(PyExc_AssertionError, |
| "_step(): already done: %R %R", |
| task, |
| exc ? exc : Py_None); |
| goto fail; |
| } |
| |
| if (task->task_must_cancel) { |
| assert(exc != Py_None); |
| |
| if (exc) { |
| /* Check if exc is a CancelledError */ |
| res = PyObject_IsInstance(exc, asyncio_CancelledError); |
| if (res == -1) { |
| /* An error occurred, abort */ |
| goto fail; |
| } |
| if (res == 0) { |
| /* exc is not CancelledError; reset it to NULL */ |
| exc = NULL; |
| } |
| } |
| |
| if (!exc) { |
| /* exc was not a CancelledError */ |
| exc = PyObject_CallFunctionObjArgs(asyncio_CancelledError, NULL); |
| if (!exc) { |
| goto fail; |
| } |
| clear_exc = 1; |
| } |
| |
| task->task_must_cancel = 0; |
| } |
| |
| Py_CLEAR(task->task_fut_waiter); |
| |
| if (exc == NULL) { |
| if (PyGen_CheckExact(coro) || PyCoro_CheckExact(coro)) { |
| result = _PyGen_Send((PyGenObject*)coro, Py_None); |
| } |
| else { |
| result = _PyObject_CallMethodIdObjArgs( |
| coro, &PyId_send, Py_None, NULL); |
| } |
| } |
| else { |
| result = _PyObject_CallMethodIdObjArgs( |
| coro, &PyId_throw, exc, NULL); |
| if (clear_exc) { |
| /* We created 'exc' during this call */ |
| Py_CLEAR(exc); |
| } |
| } |
| |
| if (result == NULL) { |
| PyObject *et, *ev, *tb; |
| |
| if (_PyGen_FetchStopIterationValue(&o) == 0) { |
| /* The error is StopIteration and that means that |
| the underlying coroutine has resolved */ |
| PyObject *res = future_set_result((FutureObj*)task, o); |
| Py_DECREF(o); |
| if (res == NULL) { |
| return NULL; |
| } |
| Py_DECREF(res); |
| Py_RETURN_NONE; |
| } |
| |
| if (PyErr_ExceptionMatches(asyncio_CancelledError)) { |
| /* CancelledError */ |
| PyErr_Clear(); |
| return future_cancel((FutureObj*)task); |
| } |
| |
| /* Some other exception; pop it and call Task.set_exception() */ |
| PyErr_Fetch(&et, &ev, &tb); |
| assert(et); |
| if (!ev || !PyObject_TypeCheck(ev, (PyTypeObject *) et)) { |
| PyErr_NormalizeException(&et, &ev, &tb); |
| } |
| if (tb != NULL) { |
| PyException_SetTraceback(ev, tb); |
| } |
| o = future_set_exception((FutureObj*)task, ev); |
| if (!o) { |
| /* An exception in Task.set_exception() */ |
| Py_XDECREF(et); |
| Py_XDECREF(tb); |
| Py_XDECREF(ev); |
| goto fail; |
| } |
| assert(o == Py_None); |
| Py_CLEAR(o); |
| |
| if (!PyErr_GivenExceptionMatches(et, PyExc_Exception)) { |
| /* We've got a BaseException; re-raise it */ |
| PyErr_Restore(et, ev, tb); |
| goto fail; |
| } |
| |
| Py_XDECREF(et); |
| Py_XDECREF(tb); |
| Py_XDECREF(ev); |
| |
| Py_RETURN_NONE; |
| } |
| |
| if (result == (PyObject*)task) { |
| /* We have a task that wants to await on itself */ |
| goto self_await; |
| } |
| |
| /* Check if `result` is FutureObj or TaskObj (and not a subclass) */ |
| if (Future_CheckExact(result) || Task_CheckExact(result)) { |
| PyObject *wrapper; |
| PyObject *res; |
| FutureObj *fut = (FutureObj*)result; |
| |
| /* Check if `result` future is attached to a different loop */ |
| if (fut->fut_loop != task->task_loop) { |
| goto different_loop; |
| } |
| |
| if (fut->fut_blocking) { |
| fut->fut_blocking = 0; |
| |
| /* result.add_done_callback(task._wakeup) */ |
| wrapper = TaskWakeupMethWrapper_new(task); |
| if (wrapper == NULL) { |
| goto fail; |
| } |
| res = future_add_done_callback((FutureObj*)result, wrapper); |
| Py_DECREF(wrapper); |
| if (res == NULL) { |
| goto fail; |
| } |
| Py_DECREF(res); |
| |
| /* task._fut_waiter = result */ |
| task->task_fut_waiter = result; /* no incref is necessary */ |
| |
| if (task->task_must_cancel) { |
| PyObject *r; |
| r = future_cancel(fut); |
| if (r == NULL) { |
| return NULL; |
| } |
| if (r == Py_True) { |
| task->task_must_cancel = 0; |
| } |
| Py_DECREF(r); |
| } |
| |
| Py_RETURN_NONE; |
| } |
| else { |
| goto yield_insteadof_yf; |
| } |
| } |
| |
| /* Check if `result` is a Future-compatible object */ |
| o = PyObject_GetAttrString(result, "_asyncio_future_blocking"); |
| if (o == NULL) { |
| if (PyErr_ExceptionMatches(PyExc_AttributeError)) { |
| PyErr_Clear(); |
| } |
| else { |
| goto fail; |
| } |
| } |
| else { |
| if (o == Py_None) { |
| Py_CLEAR(o); |
| } |
| else { |
| /* `result` is a Future-compatible object */ |
| PyObject *wrapper; |
| PyObject *res; |
| |
| int blocking = PyObject_IsTrue(o); |
| Py_CLEAR(o); |
| if (blocking < 0) { |
| goto fail; |
| } |
| |
| /* Check if `result` future is attached to a different loop */ |
| PyObject *oloop = PyObject_GetAttrString(result, "_loop"); |
| if (oloop == NULL) { |
| goto fail; |
| } |
| if (oloop != task->task_loop) { |
| Py_DECREF(oloop); |
| goto different_loop; |
| } |
| else { |
| Py_DECREF(oloop); |
| } |
| |
| if (blocking) { |
| /* result._asyncio_future_blocking = False */ |
| if (PyObject_SetAttrString( |
| result, "_asyncio_future_blocking", Py_False) == -1) { |
| goto fail; |
| } |
| |
| /* result.add_done_callback(task._wakeup) */ |
| wrapper = TaskWakeupMethWrapper_new(task); |
| if (wrapper == NULL) { |
| goto fail; |
| } |
| res = _PyObject_CallMethodIdObjArgs(result, |
| &PyId_add_done_callback, |
| wrapper, NULL); |
| Py_DECREF(wrapper); |
| if (res == NULL) { |
| goto fail; |
| } |
| Py_DECREF(res); |
| |
| /* task._fut_waiter = result */ |
| task->task_fut_waiter = result; /* no incref is necessary */ |
| |
| if (task->task_must_cancel) { |
| PyObject *r; |
| int is_true; |
| r = _PyObject_CallMethodId(result, &PyId_cancel, NULL); |
| if (r == NULL) { |
| return NULL; |
| } |
| is_true = PyObject_IsTrue(r); |
| Py_DECREF(r); |
| if (is_true < 0) { |
| return NULL; |
| } |
| else if (is_true) { |
| task->task_must_cancel = 0; |
| } |
| } |
| |
| Py_RETURN_NONE; |
| } |
| else { |
| goto yield_insteadof_yf; |
| } |
| } |
| } |
| |
| /* Check if `result` is None */ |
| if (result == Py_None) { |
| /* Bare yield relinquishes control for one event loop iteration. */ |
| if (task_call_step_soon(task, NULL)) { |
| goto fail; |
| } |
| return result; |
| } |
| |
| /* Check if `result` is a generator */ |
| o = PyObject_CallFunctionObjArgs(inspect_isgenerator, result, NULL); |
| if (o == NULL) { |
| /* An exception in inspect.isgenerator */ |
| goto fail; |
| } |
| res = PyObject_IsTrue(o); |
| Py_CLEAR(o); |
| if (res == -1) { |
| /* An exception while checking if 'val' is True */ |
| goto fail; |
| } |
| if (res == 1) { |
| /* `result` is a generator */ |
| PyObject *ret; |
| ret = task_set_error_soon( |
| task, PyExc_RuntimeError, |
| "yield was used instead of yield from for " |
| "generator in task %R with %S", task, result); |
| Py_DECREF(result); |
| return ret; |
| } |
| |
| /* The `result` is none of the above */ |
| Py_DECREF(result); |
| return task_set_error_soon( |
| task, PyExc_RuntimeError, "Task got bad yield: %R", result); |
| |
| self_await: |
| o = task_set_error_soon( |
| task, PyExc_RuntimeError, |
| "Task cannot await on itself: %R", task); |
| Py_DECREF(result); |
| return o; |
| |
| yield_insteadof_yf: |
| o = task_set_error_soon( |
| task, PyExc_RuntimeError, |
| "yield was used instead of yield from " |
| "in task %R with %R", |
| task, result); |
| Py_DECREF(result); |
| return o; |
| |
| different_loop: |
| o = task_set_error_soon( |
| task, PyExc_RuntimeError, |
| "Task %R got Future %R attached to a different loop", |
| task, result); |
| Py_DECREF(result); |
| return o; |
| |
| fail: |
| Py_XDECREF(result); |
| return NULL; |
| } |
| |
| static PyObject * |
| task_step(TaskObj *task, PyObject *exc) |
| { |
| PyObject *res; |
| PyObject *ot; |
| |
| if (PyDict_SetItem(current_tasks, |
| task->task_loop, (PyObject*)task) == -1) |
| { |
| return NULL; |
| } |
| |
| res = task_step_impl(task, exc); |
| |
| if (res == NULL) { |
| PyObject *et, *ev, *tb; |
| PyErr_Fetch(&et, &ev, &tb); |
| ot = _PyDict_Pop(current_tasks, task->task_loop, NULL); |
| if (ot == NULL) { |
| Py_XDECREF(et); |
| Py_XDECREF(tb); |
| Py_XDECREF(ev); |
| return NULL; |
| } |
| Py_DECREF(ot); |
| PyErr_Restore(et, ev, tb); |
| return NULL; |
| } |
| else { |
| ot = _PyDict_Pop(current_tasks, task->task_loop, NULL); |
| if (ot == NULL) { |
| Py_DECREF(res); |
| return NULL; |
| } |
| else { |
| Py_DECREF(ot); |
| return res; |
| } |
| } |
| } |
| |
| static PyObject * |
| task_wakeup(TaskObj *task, PyObject *o) |
| { |
| assert(o); |
| |
| if (Future_CheckExact(o) || Task_CheckExact(o)) { |
| PyObject *fut_result = NULL; |
| int res = future_get_result((FutureObj*)o, &fut_result); |
| PyObject *result; |
| |
| switch(res) { |
| case -1: |
| assert(fut_result == NULL); |
| return NULL; |
| case 0: |
| Py_DECREF(fut_result); |
| return task_call_step(task, NULL); |
| default: |
| assert(res == 1); |
| result = task_call_step(task, fut_result); |
| Py_DECREF(fut_result); |
| return result; |
| } |
| } |
| |
| PyObject *fut_result = PyObject_CallMethod(o, "result", NULL); |
| if (fut_result == NULL) { |
| PyObject *et, *ev, *tb; |
| PyObject *res; |
| |
| PyErr_Fetch(&et, &ev, &tb); |
| if (!ev || !PyObject_TypeCheck(ev, (PyTypeObject *) et)) { |
| PyErr_NormalizeException(&et, &ev, &tb); |
| } |
| |
| res = task_call_step(task, ev); |
| |
| Py_XDECREF(et); |
| Py_XDECREF(tb); |
| Py_XDECREF(ev); |
| |
| return res; |
| } |
| else { |
| Py_DECREF(fut_result); |
| return task_call_step(task, NULL); |
| } |
| } |
| |
| |
| /*********************** Module **************************/ |
| |
| |
| static void |
| module_free(void *m) |
| { |
| Py_CLEAR(current_tasks); |
| Py_CLEAR(all_tasks); |
| Py_CLEAR(traceback_extract_stack); |
| Py_CLEAR(asyncio_get_event_loop); |
| Py_CLEAR(asyncio_future_repr_info_func); |
| Py_CLEAR(asyncio_task_repr_info_func); |
| Py_CLEAR(asyncio_task_get_stack_func); |
| Py_CLEAR(asyncio_task_print_stack_func); |
| Py_CLEAR(asyncio_InvalidStateError); |
| Py_CLEAR(asyncio_CancelledError); |
| Py_CLEAR(inspect_isgenerator); |
| } |
| |
| static int |
| module_init(void) |
| { |
| PyObject *module = NULL; |
| PyObject *cls; |
| |
| #define WITH_MOD(NAME) \ |
| Py_CLEAR(module); \ |
| module = PyImport_ImportModule(NAME); \ |
| if (module == NULL) { \ |
| return -1; \ |
| } |
| |
| #define GET_MOD_ATTR(VAR, NAME) \ |
| VAR = PyObject_GetAttrString(module, NAME); \ |
| if (VAR == NULL) { \ |
| goto fail; \ |
| } |
| |
| WITH_MOD("asyncio.events") |
| GET_MOD_ATTR(asyncio_get_event_loop, "get_event_loop") |
| |
| WITH_MOD("asyncio.base_futures") |
| GET_MOD_ATTR(asyncio_future_repr_info_func, "_future_repr_info") |
| GET_MOD_ATTR(asyncio_InvalidStateError, "InvalidStateError") |
| GET_MOD_ATTR(asyncio_CancelledError, "CancelledError") |
| |
| WITH_MOD("asyncio.base_tasks") |
| GET_MOD_ATTR(asyncio_task_repr_info_func, "_task_repr_info") |
| GET_MOD_ATTR(asyncio_task_get_stack_func, "_task_get_stack") |
| GET_MOD_ATTR(asyncio_task_print_stack_func, "_task_print_stack") |
| |
| WITH_MOD("inspect") |
| GET_MOD_ATTR(inspect_isgenerator, "isgenerator") |
| |
| WITH_MOD("traceback") |
| GET_MOD_ATTR(traceback_extract_stack, "extract_stack") |
| |
| WITH_MOD("weakref") |
| GET_MOD_ATTR(cls, "WeakSet") |
| all_tasks = PyObject_CallObject(cls, NULL); |
| Py_CLEAR(cls); |
| if (all_tasks == NULL) { |
| goto fail; |
| } |
| |
| current_tasks = PyDict_New(); |
| if (current_tasks == NULL) { |
| goto fail; |
| } |
| |
| Py_CLEAR(module); |
| return 0; |
| |
| fail: |
| Py_CLEAR(module); |
| module_free(NULL); |
| return -1; |
| |
| #undef WITH_MOD |
| #undef GET_MOD_ATTR |
| } |
| |
| PyDoc_STRVAR(module_doc, "Accelerator module for asyncio"); |
| |
| static struct PyModuleDef _asynciomodule = { |
| PyModuleDef_HEAD_INIT, /* m_base */ |
| "_asyncio", /* m_name */ |
| module_doc, /* m_doc */ |
| -1, /* m_size */ |
| NULL, /* m_methods */ |
| NULL, /* m_slots */ |
| NULL, /* m_traverse */ |
| NULL, /* m_clear */ |
| (freefunc)module_free /* m_free */ |
| }; |
| |
| |
| PyMODINIT_FUNC |
| PyInit__asyncio(void) |
| { |
| if (module_init() < 0) { |
| return NULL; |
| } |
| if (PyType_Ready(&FutureType) < 0) { |
| return NULL; |
| } |
| if (PyType_Ready(&FutureIterType) < 0) { |
| return NULL; |
| } |
| if (PyType_Ready(&TaskSendMethWrapper_Type) < 0) { |
| return NULL; |
| } |
| if(PyType_Ready(&TaskWakeupMethWrapper_Type) < 0) { |
| return NULL; |
| } |
| if (PyType_Ready(&TaskType) < 0) { |
| return NULL; |
| } |
| |
| PyObject *m = PyModule_Create(&_asynciomodule); |
| if (m == NULL) { |
| return NULL; |
| } |
| |
| Py_INCREF(&FutureType); |
| if (PyModule_AddObject(m, "Future", (PyObject *)&FutureType) < 0) { |
| Py_DECREF(&FutureType); |
| return NULL; |
| } |
| |
| Py_INCREF(&TaskType); |
| if (PyModule_AddObject(m, "Task", (PyObject *)&TaskType) < 0) { |
| Py_DECREF(&TaskType); |
| return NULL; |
| } |
| |
| return m; |
| } |