Skip to content

Commit be8ccd7

Browse files
committed
Fixup decode(..., type=Struct) optimization
Between the original PR being made and merged a change went in that broke the optimization (specifically, the type code for struct types was split into `MS_TYPE_STRUCT` and `MS_TYPE_STRUCT_ARRAY`). This fixes the bug, and expands the tests to provide a more localized check for this behavior.
1 parent eb4f482 commit be8ccd7

File tree

3 files changed

+15
-9
lines changed

3 files changed

+15
-9
lines changed

msgspec/_core.c

+4-2
Original file line numberDiff line numberDiff line change
@@ -7068,11 +7068,12 @@ msgspec_msgpack_decode(PyObject *self, PyObject *const *args, Py_ssize_t nargs,
70687068
res = mpack_decode(&state, &type_any, NULL, false);
70697069
}
70707070
else {
7071+
bool array_like = ((StructMetaObject *)type)->array_like == OPT_TRUE;
70717072
struct {
70727073
uint32_t types;
70737074
Py_ssize_t fixtuple_size;
70747075
void* extra[1];
7075-
} type_obj = {MS_TYPE_STRUCT, 0, {type}};
7076+
} type_obj = {array_like ? MS_TYPE_STRUCT_ARRAY : MS_TYPE_STRUCT, 0, {type}};
70767077
res = mpack_decode(&state, (TypeNode*)(&type_obj), NULL, false);
70777078
}
70787079
PyBuffer_Release(&buffer);
@@ -9265,11 +9266,12 @@ msgspec_json_decode(PyObject *self, PyObject *const *args, Py_ssize_t nargs, PyO
92659266
res = json_decode(&state, &type_any, NULL);
92669267
}
92679268
else {
9269+
bool array_like = ((StructMetaObject *)type)->array_like == OPT_TRUE;
92689270
struct {
92699271
uint32_t types;
92709272
Py_ssize_t fixtuple_size;
92719273
void* extra[1];
9272-
} type_obj = {MS_TYPE_STRUCT, 0, {type}};
9274+
} type_obj = {array_like ? MS_TYPE_STRUCT_ARRAY : MS_TYPE_STRUCT, 0, {type}};
92739275
res = json_decode(&state, (TypeNode*)(&type_obj), NULL);
92749276
}
92759277

tests/test_json.py

+7-4
Original file line numberDiff line numberDiff line change
@@ -471,13 +471,16 @@ def test_decode_type_keyword(self):
471471
def test_decode_type_any(self):
472472
assert msgspec.json.decode(b"[1, 2, 3]", type=Any) == [1, 2, 3]
473473

474-
def test_decode_type_struct(self):
475-
class Point(msgspec.Struct):
474+
@pytest.mark.parametrize("array_like", [False, True])
475+
def test_decode_type_struct(self, array_like):
476+
class Point(msgspec.Struct, array_like=array_like):
476477
x: int
477478
y: int
478479

480+
msg = msgspec.json.encode(Point(1, 2))
481+
479482
for _ in range(2):
480-
assert msgspec.json.decode(b'{"x":1,"y":2}', type=Point) == Point(1, 2)
483+
assert msgspec.json.decode(msg, type=Point) == Point(1, 2)
481484

482485
def test_decode_type_struct_not_json_compatible(self):
483486
class Test(msgspec.Struct):
@@ -491,7 +494,7 @@ class Test(msgspec.Struct):
491494
x: 1
492495

493496
with pytest.raises(TypeError):
494-
msgspec.json.decode(b'{}', type=Test)
497+
msgspec.json.decode(b"{}", type=Test)
495498

496499
def test_decode_invalid_type(self):
497500
with pytest.raises(TypeError, match="Type '1' is not supported"):

tests/test_msgpack.py

+4-3
Original file line numberDiff line numberDiff line change
@@ -204,8 +204,9 @@ def test_decode_type_keyword(self):
204204
def test_decode_type_any(self):
205205
assert msgspec.msgpack.decode(self.buf, type=Any) == [1, 2, 3]
206206

207-
def test_decode_type_struct(self):
208-
class Point(msgspec.Struct):
207+
@pytest.mark.parametrize("array_like", [False, True])
208+
def test_decode_type_struct(self, array_like):
209+
class Point(msgspec.Struct, array_like=array_like):
209210
x: int
210211
y: int
211212

@@ -226,7 +227,7 @@ class Test(msgspec.Struct):
226227
x: 1
227228

228229
with pytest.raises(TypeError):
229-
msgspec.msgpack.decode(b'{}', type=Test)
230+
msgspec.msgpack.decode(b"{}", type=Test)
230231

231232
def test_decode_invalid_type(self):
232233
with pytest.raises(TypeError, match="Type '1' is not supported"):

0 commit comments

Comments
 (0)