From 279daf1feecfbf4ca6c4ca51250e373c16989905 Mon Sep 17 00:00:00 2001 From: Joannah Nanjekye Date: Tue, 29 Nov 2022 13:59:58 -0500 Subject: [PATCH 1/2] Warn for specific thread module methods --- Lib/test/test_thread.py | 46 ++++++++++++++++++++++++++++++++++++++ Lib/test/test_threading.py | 17 +++++++++++++- Lib/threading.py | 16 +++++++++++++ Modules/threadmodule.c | 32 ++++++++++++++++++++++++++ 4 files changed, 110 insertions(+), 1 deletion(-) diff --git a/Lib/test/test_thread.py b/Lib/test/test_thread.py index 93690a60b2ff58..dce09ad83a1ad9 100644 --- a/Lib/test/test_thread.py +++ b/Lib/test/test_thread.py @@ -2,6 +2,7 @@ import unittest import random from test import support +from test.test_support import check_py3k_warnings thread = support.import_module('thread') import time import sys @@ -157,6 +158,51 @@ def mywrite(self, *args): started.acquire() self.assertIn("Traceback", stderr.getvalue()) + def test_py3k_thread_module(self): + expected = "In 3.x, the thread module is removed: use the threading module instead" + with check_py3k_warnings() as w: + import thread + + def test_py3k_thread_module_get_ident(self): + expected = "thread.get_ident is removed in 3.x: use the threading.get_ident instead" + with check_py3k_warnings() as w: + thread.get_ident() + + def test_py3k_thread_module_start_new_thread(self): + expected = "thread.start_new_thread is removed in 3.x: use the threading._start_new_thread instead" + with check_py3k_warnings() as w: + def f(): + ident.append(threading.currentThread().ident) + done.set() + thread.start_new_thread((f), ()) + + def test_py3k_thread_module_allocate(self): + expected = "thread.allocate_lock is removed in 3.x: use the threading._allocate_lock instead" + with check_py3k_warnings() as w: + thread.allocate_lock() + + def test_py3k_thread_module_exit_thread(self): + expected = "thread.exit is removed in 3.x: no equivalent method exists, raising SystemExit will exit a thread" + with check_py3k_warnings() as w: + with self.assertRaises(SystemExit): + thread.exit_thread() + + def test_py3k_thread_module_interrupt_main(self): + expected = "thread.interrupt_main is removed in 3.x: no equivalent method exists, raising KeyboardInterrupt will interruot the main thread" + with check_py3k_warnings() as w: + with self.assertRaises(KeyboardInterrupt): + thread.interrupt_main() + + def test_py3k_thread_module_count(self): + expected = "thread._count is removed in 3.x: use the threading.count instead" + with check_py3k_warnings() as w: + thread._count() + + def test_py3k_thread_module_stack_size(self): + expected = "thread.stack_size is removed in 3.x: use threading.stack_size instead" + with check_py3k_warnings() as w: + thread.stack_size() + class Barrier: def __init__(self, num_threads): diff --git a/Lib/test/test_threading.py b/Lib/test/test_threading.py index 4c21e6baa76cbe..e0a4226a66ff60 100644 --- a/Lib/test/test_threading.py +++ b/Lib/test/test_threading.py @@ -1,7 +1,7 @@ # Very rudimentary test of threading module import test.test_support -from test.test_support import verbose, cpython_only +from test.test_support import verbose, cpython_only, check_py3k_warnings from test.script_helper import assert_python_ok import random @@ -480,6 +480,21 @@ def test_BoundedSemaphore_limit(self): t.join() self.assertRaises(ValueError, bs.release) + def test_threading_module_method_rename(self): + import threading + expected = "_get_ident is removed in 3.x: use get_ident instead" + with check_py3k_warnings() as w: + threading. _get_ident() + expected = "_start_new_thread is removed in 3.x: use start_new_thread instead" + with check_py3k_warnings() as w: + def f(): + ident.append(threading.currentThread().ident) + done.set() + threading._start_new_thread((f), ()) + expected = "_allocate_lock is removed in 3.x: use allocate_lock instead" + with check_py3k_warnings() as w: + threading._allocate_lock() + class ThreadJoinOnShutdown(BaseTestCase): # Between fork() and exec(), only async-safe functions are allowed (issues diff --git a/Lib/threading.py b/Lib/threading.py index b08374a9639c4e..0440a1e1fce638 100644 --- a/Lib/threading.py +++ b/Lib/threading.py @@ -35,6 +35,22 @@ _start_new_thread = thread.start_new_thread _allocate_lock = thread.allocate_lock _get_ident = thread.get_ident + +# def _get_ident(): +# warnings.warnpy3k_with_fix("_get_ident() is renamed in 3.x", "use get_ident() instead", +# stacklevel=2) +# thread.get_ident + +# def _start_new_thread(): +# warnings.warnpy3k_with_fix("_start_new_thread() is removed in 3.x", "use start_new_thread() instead", +# stacklevel=2) +# thread.start_new_thread + +# def _allocate_lock(): +# warnings.warnpy3k_with_fix("_allocate_lock() is removed in 3.x", "use allocate_lock instead", +# stacklevel=2) +# thread.allocate_lock + ThreadError = thread.error del thread diff --git a/Modules/threadmodule.c b/Modules/threadmodule.c index 82eff0072b3903..bf71852b8179b2 100644 --- a/Modules/threadmodule.c +++ b/Modules/threadmodule.c @@ -657,6 +657,10 @@ thread_PyThread_start_new_thread(PyObject *self, PyObject *fargs) struct bootstate *boot; long ident; + if (PyErr_WarnPy3k_WithFix("thread.start_new_thread is removed in 3.x", + "use the threading._start_new_thread instead", 1)) + return NULL; + if (!PyArg_UnpackTuple(fargs, "start_new_thread", 2, 3, &func, &args, &keyw)) return NULL; @@ -718,6 +722,10 @@ printed unless the exception is SystemExit.\n"); static PyObject * thread_PyThread_exit_thread(PyObject *self) { + if (PyErr_WarnPy3k_WithFix("thread.exit is removed in 3.x", + "no equivalent method exists, raising SystemExit will exit a thread", 1)) + return NULL; + PyErr_SetNone(PyExc_SystemExit); return NULL; } @@ -732,6 +740,10 @@ thread to exit silently unless the exception is caught."); static PyObject * thread_PyThread_interrupt_main(PyObject * self) { + if (PyErr_WarnPy3k_WithFix("thread.interrupt_main is removed in 3.x", + "no equivalent method exists, raising KeyboardInterrupt will interruot the main thread", 1)) + return NULL; + PyErr_SetInterrupt(); Py_INCREF(Py_None); return Py_None; @@ -749,6 +761,10 @@ static lockobject *newlockobject(void); static PyObject * thread_PyThread_allocate_lock(PyObject *self) { + if (PyErr_WarnPy3k_WithFix("thread.allocate_lock is removed in 3.x", + "use the threading._allocate_lock instead", 1)) + return NULL; + return (PyObject *) newlockobject(); } @@ -762,6 +778,10 @@ static PyObject * thread_get_ident(PyObject *self) { long ident; + if (PyErr_WarnPy3k_WithFix("thread.get_ident is removed in 3.x", + "use the threading.get_ident instead", 1)) + return NULL; + ident = PyThread_get_thread_ident(); if (ident == -1) { PyErr_SetString(ThreadError, "no current thread ident"); @@ -784,6 +804,10 @@ A thread's identity may be reused for another thread after it exits."); static PyObject * thread__count(PyObject *self) { + if (PyErr_WarnPy3k_WithFix("thread.count is removed in 3.x", + "use the threading._count instead", 1)) + return NULL; + return PyInt_FromLong(nb_threads); } @@ -806,6 +830,10 @@ thread_stack_size(PyObject *self, PyObject *args) Py_ssize_t new_size = 0; int rc; + if (PyErr_WarnPy3k_WithFix("thread.stack_size is removed in 3.x", + "use the threading.stack_size instead", 1)) + return NULL; + if (!PyArg_ParseTuple(args, "|n:stack_size", &new_size)) return NULL; @@ -904,6 +932,10 @@ initthread(void) { PyObject *m, *d; + if (PyErr_WarnPy3k_WithFix("In 3.x, the thread module is removed", + "use the threading module instead", 1)) + return; + /* Initialize types: */ if (PyType_Ready(&localdummytype) < 0) return; From 96a83708947837655affad498573cac06e15b9a3 Mon Sep 17 00:00:00 2001 From: Joannah Nanjekye Date: Tue, 29 Nov 2022 14:22:00 -0500 Subject: [PATCH 2/2] fix threading --- Lib/threading.py | 31 +++++++++++++++---------------- 1 file changed, 15 insertions(+), 16 deletions(-) diff --git a/Lib/threading.py b/Lib/threading.py index 0440a1e1fce638..08144314df4128 100644 --- a/Lib/threading.py +++ b/Lib/threading.py @@ -32,25 +32,24 @@ 'Lock', 'RLock', 'Semaphore', 'BoundedSemaphore', 'Thread', 'Timer', 'setprofile', 'settrace', 'local', 'stack_size'] -_start_new_thread = thread.start_new_thread -_allocate_lock = thread.allocate_lock -_get_ident = thread.get_ident - -# def _get_ident(): -# warnings.warnpy3k_with_fix("_get_ident() is renamed in 3.x", "use get_ident() instead", -# stacklevel=2) -# thread.get_ident +def _get_ident(): + warnings.warnpy3k_with_fix("_get_ident() is renamed in 3.x", "use get_ident() instead", + stacklevel=2) + thread.get_ident -# def _start_new_thread(): -# warnings.warnpy3k_with_fix("_start_new_thread() is removed in 3.x", "use start_new_thread() instead", -# stacklevel=2) -# thread.start_new_thread +def _start_new_thread(): + warnings.warnpy3k_with_fix("_start_new_thread() is removed in 3.x", "use start_new_thread() instead", + stacklevel=2) + thread.start_new_thread -# def _allocate_lock(): -# warnings.warnpy3k_with_fix("_allocate_lock() is removed in 3.x", "use allocate_lock instead", -# stacklevel=2) -# thread.allocate_lock +def _allocate_lock(): + warnings.warnpy3k_with_fix("_allocate_lock() is removed in 3.x", "use allocate_lock instead", + stacklevel=2) + thread.allocate_lock +_start_new_thread = thread.start_new_thread +_allocate_lock = thread.allocate_lock +_get_ident = thread.get_ident ThreadError = thread.error del thread