Skip to content

Commit

Permalink
pythongh-111138: Add test_capi.list tests
Browse files Browse the repository at this point in the history
Test the public PyList C API.
  • Loading branch information
vstinner committed Nov 1, 2023
1 parent f6a0232 commit d0356f2
Show file tree
Hide file tree
Showing 7 changed files with 412 additions and 1 deletion.
226 changes: 226 additions & 0 deletions Lib/test/test_capi/test_list.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
import unittest
from collections import UserList
import _testcapi


NULL = None
PY_SSIZE_T_MIN = _testcapi.PY_SSIZE_T_MIN
PY_SSIZE_T_MAX = _testcapi.PY_SSIZE_T_MAX

PyList_Check = _testcapi.list_check
PyList_CheckExact = _testcapi.list_checkexact
PyList_Size = _testcapi.list_size
PyList_GetItem = _testcapi.list_getitem
PyList_SetItem = _testcapi.list_setitem
PyList_Insert = _testcapi.list_insert
PyList_Append = _testcapi.list_append
PyList_Sort = _testcapi.list_sort
PyList_AsTuple = _testcapi.list_astuple
PyList_Reverse = _testcapi.list_reverse
PyList_GetSlice = _testcapi.list_getslice
PyList_SetSlice = _testcapi.list_setslice


class ListSubclass(list):
pass


class CAPITest(unittest.TestCase):

def test_not_list_objects(self):
for obj in (
123,
UserList([2, 3, 5]),
(2, 3, 5), # tuple
object(),
):
with self.subTest(obj=obj):
self.assertFalse(PyList_Check(obj))
self.assertFalse(PyList_CheckExact(obj))
with self.assertRaises(SystemError):
PyList_Size(obj)
with self.assertRaises(SystemError):
PyList_SetItem(obj, 0, "x")
with self.assertRaises(SystemError):
PyList_Insert(obj, 0, "x")
with self.assertRaises(SystemError):
PyList_Append(obj, "x")
with self.assertRaises(SystemError):
PyList_Sort(obj)
with self.assertRaises(SystemError):
PyList_AsTuple(obj)
with self.assertRaises(SystemError):
PyList_Reverse(obj)
with self.assertRaises(SystemError):
PyList_GetSlice(obj, 0, 1)
with self.assertRaises(SystemError):
PyList_SetSlice(obj, 0, 1, ["x"])

def test_list_check(self):
self.assertTrue(PyList_Check([2, 3, 5]))
#self.assertFalse(PyList_Check(NULL))

def test_list_checkexact(self):
self.assertTrue(PyList_CheckExact([2, 3, 5]))
#self.assertFalse(check(NULL))

def test_list_new(self):
expected = [None, None, None]

PyList_New = _testcapi.list_new
lst = PyList_New(3)
self.assertEqual(lst, expected)
self.assertIs(type(lst), list)
lst2 = PyList_New(3)
self.assertIsNot(lst2, lst)

def test_list_size(self):
self.assertEqual(PyList_Size([2, 3, 5]), 3)

def test_list_getitem(self):
lst = list("abc")
self.assertEqual(PyList_GetItem(lst, 0), "a")
self.assertEqual(PyList_GetItem(lst, 1), "b")
self.assertEqual(PyList_GetItem(lst, 2), "c")

for invalid_index in (PY_SSIZE_T_MIN, -1, 3, PY_SSIZE_T_MAX):
with self.subTest(invalid_index=invalid_index):
self.assertRaises(IndexError, PyList_GetItem, lst, invalid_index)

lst2 = ListSubclass(lst)
self.assertEqual(PyList_GetItem(lst2, 1), "b")
self.assertRaises(IndexError, PyList_GetItem, lst2, 3)

# CRASHES PyList_GetItem(NULL, 0)

def test_list_setitem(self):
lst = list("abc")
PyList_SetItem(lst, 1, "X")
self.assertEqual(lst, ["a", "X", "c"])
self.assertRaises(IndexError, PyList_SetItem, lst, 3, "Y")

lst2 = ListSubclass(list("abc"))
PyList_SetItem(lst2, 1, "X")
self.assertEqual(lst2, ["a", "X", "c"])

# CRASHES PyList_SetItem(list("abc"), 0, NULL)
# CRASHES PyList_SetItem(NULL, 0, 'x')

def test_list_insert(self):
lst = list("abc")
PyList_Insert(lst, 1, "X")
self.assertEqual(lst, ["a", "X", "b", "c"])
PyList_Insert(lst, -1, "Y")
self.assertEqual(lst, ["a", "X", "b", "Y", "c"])
PyList_Insert(lst, 100, "Z")
self.assertEqual(lst, ["a", "X", "b", "Y", "c", "Z"])
PyList_Insert(lst, -100, "0")
self.assertEqual(lst, ["0", "a", "X", "b", "Y", "c", "Z"])

lst2 = ListSubclass(list("abc"))
PyList_Insert(lst2, 1, "X")
self.assertEqual(lst2, ["a", "X", "b", "c"])

with self.assertRaises(SystemError):
PyList_Insert(list("abc"), 0, NULL)

# CRASHES PyList_Insert(NULL, 0, 'x')

def test_list_append(self):

lst = []
PyList_Append(lst, "a")
self.assertEqual(lst, ["a"])
PyList_Append(lst, "b")
self.assertEqual(lst, ["a", "b"])

lst2 = ListSubclass()
PyList_Append(lst2, "X")
self.assertEqual(lst2, ["X"])

self.assertRaises(SystemError, PyList_Append, [], NULL)

# CRASHES PyList_Append(NULL, 'a')

def test_list_sort(self):
lst = [4, 6, 7, 3, 1, 5, 9, 2, 0, 8]
PyList_Sort(lst)
self.assertEqual(lst, list(range(10)))

lst2 = ListSubclass([4, 6, 7, 3, 1, 5, 9, 2, 0, 8])
PyList_Sort(lst2)
self.assertEqual(lst2, list(range(10)))

with self.assertRaises(SystemError):
PyList_Sort(NULL)

def test_list_astuple(self):
self.assertEqual(PyList_AsTuple([]), ())
self.assertEqual(PyList_AsTuple([2, 5, 10]), (2, 5, 10))

with self.assertRaises(SystemError):
PyList_AsTuple(NULL)

def test_list_reverse(self):
def list_reverse(lst):
self.assertEqual(PyList_Reverse(lst), 0)
return lst

self.assertEqual(list_reverse([]), [])
self.assertEqual(list_reverse([2, 5, 10]), [10, 5, 2])

with self.assertRaises(SystemError):
PyList_Reverse(NULL)

def test_list_getslice(self):
lst = list("abcdef")
self.assertEqual(PyList_GetSlice(lst, -100, 0), [])
self.assertEqual(PyList_GetSlice(lst, 0, 0), [])

self.assertEqual(PyList_GetSlice(lst, 1, 3), list("bc"))
self.assertEqual(PyList_GetSlice(lst, 0, len(lst)), lst)
self.assertEqual(PyList_GetSlice(lst, 0, 100), lst)
self.assertEqual(PyList_GetSlice(lst, -100, 100), lst)

self.assertEqual(PyList_GetSlice(lst, 100, 0), [])

# CRASHES PyList_GetSlice(NULL, 0, 0)

def test_list_setslice(self):
def set_slice(lst, low, high, value):
lst = lst.copy()
self.assertEqual(PyList_SetSlice(lst, low, high, value), 0)
return lst

# insert items
self.assertEqual(set_slice([], 0, 0, list("abc")), list("abc"))
lst = list("abc")
self.assertEqual(set_slice(lst, 0, 0, ["X"]), list("Xabc"))
self.assertEqual(set_slice(lst, 1, 1, list("XY")), list("aXYbc"))
self.assertEqual(set_slice(lst, len(lst), len(lst), ["X"]), list("abcX"))
self.assertEqual(set_slice(lst, 100, 100, ["X"]), list("abcX"))

# replace items
lst = list("abc")
self.assertEqual(set_slice(lst, -100, 1, list("X")), list("Xbc"))
self.assertEqual(set_slice(lst, 1, 2, list("X")), list("aXc"))
self.assertEqual(set_slice(lst, 1, 3, list("XY")), list("aXY"))
self.assertEqual(set_slice(lst, 0, 3, list("XYZ")), list("XYZ"))

# delete items
lst = list("abcdef")
self.assertEqual(set_slice(lst, 0, len(lst), []), [])
self.assertEqual(set_slice(lst, -100, 100, []), [])
self.assertEqual(set_slice(lst, 1, 5, []), list("af"))
self.assertEqual(set_slice(lst, 3, len(lst), []), list("abc"))

# delete items with NULL
lst = list("abcdef")
self.assertEqual(set_slice(lst, 0, len(lst), NULL), [])
self.assertEqual(set_slice(lst, 3, len(lst), NULL), list("abc"))

# CRASHES PyList_SetSlice(NULL, 0, 0, ["x"])


if __name__ == "__main__":
unittest.main()
2 changes: 1 addition & 1 deletion Modules/Setup.stdlib.in
Original file line number Diff line number Diff line change
Expand Up @@ -159,7 +159,7 @@
@MODULE__XXTESTFUZZ_TRUE@_xxtestfuzz _xxtestfuzz/_xxtestfuzz.c _xxtestfuzz/fuzzer.c
@MODULE__TESTBUFFER_TRUE@_testbuffer _testbuffer.c
@MODULE__TESTINTERNALCAPI_TRUE@_testinternalcapi _testinternalcapi.c _testinternalcapi/test_lock.c _testinternalcapi/pytime.c _testinternalcapi/set.c
@MODULE__TESTCAPI_TRUE@_testcapi _testcapimodule.c _testcapi/vectorcall.c _testcapi/vectorcall_limited.c _testcapi/heaptype.c _testcapi/abstract.c _testcapi/unicode.c _testcapi/dict.c _testcapi/set.c _testcapi/getargs.c _testcapi/datetime.c _testcapi/docstring.c _testcapi/mem.c _testcapi/watchers.c _testcapi/long.c _testcapi/float.c _testcapi/structmember.c _testcapi/exceptions.c _testcapi/code.c _testcapi/buffer.c _testcapi/pyatomic.c _testcapi/pyos.c _testcapi/immortal.c _testcapi/heaptype_relative.c _testcapi/gc.c _testcapi/sys.c
@MODULE__TESTCAPI_TRUE@_testcapi _testcapimodule.c _testcapi/vectorcall.c _testcapi/vectorcall_limited.c _testcapi/heaptype.c _testcapi/abstract.c _testcapi/unicode.c _testcapi/dict.c _testcapi/list.c _testcapi/set.c _testcapi/getargs.c _testcapi/datetime.c _testcapi/docstring.c _testcapi/mem.c _testcapi/watchers.c _testcapi/long.c _testcapi/float.c _testcapi/structmember.c _testcapi/exceptions.c _testcapi/code.c _testcapi/buffer.c _testcapi/pyatomic.c _testcapi/pyos.c _testcapi/immortal.c _testcapi/heaptype_relative.c _testcapi/gc.c _testcapi/sys.c
@MODULE__TESTCLINIC_TRUE@_testclinic _testclinic.c
@MODULE__TESTCLINIC_LIMITED_TRUE@_testclinic_limited _testclinic_limited.c

Expand Down
Loading

0 comments on commit d0356f2

Please sign in to comment.