-
Notifications
You must be signed in to change notification settings - Fork 188
/
Copy pathserialize.jl
42 lines (36 loc) · 1.44 KB
/
serialize.jl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
using Serialization
import Serialization: serialize, deserialize
import Serialization: serialize_type
const _pickle = PyNULL()
pickle() = ispynull(_pickle) ? copy!(_pickle, pyimport(PyCall.pyversion.major ≥ 3 ? "pickle" : "cPickle")) : _pickle
function serialize(s::AbstractSerializer, pyo::PyObject)
serialize_type(s, PyObject)
if ispynull(pyo)
serialize(s, PyPtr_NULL)
else
b = PyBuffer(pycall(pickle()."dumps", PyObject, pyo))
serialize(s, unsafe_wrap(Array, Ptr{UInt8}(pointer(b)), sizeof(b)))
end
end
"""
pybytes(b::Union{String,DenseVector{UInt8}})
Convert `b` to a Python `bytes` object. This differs from the default
`PyObject(b)` conversion of `String` to a Python string (which may fail if `b`
does not contain valid Unicode), or from the default conversion of a
`Vector{UInt8}` to a `bytearray` object (which is mutable, unlike `bytes`).
"""
function pybytes(b::Union{String,DenseVector{UInt8}})
b isa String || stride(b,1) == 1 || throw(ArgumentError("pybytes requires stride-1 byte arrays"))
PyObject(@pycheckn ccall(@pysym(PyString_FromStringAndSize),
PyPtr, (Ptr{UInt8}, Int),
b, sizeof(b)))
end
function deserialize(s::AbstractSerializer, t::Type{PyObject})
b = deserialize(s)
if isa(b, PyPtr)
@assert b == C_NULL
return PyNULL()
else
return pycall(pickle()."loads", PyObject, pybytes(b))
end
end