Skip to content

Commit 759168a

Browse files
miss-islingtonrawwarserhiy-storchakavstinner
authored
[3.12] gh-111495: Add tests for PyList C API (GH-111562) (GH-111861)
(cherry picked from commit a3903c8) Signed-off-by: kalyanr <[email protected]> Co-authored-by: Kalyan <[email protected]> Co-authored-by: Serhiy Storchaka <[email protected]> Co-authored-by: Victor Stinner <[email protected]>
1 parent 992c3f6 commit 759168a

File tree

2 files changed

+456
-1
lines changed

2 files changed

+456
-1
lines changed

Lib/test/test_capi/test_list.py

+277
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,277 @@
1+
import unittest
2+
import sys
3+
from test.support import import_helper
4+
from collections import UserList
5+
_testcapi = import_helper.import_module('_testcapi')
6+
7+
NULL = None
8+
PY_SSIZE_T_MIN = _testcapi.PY_SSIZE_T_MIN
9+
PY_SSIZE_T_MAX = _testcapi.PY_SSIZE_T_MAX
10+
11+
class ListSubclass(list):
12+
pass
13+
14+
15+
class CAPITest(unittest.TestCase):
16+
def test_check(self):
17+
# Test PyList_Check()
18+
check = _testcapi.list_check
19+
self.assertTrue(check([1, 2]))
20+
self.assertTrue(check([]))
21+
self.assertTrue(check(ListSubclass([1, 2])))
22+
self.assertFalse(check({1: 2}))
23+
self.assertFalse(check((1, 2)))
24+
self.assertFalse(check(42))
25+
self.assertFalse(check(object()))
26+
27+
# CRASHES check(NULL)
28+
29+
30+
def test_list_check_exact(self):
31+
# Test PyList_CheckExact()
32+
check = _testcapi.list_check_exact
33+
self.assertTrue(check([1]))
34+
self.assertTrue(check([]))
35+
self.assertFalse(check(ListSubclass([1])))
36+
self.assertFalse(check(UserList([1, 2])))
37+
self.assertFalse(check({1: 2}))
38+
self.assertFalse(check(object()))
39+
40+
# CRASHES check(NULL)
41+
42+
def test_list_new(self):
43+
# Test PyList_New()
44+
list_new = _testcapi.list_new
45+
lst = list_new(0)
46+
self.assertEqual(lst, [])
47+
self.assertIs(type(lst), list)
48+
lst2 = list_new(0)
49+
self.assertIsNot(lst2, lst)
50+
self.assertRaises(SystemError, list_new, NULL)
51+
self.assertRaises(SystemError, list_new, -1)
52+
53+
def test_list_size(self):
54+
# Test PyList_Size()
55+
size = _testcapi.list_size
56+
self.assertEqual(size([1, 2]), 2)
57+
self.assertEqual(size(ListSubclass([1, 2])), 2)
58+
self.assertRaises(SystemError, size, UserList())
59+
self.assertRaises(SystemError, size, {})
60+
self.assertRaises(SystemError, size, 23)
61+
self.assertRaises(SystemError, size, object())
62+
# CRASHES size(NULL)
63+
64+
def test_list_get_size(self):
65+
# Test PyList_GET_SIZE()
66+
size = _testcapi.list_get_size
67+
self.assertEqual(size([1, 2]), 2)
68+
self.assertEqual(size(ListSubclass([1, 2])), 2)
69+
# CRASHES size(object())
70+
# CRASHES size(23)
71+
# CRASHES size({})
72+
# CRASHES size(UserList())
73+
# CRASHES size(NULL)
74+
75+
76+
def test_list_getitem(self):
77+
# Test PyList_GetItem()
78+
getitem = _testcapi.list_getitem
79+
lst = [1, 2, 3]
80+
self.assertEqual(getitem(lst, 0), 1)
81+
self.assertEqual(getitem(lst, 2), 3)
82+
self.assertRaises(IndexError, getitem, lst, 3)
83+
self.assertRaises(IndexError, getitem, lst, -1)
84+
self.assertRaises(IndexError, getitem, lst, PY_SSIZE_T_MIN)
85+
self.assertRaises(IndexError, getitem, lst, PY_SSIZE_T_MAX)
86+
self.assertRaises(SystemError, getitem, 42, 1)
87+
self.assertRaises(SystemError, getitem, (1, 2, 3), 1)
88+
self.assertRaises(SystemError, getitem, {1: 2}, 1)
89+
90+
# CRASHES getitem(NULL, 1)
91+
92+
def test_list_get_item(self):
93+
# Test PyList_GET_ITEM()
94+
get_item = _testcapi.list_get_item
95+
lst = [1, 2, [1, 2, 3]]
96+
self.assertEqual(get_item(lst, 0), 1)
97+
self.assertEqual(get_item(lst, 2), [1, 2, 3])
98+
99+
# CRASHES for out of index: get_item(lst, 3)
100+
# CRASHES for get_item(lst, PY_SSIZE_T_MIN)
101+
# CRASHES for get_item(lst, PY_SSIZE_T_MAX)
102+
# CRASHES get_item(21, 2)
103+
# CRASHES get_item(NULL, 1)
104+
105+
106+
def test_list_setitem(self):
107+
# Test PyList_SetItem()
108+
setitem = _testcapi.list_setitem
109+
lst = [1, 2, 3]
110+
setitem(lst, 0, 10)
111+
self.assertEqual(lst, [10, 2, 3])
112+
setitem(lst, 2, 12)
113+
self.assertEqual(lst, [10, 2, 12])
114+
self.assertRaises(IndexError, setitem, lst, 3 , 5)
115+
self.assertRaises(IndexError, setitem, lst, -1, 5)
116+
self.assertRaises(IndexError, setitem, lst, PY_SSIZE_T_MIN, 5)
117+
self.assertRaises(IndexError, setitem, lst, PY_SSIZE_T_MAX, 5)
118+
self.assertRaises(SystemError, setitem, (1, 2, 3), 1, 5)
119+
self.assertRaises(SystemError, setitem, {1: 2}, 1, 5)
120+
121+
# CRASHES setitem(NULL, 'a', 5)
122+
123+
def test_list_set_item(self):
124+
# Test PyList_SET_ITEM()
125+
set_item = _testcapi.list_set_item
126+
lst = [1, 2, 3]
127+
set_item(lst, 1, 10)
128+
set_item(lst, 2, [1, 2, 3])
129+
self.assertEqual(lst, [1, 10, [1, 2, 3]])
130+
131+
# CRASHES for set_item([1], -1, 5)
132+
# CRASHES for set_item([1], PY_SSIZE_T_MIN, 5)
133+
# CRASHES for set_item([1], PY_SSIZE_T_MAX, 5)
134+
# CRASHES for set_item([], 0, 1)
135+
# CRASHES for set_item(NULL, 0, 1)
136+
137+
138+
def test_list_insert(self):
139+
# Test PyList_Insert()
140+
insert = _testcapi.list_insert
141+
lst = [1, 2, 3]
142+
insert(lst, 0, 23)
143+
self.assertEqual(lst, [23, 1, 2, 3])
144+
insert(lst, -1, 22)
145+
self.assertEqual(lst, [23, 1, 2, 22, 3])
146+
insert(lst, PY_SSIZE_T_MIN, 1)
147+
self.assertEqual(lst[0], 1)
148+
insert(lst, len(lst), 123)
149+
self.assertEqual(lst[-1], 123)
150+
insert(lst, len(lst)-1, 124)
151+
self.assertEqual(lst[-2], 124)
152+
insert(lst, PY_SSIZE_T_MAX, 223)
153+
self.assertEqual(lst[-1], 223)
154+
155+
self.assertRaises(SystemError, insert, (1, 2, 3), 1, 5)
156+
self.assertRaises(SystemError, insert, {1: 2}, 1, 5)
157+
158+
# CRASHES insert(NULL, 1, 5)
159+
160+
def test_list_append(self):
161+
# Test PyList_Append()
162+
append = _testcapi.list_append
163+
lst = [1, 2, 3]
164+
append(lst, 10)
165+
self.assertEqual(lst, [1, 2, 3, 10])
166+
append(lst, [4, 5])
167+
self.assertEqual(lst, [1, 2, 3, 10, [4, 5]])
168+
self.assertRaises(SystemError, append, lst, NULL)
169+
self.assertRaises(SystemError, append, (), 0)
170+
self.assertRaises(SystemError, append, 42, 0)
171+
# CRASHES append(NULL, 0)
172+
173+
def test_list_getslice(self):
174+
# Test PyList_GetSlice()
175+
getslice = _testcapi.list_getslice
176+
lst = [1, 2, 3]
177+
178+
# empty
179+
self.assertEqual(getslice(lst, PY_SSIZE_T_MIN, 0), [])
180+
self.assertEqual(getslice(lst, -1, 0), [])
181+
self.assertEqual(getslice(lst, 3, PY_SSIZE_T_MAX), [])
182+
183+
# slice
184+
self.assertEqual(getslice(lst, 1, 3), [2, 3])
185+
186+
# whole
187+
self.assertEqual(getslice(lst, 0, len(lst)), lst)
188+
self.assertEqual(getslice(lst, 0, 100), lst)
189+
self.assertEqual(getslice(lst, -100, 100), lst)
190+
191+
self.assertRaises(SystemError, getslice, (1, 2, 3), 0, 0)
192+
self.assertRaises(SystemError, getslice, 'abc', 0, 0)
193+
self.assertRaises(SystemError, getslice, 42, 0, 0)
194+
195+
# CRASHES getslice(NULL, 0, 0)
196+
197+
def test_list_setslice(self):
198+
# Test PyList_SetSlice()
199+
setslice = _testcapi.list_setslice
200+
def set_slice(lst, low, high, value):
201+
lst = lst.copy()
202+
self.assertEqual(setslice(lst, low, high, value), 0)
203+
return lst
204+
205+
# insert items
206+
self.assertEqual(set_slice([], 0, 0, list("abc")), list("abc"))
207+
self.assertEqual(set_slice([], PY_SSIZE_T_MIN, PY_SSIZE_T_MIN, list("abc")), list("abc"))
208+
self.assertEqual(set_slice([], PY_SSIZE_T_MAX, PY_SSIZE_T_MAX, list("abc")), list("abc"))
209+
lst = list("abc")
210+
self.assertEqual(set_slice(lst, 0, 0, ["X"]), list("Xabc"))
211+
self.assertEqual(set_slice(lst, 1, 1, list("XY")), list("aXYbc"))
212+
self.assertEqual(set_slice(lst, len(lst), len(lst), ["X"]), list("abcX"))
213+
# self.assertEqual(set_slice(lst, PY_SSIZE_T_MAX, PY_SSIZE_T_MAX, ["X"]), list("abcX"))
214+
215+
# replace items
216+
lst = list("abc")
217+
self.assertEqual(set_slice(lst, -100, 1, list("X")), list("Xbc"))
218+
self.assertEqual(set_slice(lst, 1, 2, list("X")), list("aXc"))
219+
self.assertEqual(set_slice(lst, 1, 3, list("XY")), list("aXY"))
220+
self.assertEqual(set_slice(lst, 0, 3, list("XYZ")), list("XYZ"))
221+
222+
# delete items
223+
lst = list("abcdef")
224+
self.assertEqual(set_slice(lst, 0, len(lst), []), [])
225+
self.assertEqual(set_slice(lst, -100, 100, []), [])
226+
self.assertEqual(set_slice(lst, 1, 5, []), list("af"))
227+
self.assertEqual(set_slice(lst, 3, len(lst), []), list("abc"))
228+
229+
# delete items with NULL
230+
lst = list("abcdef")
231+
self.assertEqual(set_slice(lst, 0, len(lst), NULL), [])
232+
self.assertEqual(set_slice(lst, 3, len(lst), NULL), list("abc"))
233+
234+
self.assertRaises(SystemError, setslice, (), 0, 0, [])
235+
self.assertRaises(SystemError, setslice, 42, 0, 0, [])
236+
237+
# CRASHES setslice(NULL, 0, 0, [])
238+
239+
def test_list_sort(self):
240+
# Test PyList_Sort()
241+
sort = _testcapi.list_sort
242+
lst = [4, 6, 7, 3, 1, 5, 9, 2, 0, 8]
243+
sort(lst)
244+
self.assertEqual(lst, list(range(10)))
245+
246+
lst2 = ListSubclass([4, 6, 7, 3, 1, 5, 9, 2, 0, 8])
247+
sort(lst2)
248+
self.assertEqual(lst2, list(range(10)))
249+
250+
self.assertRaises(SystemError, sort, ())
251+
self.assertRaises(SystemError, sort, object())
252+
self.assertRaises(SystemError, sort, NULL)
253+
254+
255+
def test_list_reverse(self):
256+
# Test PyList_Reverse()
257+
reverse = _testcapi.list_reverse
258+
def list_reverse(lst):
259+
self.assertEqual(reverse(lst), 0)
260+
return lst
261+
262+
self.assertEqual(list_reverse([]), [])
263+
self.assertEqual(list_reverse([2, 5, 10]), [10, 5, 2])
264+
265+
self.assertRaises(SystemError, reverse, ())
266+
self.assertRaises(SystemError, reverse, object())
267+
self.assertRaises(SystemError, reverse, NULL)
268+
269+
def test_list_astuple(self):
270+
# Test PyList_AsTuple()
271+
astuple = _testcapi.list_astuple
272+
self.assertEqual(astuple([]), ())
273+
self.assertEqual(astuple([2, 5, 10]), (2, 5, 10))
274+
275+
self.assertRaises(SystemError, astuple, ())
276+
self.assertRaises(SystemError, astuple, object())
277+
self.assertRaises(SystemError, astuple, NULL)

0 commit comments

Comments
 (0)