in src/py_serde.cpp [74:121]
size_t py_object_serde::deserialize(const void* ptr, size_t capacity, nb::object* items, unsigned num) const {
size_t bytes_read = 0;
unsigned i = 0;
bool failure = false;
bool error_from_python = false;
nb::gil_scoped_acquire acquire;
// copy data into bytes only once
nb::bytes bytes(static_cast<const char*>(ptr), capacity);
for (; i < num && !failure; ++i) {
nb::tuple bytes_and_len;
try {
bytes_and_len = from_bytes(bytes, bytes_read);
} catch (nb::python_error &e) {
failure = true;
error_from_python = true;
break;
}
size_t length = nb::cast<size_t>(bytes_and_len[1]);
if (bytes_read + length > capacity) {
bytes_read += length; // use this value to report the error
failure = true;
break;
}
new (&items[i]) nb::object(nb::cast<nb::object>(bytes_and_len[0]));
ptr = static_cast<const char*>(ptr) + length;
bytes_read += length;
}
if (failure) {
// clean up what we've allocated
for (unsigned j = 0; j < i; ++j) {
items[j].dec_ref();
}
if (error_from_python) {
throw nb::value_error("Error reading value in from_bytes");
} else {
// this next call will throw
check_memory_size(bytes_read, capacity);
}
}
nb::gil_scoped_release release;
return bytes_read;
}