mirror of
https://github.com/msgpack/msgpack-c.git
synced 2025-10-24 09:12:30 +02:00
Compare commits
702 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
076bd07f39 | ||
![]() |
00e897503b | ||
![]() |
2f3c47f385 | ||
![]() |
2b28211701 | ||
![]() |
46c794fc2e | ||
![]() |
0408738b48 | ||
![]() |
eb8a338a0e | ||
![]() |
23480bfe8a | ||
![]() |
3384dbc92d | ||
![]() |
8446549ed8 | ||
![]() |
6a9f3ae71d | ||
![]() |
c58ce1a975 | ||
![]() |
c6d9bbd7b0 | ||
![]() |
65ddd1a455 | ||
![]() |
58c0fe0f91 | ||
![]() |
b96b62f2ac | ||
![]() |
43fc0a52a6 | ||
![]() |
4f3d9a1ded | ||
![]() |
b50ff920f0 | ||
![]() |
983ae0c5a2 | ||
![]() |
9d4db84987 | ||
![]() |
dfaa281476 | ||
![]() |
c1c25c455c | ||
![]() |
e65197f386 | ||
![]() |
ecbb8f8711 | ||
![]() |
03aa736199 | ||
![]() |
821b1f0384 | ||
![]() |
233ad3aaff | ||
![]() |
9c4812f4d4 | ||
![]() |
337cc9fa78 | ||
![]() |
dfe29a0329 | ||
![]() |
85e90aa81e | ||
![]() |
86d30fe571 | ||
![]() |
173302267f | ||
![]() |
8e3ea8d26c | ||
![]() |
cb065563c2 | ||
![]() |
27b89b237b | ||
![]() |
21f0d0bfc4 | ||
![]() |
d5e583b09e | ||
![]() |
15fb9bbcb2 | ||
![]() |
a03418ab12 | ||
![]() |
035513844d | ||
![]() |
193906b8ce | ||
![]() |
d354f2e8c2 | ||
![]() |
24a8ee436f | ||
![]() |
6e5f9404b7 | ||
![]() |
64fe90aabb | ||
![]() |
7e31d487e0 | ||
![]() |
ee16e2eb30 | ||
![]() |
003fb3392f | ||
![]() |
0e82278cd5 | ||
![]() |
5b544248bc | ||
![]() |
d4d7495c0a | ||
![]() |
8325cce44c | ||
![]() |
ee0debc5b3 | ||
![]() |
8f1bd12a17 | ||
![]() |
fccf1016e4 | ||
![]() |
1b010a082b | ||
![]() |
324f215316 | ||
![]() |
ace4f1e7f0 | ||
![]() |
5c5f16f148 | ||
![]() |
f50855d5c0 | ||
![]() |
86917b0ba7 | ||
![]() |
7201fcbe0f | ||
![]() |
83f1735fbb | ||
![]() |
80fd8e70f0 | ||
![]() |
698c1c2b7a | ||
![]() |
5d3287f5aa | ||
![]() |
6990fe6f51 | ||
![]() |
0b81068368 | ||
![]() |
5b1851ae22 | ||
![]() |
0d5708a01d | ||
![]() |
67ab510b5d | ||
![]() |
d1264a1289 | ||
![]() |
3b973021a1 | ||
![]() |
182624895f | ||
![]() |
548de3739c | ||
![]() |
718a3efd64 | ||
![]() |
76b8c3250b | ||
![]() |
dafaa8bd8c | ||
![]() |
9c9b2c25ea | ||
![]() |
e89cd81022 | ||
![]() |
5debbd2be8 | ||
![]() |
fe26df5355 | ||
![]() |
85778494e4 | ||
![]() |
a94ce3c715 | ||
![]() |
a440ff1117 | ||
![]() |
8195137bc5 | ||
![]() |
3e2bd25e4e | ||
![]() |
ffd0c2f624 | ||
![]() |
23333c98d7 | ||
![]() |
5a12ab98c4 | ||
![]() |
0b327a63fc | ||
![]() |
15b28a245c | ||
![]() |
6ef5684797 | ||
![]() |
d439b1495b | ||
![]() |
9e096a3f0e | ||
![]() |
0acf6ec150 | ||
![]() |
4b36340474 | ||
![]() |
910e642a8b | ||
![]() |
9f571146fb | ||
![]() |
339725f73d | ||
![]() |
aff964c58b | ||
![]() |
0c07e745f8 | ||
![]() |
419d2e9564 | ||
![]() |
f936a307c6 | ||
![]() |
4e4678edfa | ||
![]() |
cb7a4b3116 | ||
![]() |
5ca4c42a74 | ||
![]() |
1238a7ca07 | ||
![]() |
a9db60a73d | ||
![]() |
bd32ac19d4 | ||
![]() |
eaf9944e43 | ||
![]() |
703fdbc01d | ||
![]() |
2daa08b0e7 | ||
![]() |
05d9d22d9e | ||
![]() |
8b7894f9bd | ||
![]() |
a71439607f | ||
![]() |
aca0d7f969 | ||
![]() |
310a8e4342 | ||
![]() |
24fbe1ef5b | ||
![]() |
33d8faa35d | ||
![]() |
44be714f65 | ||
![]() |
0df3da6b10 | ||
![]() |
79197b6ec7 | ||
![]() |
353b6b51cb | ||
![]() |
bd9a2c0d3a | ||
![]() |
53b0ee6536 | ||
![]() |
78daac0f1b | ||
![]() |
b970b9b9a8 | ||
![]() |
0339db57f1 | ||
![]() |
461b147897 | ||
![]() |
a3cd13b399 | ||
![]() |
eb5d326a1e | ||
![]() |
cce6eb94e2 | ||
![]() |
8311c72c69 | ||
![]() |
9a059285d9 | ||
![]() |
469ac7891d | ||
![]() |
16264a5693 | ||
![]() |
6eedb50f56 | ||
![]() |
5f07215662 | ||
![]() |
d3a02fb2ca | ||
![]() |
2c823f1aaa | ||
![]() |
706293aadc | ||
![]() |
dfdceb4258 | ||
![]() |
0014b7fdb9 | ||
![]() |
3b28f1f8af | ||
![]() |
c689d476ca | ||
![]() |
cc534fd21f | ||
![]() |
124efcf247 | ||
![]() |
dd452b05e3 | ||
![]() |
b5617f7df9 | ||
![]() |
b73ca1ba3a | ||
![]() |
39ad071c4f | ||
![]() |
fa0b576a45 | ||
![]() |
c2c7591987 | ||
![]() |
3e939e3775 | ||
![]() |
f6384e10bc | ||
![]() |
c283842563 | ||
![]() |
1135976225 | ||
![]() |
76679d33df | ||
![]() |
e9d44b90bc | ||
![]() |
8a7a391166 | ||
![]() |
d6c5a9eece | ||
![]() |
56ad6915d0 | ||
![]() |
466c260782 | ||
![]() |
d8e2d1725a | ||
![]() |
33b43d03ac | ||
![]() |
517509db6e | ||
![]() |
95f01a5976 | ||
![]() |
d08d3bf56d | ||
![]() |
587fd669e8 | ||
![]() |
e3553b87fe | ||
![]() |
a078d2360c | ||
![]() |
2d3abf8e6e | ||
![]() |
cacae0fb7d | ||
![]() |
d7fc215c16 | ||
![]() |
bb69aa18f2 | ||
![]() |
e4f515166a | ||
![]() |
f325acb0ea | ||
![]() |
d1b6e65dd8 | ||
![]() |
4688252bd4 | ||
![]() |
b1df5d3ad7 | ||
![]() |
09bad2938e | ||
![]() |
bad69fd397 | ||
![]() |
b4ae6bf82c | ||
![]() |
eac0f83864 | ||
![]() |
3f16f080ac | ||
![]() |
5de2b974fb | ||
![]() |
6a9cb51828 | ||
![]() |
f1c294ca50 | ||
![]() |
c320e44a23 | ||
![]() |
ea36ef3107 | ||
![]() |
fedc37d079 | ||
![]() |
86ccfcc03c | ||
![]() |
bc8d8ab65a | ||
![]() |
a7a23d3bc8 | ||
![]() |
82d33944e6 | ||
![]() |
8a629ad6fb | ||
![]() |
e1711ffcf2 | ||
![]() |
063d51c662 | ||
![]() |
d8e3575a46 | ||
![]() |
6fa609be3f | ||
![]() |
70982e204c | ||
![]() |
2af7df5865 | ||
![]() |
59610e81de | ||
![]() |
cdfac703ef | ||
![]() |
ad5ebd007e | ||
![]() |
1bd347d997 | ||
![]() |
147056073d | ||
![]() |
1b3231e617 | ||
![]() |
7ac4ad3e38 | ||
![]() |
0a345cb12b | ||
![]() |
19ff0dd17f | ||
![]() |
e3bf8a404b | ||
![]() |
dbb28d9a8f | ||
![]() |
19fd4e755c | ||
![]() |
77698cd924 | ||
![]() |
945d279f28 | ||
![]() |
645d296841 | ||
![]() |
57446de875 | ||
![]() |
3473800ab6 | ||
![]() |
7e5c5153a8 | ||
![]() |
1b8979f285 | ||
![]() |
2aac51dd20 | ||
![]() |
69e32d264c | ||
![]() |
71ae75a5bf | ||
![]() |
86043fd87e | ||
![]() |
d8aaef4f04 | ||
![]() |
7fc34b6369 | ||
![]() |
92b346efa7 | ||
![]() |
c790735b9f | ||
![]() |
64711e615e | ||
![]() |
2065affd45 | ||
![]() |
4067c56b5d | ||
![]() |
65515638aa | ||
![]() |
371ba3ffe3 | ||
![]() |
5395b62f2f | ||
![]() |
3e164eab26 | ||
![]() |
ad85533975 | ||
![]() |
d6c2a97931 | ||
![]() |
17def94ba4 | ||
![]() |
cf254ea240 | ||
![]() |
2d05110239 | ||
![]() |
0017f4fce8 | ||
![]() |
02f3dd947a | ||
![]() |
ef0874feba | ||
![]() |
14aa1420f0 | ||
![]() |
c506cd97e0 | ||
![]() |
3761aacb1d | ||
![]() |
233f13aac5 | ||
![]() |
0ced3ec2d2 | ||
![]() |
5b786f65a4 | ||
![]() |
770542c8c7 | ||
![]() |
4f1207a38c | ||
![]() |
2b65f81e23 | ||
![]() |
c10eb2c17b | ||
![]() |
3f5ac54bf5 | ||
![]() |
790f9409b1 | ||
![]() |
a3d831b723 | ||
![]() |
b9483deea3 | ||
![]() |
c211d2ac45 | ||
![]() |
119a03dd5f | ||
![]() |
f3fbb7e1f6 | ||
![]() |
2be98e8467 | ||
![]() |
91b1806e9d | ||
![]() |
8548b3645a | ||
![]() |
4bbfb3f9a3 | ||
![]() |
98eec72522 | ||
![]() |
562d50df4d | ||
![]() |
f3ee5ab372 | ||
![]() |
aa2a3e5b07 | ||
![]() |
8467307239 | ||
![]() |
90da951a6f | ||
![]() |
59f81b331c | ||
![]() |
77a7d3d26a | ||
![]() |
4321b80999 | ||
![]() |
4b93b06323 | ||
![]() |
96fb2e4b7c | ||
![]() |
3dc1048dfb | ||
![]() |
28f4bd5a67 | ||
![]() |
b01c270889 | ||
![]() |
7c92f8a90b | ||
![]() |
a4a04872a3 | ||
![]() |
f2d13cd647 | ||
![]() |
9346908485 | ||
![]() |
b3a7ba14f7 | ||
![]() |
524ef9553c | ||
![]() |
a85f6e72fe | ||
![]() |
1ae6a41336 | ||
![]() |
3ec55791c6 | ||
![]() |
a3accd28ea | ||
![]() |
fc5bc84207 | ||
![]() |
d8b4051d6d | ||
![]() |
804a1cc22d | ||
![]() |
cdd60e5f9c | ||
![]() |
7c76f07384 | ||
![]() |
a3b1ef9527 | ||
![]() |
0bd4150a80 | ||
![]() |
7f7f5253f2 | ||
![]() |
2a0a847634 | ||
![]() |
92ddb37ed3 | ||
![]() |
01f944e6bd | ||
![]() |
732c8d7350 | ||
![]() |
21678aeef5 | ||
![]() |
cda1ca35a4 | ||
![]() |
923580d2cd | ||
![]() |
190af1d32b | ||
![]() |
bffe0443f9 | ||
![]() |
29e99e229b | ||
![]() |
2736b88dd5 | ||
![]() |
e739c60e9f | ||
![]() |
1c0afbc5c5 | ||
![]() |
ee1ba5c0f2 | ||
![]() |
02342ba540 | ||
![]() |
12a130e9bc | ||
![]() |
dfb97e7961 | ||
![]() |
54e03a62bd | ||
![]() |
ec9659ff25 | ||
![]() |
0a41b253f3 | ||
![]() |
002b86198c | ||
![]() |
7161a235f1 | ||
![]() |
391034a785 | ||
![]() |
446a7fbd67 | ||
![]() |
2ccb09434f | ||
![]() |
7974060a40 | ||
![]() |
e8abcc1765 | ||
![]() |
1be1927a1f | ||
![]() |
894ff71664 | ||
![]() |
93bed9c5df | ||
![]() |
34c008adce | ||
![]() |
22ddd91b1f | ||
![]() |
e121f34407 | ||
![]() |
df8a3e870a | ||
![]() |
13b6708a09 | ||
![]() |
6aa196cf55 | ||
![]() |
142493076a | ||
![]() |
3d905a7a4f | ||
![]() |
0a8a6ed168 | ||
![]() |
68b6fa46e6 | ||
![]() |
80f7c54e4d | ||
![]() |
c0e2041006 | ||
![]() |
f6d2cd7704 | ||
![]() |
6d9a629b15 | ||
![]() |
ead8edc7cd | ||
![]() |
664eefdddb | ||
![]() |
220d76c974 | ||
![]() |
d973192b5e | ||
![]() |
978bb5059f | ||
![]() |
f59178bc33 | ||
![]() |
53899cc492 | ||
![]() |
6379d0fe0f | ||
![]() |
a1c4d8696a | ||
![]() |
7f42ed86f2 | ||
![]() |
6bb8b4c994 | ||
![]() |
afefbe4e56 | ||
![]() |
5cd37e5505 | ||
![]() |
d6a825981d | ||
![]() |
a1c01c6722 | ||
![]() |
8d182f1d79 | ||
![]() |
29707bd2ea | ||
![]() |
c2bf2a8174 | ||
![]() |
b402849557 | ||
![]() |
cb85dcfcb8 | ||
![]() |
49379140c7 | ||
![]() |
63f6c86b46 | ||
![]() |
1865898cd4 | ||
![]() |
8935ecfdb8 | ||
![]() |
4902bed409 | ||
![]() |
e6f6aba207 | ||
![]() |
bab622de25 | ||
![]() |
a86c1624a7 | ||
![]() |
c707392a5a | ||
![]() |
1f07721ec4 | ||
![]() |
4767e45035 | ||
![]() |
f2a64ed685 | ||
![]() |
2c9966a0a3 | ||
![]() |
953aa95c64 | ||
![]() |
446266776e | ||
![]() |
845af014dc | ||
![]() |
7c8f8703a1 | ||
![]() |
a0c18e4380 | ||
![]() |
e8d8099563 | ||
![]() |
2c9d90d463 | ||
![]() |
130d2064d5 | ||
![]() |
a10eb2a0d7 | ||
![]() |
b71cc5d7ee | ||
![]() |
8512f9eda1 | ||
![]() |
80058083b8 | ||
![]() |
d5a17a3c25 | ||
![]() |
eab7c87781 | ||
![]() |
d2962d8676 | ||
![]() |
5e602fb575 | ||
![]() |
599964ea5f | ||
![]() |
562de7926b | ||
![]() |
8eaed95e02 | ||
![]() |
3cffd46008 | ||
![]() |
e239bfda8a | ||
![]() |
bd887b660d | ||
![]() |
7c1e0ea95d | ||
![]() |
afbddbfcda | ||
![]() |
9eeb702ca5 | ||
![]() |
4cb6d6995f | ||
![]() |
fe7e7a8d07 | ||
![]() |
f8ee79ab72 | ||
![]() |
a11165830b | ||
![]() |
11cde61eab | ||
![]() |
cd862409cc | ||
![]() |
7644555d6b | ||
![]() |
07e68aa694 | ||
![]() |
0ae206b1bb | ||
![]() |
5bdac96375 | ||
![]() |
f0e044ecd8 | ||
![]() |
f32234291e | ||
![]() |
83acd6529f | ||
![]() |
6a60cb4dc0 | ||
![]() |
1de03fbe18 | ||
![]() |
0e0a2aa981 | ||
![]() |
af73b9d11b | ||
![]() |
859969241a | ||
![]() |
6852a8ca9d | ||
![]() |
c694f1a4a9 | ||
![]() |
d36543b204 | ||
![]() |
6981234736 | ||
![]() |
10bf3ee9de | ||
![]() |
9953218de1 | ||
![]() |
50c74103aa | ||
![]() |
c5e15123fd | ||
![]() |
4adcdb5ba8 | ||
![]() |
60b36ffaa3 | ||
![]() |
d86104ed5d | ||
![]() |
0f02ef20a9 | ||
![]() |
0768cf17b6 | ||
![]() |
197205853f | ||
![]() |
bebcc24ab8 | ||
![]() |
987248ccbb | ||
![]() |
56ece4db0f | ||
![]() |
c7f8b94ccd | ||
![]() |
95b820305a | ||
![]() |
599b200ca5 | ||
![]() |
a9566b31be | ||
![]() |
f30837d726 | ||
![]() |
1242ffa4c6 | ||
![]() |
37c0347c47 | ||
![]() |
65befb84a0 | ||
![]() |
a41f7ce3bd | ||
![]() |
81ced6bf2a | ||
![]() |
fa8033f998 | ||
![]() |
19e3178d0c | ||
![]() |
1e6262f24f | ||
![]() |
beb2284440 | ||
![]() |
0c4f0de13d | ||
![]() |
0cd31a4b96 | ||
![]() |
ef0a86e7cc | ||
![]() |
b79c1345b9 | ||
![]() |
5bb8b6f16c | ||
![]() |
f6f675d1e1 | ||
![]() |
9f684e7670 | ||
![]() |
a998706452 | ||
![]() |
5e19bc6f84 | ||
![]() |
169f287970 | ||
![]() |
c56926428c | ||
![]() |
43eab5c4e5 | ||
![]() |
c6424c2ce7 | ||
![]() |
dfe19d308c | ||
![]() |
b75db110dc | ||
![]() |
9e50ba6ec6 | ||
![]() |
aca2ba13c2 | ||
![]() |
c5afe7a573 | ||
![]() |
8b90968cb1 | ||
![]() |
9281dba896 | ||
![]() |
c7555f1c3c | ||
![]() |
e781831032 | ||
![]() |
8025895168 | ||
![]() |
799935e44c | ||
![]() |
209d8d058c | ||
![]() |
0368a70dd7 | ||
![]() |
c3603426de | ||
![]() |
80db9971b5 | ||
![]() |
e3e771708e | ||
![]() |
b9bca2a19f | ||
![]() |
ffae70a99a | ||
![]() |
10ec1e48b0 | ||
![]() |
84123f5445 | ||
![]() |
25531d8393 | ||
![]() |
adfadc542a | ||
![]() |
7682e1cb57 | ||
![]() |
1fe4109a42 | ||
![]() |
b97baf4d47 | ||
![]() |
f91728561f | ||
![]() |
2b75d54ce1 | ||
![]() |
cdc09a7d30 | ||
![]() |
4cc6c3e535 | ||
![]() |
8f43e033a4 | ||
![]() |
918dbd1926 | ||
![]() |
8fc86ce7fa | ||
![]() |
bf0cb40586 | ||
![]() |
a62aefe74b | ||
![]() |
138d232149 | ||
![]() |
8fa64e3ab2 | ||
![]() |
8d0d2bd3fc | ||
![]() |
2146f5f623 | ||
![]() |
623df23570 | ||
![]() |
4a15d8b6d2 | ||
![]() |
af83a62474 | ||
![]() |
a0705a6c67 | ||
![]() |
712b8eec3d | ||
![]() |
558e9c21ed | ||
![]() |
23a7137e6a | ||
![]() |
71a1cb0184 | ||
![]() |
09b47cc536 | ||
![]() |
b5c78de2dd | ||
![]() |
a1bd14e516 | ||
![]() |
9684c8664f | ||
![]() |
3c75361e5a | ||
![]() |
c44c9ab74d | ||
![]() |
8cc9c871b7 | ||
![]() |
31d211cded | ||
![]() |
c42cba1d54 | ||
![]() |
c87f7cb9ac | ||
![]() |
421bee3871 | ||
![]() |
2c7573a032 | ||
![]() |
fe2a0f5089 | ||
![]() |
59ba8dec4e | ||
![]() |
ff0e1bbbc0 | ||
![]() |
18c712cd99 | ||
![]() |
8de1f764fd | ||
![]() |
a91c1ec6d9 | ||
![]() |
c8e351b31e | ||
![]() |
b4c98584db | ||
![]() |
1d17836b7d | ||
![]() |
193a739749 | ||
![]() |
48da2b8353 | ||
![]() |
40dc9de6c9 | ||
![]() |
d7469e4694 | ||
![]() |
fdfabc9f88 | ||
![]() |
5658ca5b90 | ||
![]() |
8b79e6d3c7 | ||
![]() |
8c67087a15 | ||
![]() |
057f73a73e | ||
![]() |
d3bb37d113 | ||
![]() |
6c91b862c9 | ||
![]() |
cba47b635a | ||
![]() |
cd83388f8b | ||
![]() |
02ae247536 | ||
![]() |
1621a68191 | ||
![]() |
2aef495d62 | ||
![]() |
227c168b65 | ||
![]() |
dcbcf5842f | ||
![]() |
8a3ac6d9bd | ||
![]() |
6cabad19d5 | ||
![]() |
dad7a03d19 | ||
![]() |
7b152640d9 | ||
![]() |
78fddff34e | ||
![]() |
78f542f6c0 | ||
![]() |
f8a016edb5 | ||
![]() |
f5453d38ec | ||
![]() |
331bf0af21 | ||
![]() |
9ac69337e8 | ||
![]() |
ca0c844f32 | ||
![]() |
e629e8784f | ||
![]() |
e944c1ee93 | ||
![]() |
02c882bda3 | ||
![]() |
2c29377abf | ||
![]() |
21992f1b9e | ||
![]() |
8a3f090684 | ||
![]() |
a4258505a9 | ||
![]() |
ba4a971bfa | ||
![]() |
6abc120279 | ||
![]() |
64c36b7a8f | ||
![]() |
64b9f0762c | ||
![]() |
e799082e5c | ||
![]() |
eab66a022e | ||
![]() |
485915c27a | ||
![]() |
45fb482ab4 | ||
![]() |
167e2475d8 | ||
![]() |
fe77251242 | ||
![]() |
0c331d2887 | ||
![]() |
39facd5dc6 | ||
![]() |
a2bd5ae638 | ||
![]() |
c57f616141 | ||
![]() |
3af10a1d00 | ||
![]() |
b3987e2402 | ||
![]() |
71dd44f430 | ||
![]() |
584462f9b9 | ||
![]() |
ff5d5d7cbc | ||
![]() |
370e92b1a6 | ||
![]() |
2469768a85 | ||
![]() |
acb8fa613e | ||
![]() |
83b4b7d83d | ||
![]() |
33a7d56042 | ||
![]() |
20de730541 | ||
![]() |
134c27c900 | ||
![]() |
123ae024c6 | ||
![]() |
34a29cd0a5 | ||
![]() |
9fffa9800a | ||
![]() |
8f7f23a0e5 | ||
![]() |
358457f49d | ||
![]() |
90e305d789 | ||
![]() |
b471e52e28 | ||
![]() |
9b5fc37399 | ||
![]() |
537322e3b5 | ||
![]() |
279121f87f | ||
![]() |
a1b2b41cdc | ||
![]() |
0cca90c21d | ||
![]() |
ad052cb510 | ||
![]() |
57f0598373 | ||
![]() |
92d192277e | ||
![]() |
2cdfbd8970 | ||
![]() |
bc0c5f0cdc | ||
![]() |
230ee3a03b | ||
![]() |
ab0bf37d30 | ||
![]() |
b1e66256ce | ||
![]() |
fd80693420 | ||
![]() |
f222f5ed9b | ||
![]() |
59603b902a | ||
![]() |
82a5dd6cf9 | ||
![]() |
a97f9081a3 | ||
![]() |
b3e0ad1303 | ||
![]() |
251090406a | ||
![]() |
9c3ed173b1 | ||
![]() |
7cd41aeb72 | ||
![]() |
8ecaf7ad4c | ||
![]() |
d4049fe593 | ||
![]() |
989b14b519 | ||
![]() |
fb3e11408c | ||
![]() |
3d3af3284e | ||
![]() |
eabcf15790 | ||
![]() |
684bca203a | ||
![]() |
d42ecccf6f | ||
![]() |
5a92c861e3 | ||
![]() |
103b14ea3c | ||
![]() |
e49f091b4e | ||
![]() |
6056f93910 | ||
![]() |
18fa2d1af4 | ||
![]() |
062ed8a4c4 | ||
![]() |
49f3872d04 | ||
![]() |
d9b467098a | ||
![]() |
7d1e51437e | ||
![]() |
f5a7d444e2 | ||
![]() |
a0071c2f9f | ||
![]() |
98a5e43883 | ||
![]() |
f40ebe5b43 | ||
![]() |
d7d78d9a2b | ||
![]() |
5a12d36a0a | ||
![]() |
e61dc76ae1 | ||
![]() |
0da22193bd | ||
![]() |
d43921823e | ||
![]() |
6b5b76b0c9 | ||
![]() |
602971408b | ||
![]() |
2f5d83f07d | ||
![]() |
81b0c316cd | ||
![]() |
6df86384ca | ||
![]() |
3fbcde4bd7 | ||
![]() |
293293c23c | ||
![]() |
47185d757e | ||
![]() |
94c3998507 | ||
![]() |
5fa589691c | ||
![]() |
26bc835c7e | ||
![]() |
fc7da17fa2 | ||
![]() |
dbebe9771b | ||
![]() |
d0af8aa9f1 | ||
![]() |
f8173e93f5 | ||
![]() |
fa6ea6848f | ||
![]() |
5982970e21 | ||
![]() |
c43e5e0c95 | ||
![]() |
b4fc79c38e | ||
![]() |
b9cb270b8f | ||
![]() |
1fe35d7efe | ||
![]() |
ec8c19b1f0 | ||
![]() |
c2525bcc05 | ||
![]() |
985c31b378 | ||
![]() |
135a9f5586 | ||
![]() |
979ff80982 | ||
![]() |
6cde9f3a9d | ||
![]() |
5cad81bf4c | ||
![]() |
18967162cf | ||
![]() |
6ea75f3a9f | ||
![]() |
f51123d009 | ||
![]() |
be6376ee2d | ||
![]() |
120a85a3e5 | ||
![]() |
262fe96c29 | ||
![]() |
1864df5ed0 | ||
![]() |
09bae0a9e8 | ||
![]() |
ebe41a24f1 | ||
![]() |
f0f574a15b | ||
![]() |
9420436c09 | ||
![]() |
2b8f853b96 | ||
![]() |
2c2bf60d0c | ||
![]() |
62b82448d5 | ||
![]() |
2f12e6c3d0 | ||
![]() |
8ce23f8e3e | ||
![]() |
dbe760d6e2 | ||
![]() |
674c26d9c7 | ||
![]() |
7b68b04efd | ||
![]() |
e0b65bf196 | ||
![]() |
77f5cb1f1f | ||
![]() |
2a222737f8 | ||
![]() |
77d48f9cee | ||
![]() |
c77eac325e | ||
![]() |
517ced2a54 | ||
![]() |
70d2c47367 |
8
.gitignore
vendored
Normal file
8
.gitignore
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
*.o
|
||||||
|
*.so
|
||||||
|
ruby/Makefile
|
||||||
|
*.5
|
||||||
|
*.8
|
||||||
|
*.6
|
||||||
|
_obj
|
||||||
|
_test
|
@@ -1,4 +1,41 @@
|
|||||||
|
|
||||||
|
2011-04-24 version 0.5.6:
|
||||||
|
|
||||||
|
* #42 fixes double-free problem on msgpack_unpacker_release_zone
|
||||||
|
|
||||||
|
2011-02-24 version 0.5.5:
|
||||||
|
|
||||||
|
* eliminates dependency of winsock2.h header
|
||||||
|
* fixes msgpack_vc.postbuild.bat file
|
||||||
|
* fixes some implicit cast warnings
|
||||||
|
|
||||||
|
2010-08-29 version 0.5.4:
|
||||||
|
|
||||||
|
* includes msgpack_vc2008.vcproj file in source package
|
||||||
|
* fixes type::fix_int types
|
||||||
|
|
||||||
|
2010-08-27 version 0.5.3:
|
||||||
|
|
||||||
|
* adds type::fix_{u,}int{8,16,32,64} types
|
||||||
|
* adds msgpack_pack_fix_{u,}int{8,16,32,64} functions
|
||||||
|
* adds packer<Stream>::pack_fix_{u,}int{8,16,32,64} functions
|
||||||
|
* fixes include paths
|
||||||
|
|
||||||
|
2010-07-14 version 0.5.2:
|
||||||
|
|
||||||
|
* type::raw::str(), operator==, operator!=, operator< and operator> are now const
|
||||||
|
* generates version.h using AC_OUTPUT macro in ./configure
|
||||||
|
|
||||||
|
2010-07-06 version 0.5.1:
|
||||||
|
|
||||||
|
* Add msgpack_vrefbuffer_new and msgpack_vrefbuffer_free
|
||||||
|
* Add msgpack_sbuffer_new and msgpack_sbuffer_free
|
||||||
|
* Add msgpack_unpacker_next and msgpack_unpack_next
|
||||||
|
* msgpack::unpack returns void
|
||||||
|
* Add MSGPACK_VERSION{,_MAJOR,_MINOR} macros to check header version
|
||||||
|
* Add msgpack_version{,_major,_minor} functions to check library version
|
||||||
|
* ./configure supports --disable-cxx option not to build C++ API
|
||||||
|
|
||||||
2010-04-29 version 0.5.0:
|
2010-04-29 version 0.5.0:
|
||||||
|
|
||||||
* msgpack_object_type is changed. MSGPACK_OBJECT_NIL is now 0x00.
|
* msgpack_object_type is changed. MSGPACK_OBJECT_NIL is now 0x00.
|
||||||
|
1552
cpp/Doxyfile
Normal file
1552
cpp/Doxyfile
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,93 +1,20 @@
|
|||||||
|
SUBDIRS = src test
|
||||||
|
|
||||||
lib_LTLIBRARIES = libmsgpack.la
|
DOC_FILES = \
|
||||||
|
|
||||||
libmsgpack_la_SOURCES = \
|
|
||||||
unpack.c \
|
|
||||||
objectc.c \
|
|
||||||
vrefbuffer.c \
|
|
||||||
zone.c \
|
|
||||||
object.cpp
|
|
||||||
|
|
||||||
# -version-info CURRENT:REVISION:AGE
|
|
||||||
libmsgpack_la_LDFLAGS = -version-info 3:0:0
|
|
||||||
|
|
||||||
|
|
||||||
# backward compatibility
|
|
||||||
lib_LTLIBRARIES += libmsgpackc.la
|
|
||||||
|
|
||||||
libmsgpackc_la_SOURCES = \
|
|
||||||
unpack.c \
|
|
||||||
objectc.c \
|
|
||||||
vrefbuffer.c \
|
|
||||||
zone.c
|
|
||||||
|
|
||||||
libmsgpackc_la_LDFLAGS = -version-info 2:0:0
|
|
||||||
|
|
||||||
# work around for duplicated file name
|
|
||||||
kumo_manager_CFLAGS = $(AM_CFLAGS)
|
|
||||||
kumo_manager_CXXFLAGS = $(AM_CXXFLAGS)
|
|
||||||
|
|
||||||
|
|
||||||
nobase_include_HEADERS = \
|
|
||||||
msgpack/pack_define.h \
|
|
||||||
msgpack/pack_template.h \
|
|
||||||
msgpack/unpack_define.h \
|
|
||||||
msgpack/unpack_template.h \
|
|
||||||
msgpack/sysdep.h \
|
|
||||||
msgpack.h \
|
|
||||||
msgpack/sbuffer.h \
|
|
||||||
msgpack/vrefbuffer.h \
|
|
||||||
msgpack/zbuffer.h \
|
|
||||||
msgpack/pack.h \
|
|
||||||
msgpack/unpack.h \
|
|
||||||
msgpack/object.h \
|
|
||||||
msgpack/zone.h \
|
|
||||||
msgpack.hpp \
|
|
||||||
msgpack/sbuffer.hpp \
|
|
||||||
msgpack/vrefbuffer.hpp \
|
|
||||||
msgpack/zbuffer.hpp \
|
|
||||||
msgpack/pack.hpp \
|
|
||||||
msgpack/unpack.hpp \
|
|
||||||
msgpack/object.hpp \
|
|
||||||
msgpack/zone.hpp \
|
|
||||||
msgpack/type.hpp \
|
|
||||||
msgpack/type/bool.hpp \
|
|
||||||
msgpack/type/float.hpp \
|
|
||||||
msgpack/type/int.hpp \
|
|
||||||
msgpack/type/list.hpp \
|
|
||||||
msgpack/type/deque.hpp \
|
|
||||||
msgpack/type/map.hpp \
|
|
||||||
msgpack/type/nil.hpp \
|
|
||||||
msgpack/type/pair.hpp \
|
|
||||||
msgpack/type/raw.hpp \
|
|
||||||
msgpack/type/set.hpp \
|
|
||||||
msgpack/type/string.hpp \
|
|
||||||
msgpack/type/vector.hpp \
|
|
||||||
msgpack/type/tuple.hpp \
|
|
||||||
msgpack/type/define.hpp \
|
|
||||||
msgpack/type/tr1/unordered_map.hpp \
|
|
||||||
msgpack/type/tr1/unordered_set.hpp
|
|
||||||
|
|
||||||
|
|
||||||
EXTRA_DIST = \
|
|
||||||
README.md \
|
README.md \
|
||||||
LICENSE \
|
LICENSE \
|
||||||
NOTICE \
|
NOTICE \
|
||||||
msgpack_vc8.vcproj \
|
msgpack_vc8.vcproj \
|
||||||
msgpack_vc8.sln \
|
msgpack_vc8.sln \
|
||||||
msgpack_vc8.postbuild.bat
|
msgpack_vc2008.vcproj \
|
||||||
|
msgpack_vc2008.sln \
|
||||||
|
msgpack_vc.postbuild.bat
|
||||||
|
|
||||||
SUBDIRS = test
|
EXTRA_DIST = \
|
||||||
|
$(DOC_FILES)
|
||||||
|
|
||||||
check_PROGRAMS = \
|
doxygen:
|
||||||
msgpackc_test \
|
./preprocess clean
|
||||||
msgpack_test
|
cd src && $(MAKE) doxygen
|
||||||
|
./preprocess
|
||||||
msgpackc_test_SOURCES = msgpackc_test.cpp
|
|
||||||
msgpackc_test_LDADD = libmsgpack.la -lgtest_main
|
|
||||||
|
|
||||||
msgpack_test_SOURCES = msgpack_test.cpp
|
|
||||||
msgpack_test_LDADD = libmsgpack.la -lgtest_main
|
|
||||||
|
|
||||||
TESTS = $(check_PROGRAMS)
|
|
||||||
|
|
||||||
|
@@ -13,9 +13,10 @@ On UNIX-like platform, run ./configure && make && sudo make install:
|
|||||||
$ make
|
$ make
|
||||||
$ sudo make install
|
$ sudo make install
|
||||||
|
|
||||||
On Windows, open msgpack_vc8.vcproj file and build it using batch build. DLLs are built on lib folder, and the headers are built on include folder.
|
On Windows, open msgpack_vc8.vcproj or msgpack_vc2008 file and build it using batch build. DLLs are built on lib folder,
|
||||||
|
and the headers are built on include folder.
|
||||||
|
|
||||||
To use the library in your program, include msgpack.hpp header and link msgpack and msgpackc library.
|
To use the library in your program, include msgpack.hpp header and link "msgpack" library.
|
||||||
|
|
||||||
|
|
||||||
## Example
|
## Example
|
||||||
@@ -34,15 +35,9 @@ To use the library in your program, include msgpack.hpp header and link msgpack
|
|||||||
msgpack::pack(&buffer, target);
|
msgpack::pack(&buffer, target);
|
||||||
|
|
||||||
// Deserialize the serialized data.
|
// Deserialize the serialized data.
|
||||||
msgpack::zone mempool; // this manages the life of deserialized object
|
msgpack::unpacked msg; // includes memory pool and deserialized object
|
||||||
msgpack::object obj;
|
msgpack::unpack(&msg, sbuf.data(), sbuf.size());
|
||||||
msgpack::unpack_return ret =
|
msgpack::object obj = msg.get();
|
||||||
msgpack::unpack(buffer.data, buffer.size, NULL, &mempool, &obj);
|
|
||||||
|
|
||||||
if(ret != msgapck::UNPACK_SUCCESS) {
|
|
||||||
// error check
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print the deserialized object to stdout.
|
// Print the deserialized object to stdout.
|
||||||
std::cout << obj << std::endl; // ["Hello," "World!"]
|
std::cout << obj << std::endl; // ["Hello," "World!"]
|
||||||
@@ -55,24 +50,24 @@ To use the library in your program, include msgpack.hpp header and link msgpack
|
|||||||
obj.as<int>(); // type is mismatched, msgpack::type_error is thrown
|
obj.as<int>(); // type is mismatched, msgpack::type_error is thrown
|
||||||
}
|
}
|
||||||
|
|
||||||
API document and other example codes are available at the [wiki.](http://msgpack.sourceforge.net/start)
|
API documents and other example codes are available at the [wiki.](http://redmine.msgpack.org/projects/msgpack/wiki)
|
||||||
|
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
Copyright (C) 2008-2010 FURUHASHI Sadayuki
|
Copyright (C) 2008-2010 FURUHASHI Sadayuki
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
You may obtain a copy of the License at
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
Unless required by applicable law or agreed to in writing, software
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
See the License for the specific language governing permissions and
|
See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
|
|
||||||
See also NOTICE file.
|
See also NOTICE file.
|
||||||
|
|
||||||
|
@@ -38,7 +38,8 @@ test -f ChangeLog || touch ChangeLog
|
|||||||
test -f NEWS || touch NEWS
|
test -f NEWS || touch NEWS
|
||||||
test -f README || cp -f README.md README
|
test -f README || cp -f README.md README
|
||||||
|
|
||||||
if ! ./preprocess; then
|
./preprocess
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
AC_INIT(object.cpp)
|
AC_INIT(src/object.cpp)
|
||||||
AC_CONFIG_AUX_DIR(ac)
|
AC_CONFIG_AUX_DIR(ac)
|
||||||
AM_INIT_AUTOMAKE(msgpack, 0.5.0)
|
AM_INIT_AUTOMAKE(msgpack, 0.5.6)
|
||||||
AC_CONFIG_HEADER(config.h)
|
AC_CONFIG_HEADER(config.h)
|
||||||
|
|
||||||
AC_SUBST(CFLAGS)
|
AC_SUBST(CFLAGS)
|
||||||
@@ -9,17 +9,39 @@ CFLAGS="-O4 -Wall $CFLAGS"
|
|||||||
AC_SUBST(CXXFLAGS)
|
AC_SUBST(CXXFLAGS)
|
||||||
CXXFLAGS="-O4 -Wall $CXXFLAGS"
|
CXXFLAGS="-O4 -Wall $CXXFLAGS"
|
||||||
|
|
||||||
|
|
||||||
AC_PROG_CC
|
AC_PROG_CC
|
||||||
AC_PROG_CXX
|
|
||||||
|
|
||||||
|
AC_MSG_CHECKING([if C++ API is enabled])
|
||||||
|
AC_ARG_ENABLE(cxx,
|
||||||
|
AS_HELP_STRING([--disable-cxx],
|
||||||
|
[don't build C++ API]) )
|
||||||
|
AC_MSG_RESULT([$enable_cxx])
|
||||||
|
if test "$enable_cxx" != "no"; then
|
||||||
|
AC_PROG_CXX
|
||||||
|
AM_PROG_CC_C_O
|
||||||
|
fi
|
||||||
|
AM_CONDITIONAL(ENABLE_CXX, test "$enable_cxx" != "no")
|
||||||
|
|
||||||
|
|
||||||
AC_PROG_LIBTOOL
|
AC_PROG_LIBTOOL
|
||||||
AM_PROG_AS
|
AM_PROG_AS
|
||||||
AM_PROG_CC_C_O
|
|
||||||
|
|
||||||
AC_LANG_PUSH([C++])
|
|
||||||
AC_CHECK_HEADERS(tr1/unordered_map)
|
AC_MSG_CHECKING([if debug option is enabled])
|
||||||
AC_CHECK_HEADERS(tr1/unordered_set)
|
AC_ARG_ENABLE(debug,
|
||||||
AC_LANG_POP([C++])
|
AS_HELP_STRING([--disable-debug],
|
||||||
|
[disable assert macros and omit -g option]) )
|
||||||
|
AC_MSG_RESULT([$enable_debug])
|
||||||
|
if test "$enable_debug" != "no"; then
|
||||||
|
CXXFLAGS="$CXXFLAGS -g"
|
||||||
|
CFLAGS="$CFLAGS -g"
|
||||||
|
else
|
||||||
|
CXXFLAGS="$CXXFLAGS -DNDEBUG"
|
||||||
|
CFLAGS="$CFLAGS -DNDEBUG"
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
AC_CACHE_CHECK([for __sync_* atomic operations], msgpack_cv_atomic_ops, [
|
AC_CACHE_CHECK([for __sync_* atomic operations], msgpack_cv_atomic_ops, [
|
||||||
AC_TRY_LINK([
|
AC_TRY_LINK([
|
||||||
@@ -33,11 +55,21 @@ if test "$msgpack_cv_atomic_ops" != "yes"; then
|
|||||||
Note that gcc < 4.1 is not supported.
|
Note that gcc < 4.1 is not supported.
|
||||||
|
|
||||||
If you are using gcc >= 4.1 and the default target CPU architecture is "i386", try to
|
If you are using gcc >= 4.1 and the default target CPU architecture is "i386", try to
|
||||||
add CFLAGS="--march=i686" and CXXFLAGS="-march=i686" options to ./configure as follows:
|
add CFLAGS="-march=i686" and CXXFLAGS="-march=i686" options to ./configure as follows:
|
||||||
|
|
||||||
$ ./configure CFLAGS="-march=i686" CXXFLAGS="-march=i686"
|
$ ./configure CFLAGS="-march=i686" CXXFLAGS="-march=i686"
|
||||||
])
|
])
|
||||||
fi
|
fi
|
||||||
|
|
||||||
AC_OUTPUT([Makefile test/Makefile])
|
|
||||||
|
major=`echo $VERSION | sed 's/\([[0-9]]*\)\.\([[0-9]]*\).*/\1/'`
|
||||||
|
minor=`echo $VERSION | sed 's/\([[0-9]]*\)\.\([[0-9]]*\).*/\2/'`
|
||||||
|
AC_SUBST(VERSION_MAJOR, $major)
|
||||||
|
AC_SUBST(VERSION_MINOR, $minor)
|
||||||
|
|
||||||
|
|
||||||
|
AC_OUTPUT([Makefile
|
||||||
|
src/Makefile
|
||||||
|
src/msgpack/version.h
|
||||||
|
test/Makefile])
|
||||||
|
|
||||||
|
@@ -1,15 +0,0 @@
|
|||||||
#include "msgpack/type/bool.hpp"
|
|
||||||
#include "msgpack/type/float.hpp"
|
|
||||||
#include "msgpack/type/int.hpp"
|
|
||||||
#include "msgpack/type/list.hpp"
|
|
||||||
#include "msgpack/type/deque.hpp"
|
|
||||||
#include "msgpack/type/map.hpp"
|
|
||||||
#include "msgpack/type/nil.hpp"
|
|
||||||
#include "msgpack/type/pair.hpp"
|
|
||||||
#include "msgpack/type/raw.hpp"
|
|
||||||
#include "msgpack/type/set.hpp"
|
|
||||||
#include "msgpack/type/string.hpp"
|
|
||||||
#include "msgpack/type/vector.hpp"
|
|
||||||
#include "msgpack/type/tuple.hpp"
|
|
||||||
#include "msgpack/type/define.hpp"
|
|
||||||
|
|
@@ -1,123 +0,0 @@
|
|||||||
/*
|
|
||||||
* MessagePack for C unpacking routine
|
|
||||||
*
|
|
||||||
* Copyright (C) 2008-2009 FURUHASHI Sadayuki
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
#ifndef MSGPACK_UNPACKER_H__
|
|
||||||
#define MSGPACK_UNPACKER_H__
|
|
||||||
|
|
||||||
#include "msgpack/zone.h"
|
|
||||||
#include "msgpack/object.h"
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
typedef struct msgpack_unpacker {
|
|
||||||
char* buffer;
|
|
||||||
size_t used;
|
|
||||||
size_t free;
|
|
||||||
size_t off;
|
|
||||||
size_t parsed;
|
|
||||||
msgpack_zone* z;
|
|
||||||
size_t initial_buffer_size;
|
|
||||||
void* ctx;
|
|
||||||
} msgpack_unpacker;
|
|
||||||
|
|
||||||
|
|
||||||
bool msgpack_unpacker_init(msgpack_unpacker* mpac, size_t initial_buffer_size);
|
|
||||||
void msgpack_unpacker_destroy(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
msgpack_unpacker* msgpack_unpacker_new(size_t initial_buffer_size);
|
|
||||||
void msgpack_unpacker_free(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
static inline bool msgpack_unpacker_reserve_buffer(msgpack_unpacker* mpac, size_t size);
|
|
||||||
static inline char* msgpack_unpacker_buffer(msgpack_unpacker* mpac);
|
|
||||||
static inline size_t msgpack_unpacker_buffer_capacity(const msgpack_unpacker* mpac);
|
|
||||||
static inline void msgpack_unpacker_buffer_consumed(msgpack_unpacker* mpac, size_t size);
|
|
||||||
|
|
||||||
|
|
||||||
int msgpack_unpacker_execute(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
msgpack_object msgpack_unpacker_data(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
msgpack_zone* msgpack_unpacker_release_zone(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
void msgpack_unpacker_reset_zone(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
void msgpack_unpacker_reset(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
static inline size_t msgpack_unpacker_message_size(const msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
typedef enum {
|
|
||||||
MSGPACK_UNPACK_SUCCESS = 2,
|
|
||||||
MSGPACK_UNPACK_EXTRA_BYTES = 1,
|
|
||||||
MSGPACK_UNPACK_CONTINUE = 0,
|
|
||||||
MSGPACK_UNPACK_PARSE_ERROR = -1,
|
|
||||||
} msgpack_unpack_return;
|
|
||||||
|
|
||||||
msgpack_unpack_return
|
|
||||||
msgpack_unpack(const char* data, size_t len, size_t* off,
|
|
||||||
msgpack_zone* z, msgpack_object* result);
|
|
||||||
|
|
||||||
|
|
||||||
static inline size_t msgpack_unpacker_parsed_size(const msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
bool msgpack_unpacker_flush_zone(msgpack_unpacker* mpac);
|
|
||||||
|
|
||||||
bool msgpack_unpacker_expand_buffer(msgpack_unpacker* mpac, size_t size);
|
|
||||||
|
|
||||||
bool msgpack_unpacker_reserve_buffer(msgpack_unpacker* mpac, size_t size)
|
|
||||||
{
|
|
||||||
if(mpac->free >= size) { return true; }
|
|
||||||
return msgpack_unpacker_expand_buffer(mpac, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
char* msgpack_unpacker_buffer(msgpack_unpacker* mpac)
|
|
||||||
{
|
|
||||||
return mpac->buffer + mpac->used;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t msgpack_unpacker_buffer_capacity(const msgpack_unpacker* mpac)
|
|
||||||
{
|
|
||||||
return mpac->free;
|
|
||||||
}
|
|
||||||
|
|
||||||
void msgpack_unpacker_buffer_consumed(msgpack_unpacker* mpac, size_t size)
|
|
||||||
{
|
|
||||||
mpac->used += size;
|
|
||||||
mpac->free -= size;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t msgpack_unpacker_message_size(const msgpack_unpacker* mpac)
|
|
||||||
{
|
|
||||||
return mpac->parsed - mpac->off + mpac->used;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t msgpack_unpacker_parsed_size(const msgpack_unpacker* mpac)
|
|
||||||
{
|
|
||||||
return mpac->parsed;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif /* msgpack/unpack.h */
|
|
||||||
|
|
45
cpp/msgpack_vc.postbuild.bat
Normal file
45
cpp/msgpack_vc.postbuild.bat
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
IF NOT EXIST include MKDIR include
|
||||||
|
IF NOT EXIST include\msgpack MKDIR include\msgpack
|
||||||
|
IF NOT EXIST include\msgpack\type MKDIR include\msgpack\type
|
||||||
|
IF NOT EXIST include\msgpack\type\tr1 MKDIR include\msgpack\type\tr1
|
||||||
|
copy src\msgpack\pack_define.h include\msgpack\
|
||||||
|
copy src\msgpack\pack_template.h include\msgpack\
|
||||||
|
copy src\msgpack\unpack_define.h include\msgpack\
|
||||||
|
copy src\msgpack\unpack_template.h include\msgpack\
|
||||||
|
copy src\msgpack\sysdep.h include\msgpack\
|
||||||
|
copy src\msgpack.h include\
|
||||||
|
copy src\msgpack\sbuffer.h include\msgpack\
|
||||||
|
copy src\msgpack\version.h include\msgpack\
|
||||||
|
copy src\msgpack\vrefbuffer.h include\msgpack\
|
||||||
|
copy src\msgpack\zbuffer.h include\msgpack\
|
||||||
|
copy src\msgpack\pack.h include\msgpack\
|
||||||
|
copy src\msgpack\unpack.h include\msgpack\
|
||||||
|
copy src\msgpack\object.h include\msgpack\
|
||||||
|
copy src\msgpack\zone.h include\msgpack\
|
||||||
|
copy src\msgpack.hpp include\
|
||||||
|
copy src\msgpack\sbuffer.hpp include\msgpack\
|
||||||
|
copy src\msgpack\vrefbuffer.hpp include\msgpack\
|
||||||
|
copy src\msgpack\zbuffer.hpp include\msgpack\
|
||||||
|
copy src\msgpack\pack.hpp include\msgpack\
|
||||||
|
copy src\msgpack\unpack.hpp include\msgpack\
|
||||||
|
copy src\msgpack\object.hpp include\msgpack\
|
||||||
|
copy src\msgpack\zone.hpp include\msgpack\
|
||||||
|
copy src\msgpack\type.hpp include\msgpack\
|
||||||
|
copy src\msgpack\type\bool.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\deque.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\fixint.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\float.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\int.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\list.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\map.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\nil.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\pair.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\raw.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\set.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\string.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\vector.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\tuple.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\define.hpp include\msgpack\type\
|
||||||
|
copy src\msgpack\type\tr1\unordered_map.hpp include\msgpack\type\tr1\
|
||||||
|
copy src\msgpack\type\tr1\unordered_set.hpp include\msgpack\type\tr1\
|
||||||
|
|
@@ -1,43 +0,0 @@
|
|||||||
IF NOT EXIST include MKDIR include
|
|
||||||
IF NOT EXIST include\msgpack MKDIR include\msgpack
|
|
||||||
IF NOT EXIST include\msgpack\type MKDIR include\msgpack\type
|
|
||||||
IF NOT EXIST include\msgpack\type\tr1 MKDIR include\msgpack\type\tr1
|
|
||||||
copy msgpack\pack_define.h include\msgpack\
|
|
||||||
copy msgpack\pack_template.h include\msgpack\
|
|
||||||
copy msgpack\unpack_define.h include\msgpack\
|
|
||||||
copy msgpack\unpack_template.h include\msgpack\
|
|
||||||
copy msgpack\sysdep.h include\msgpack\
|
|
||||||
copy msgpack.h include\
|
|
||||||
copy msgpack\sbuffer.h include\msgpack\
|
|
||||||
copy msgpack\vrefbuffer.h include\msgpack\
|
|
||||||
copy msgpack\zbuffer.h include\msgpack\
|
|
||||||
copy msgpack\pack.h include\msgpack\
|
|
||||||
copy msgpack\unpack.h include\msgpack\
|
|
||||||
copy msgpack\object.h include\msgpack\
|
|
||||||
copy msgpack\zone.h include\msgpack\
|
|
||||||
copy msgpack.hpp include\
|
|
||||||
copy msgpack\sbuffer.hpp include\msgpack\
|
|
||||||
copy msgpack\vrefbuffer.hpp include\msgpack\
|
|
||||||
copy msgpack\zbuffer.hpp include\msgpack\
|
|
||||||
copy msgpack\pack.hpp include\msgpack\
|
|
||||||
copy msgpack\unpack.hpp include\msgpack\
|
|
||||||
copy msgpack\object.hpp include\msgpack\
|
|
||||||
copy msgpack\zone.hpp include\msgpack\
|
|
||||||
copy msgpack\type.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\bool.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\float.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\int.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\list.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\deque.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\map.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\nil.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\pair.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\raw.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\set.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\string.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\vector.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\tuple.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\define.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\tr1\unordered_map.hpp include\msgpack\type\
|
|
||||||
copy msgpack\type\tr1\unordered_set.hpp include\msgpack\type\
|
|
||||||
|
|
@@ -28,7 +28,7 @@
|
|||||||
<Tool
|
<Tool
|
||||||
Name="VCCustomBuildTool"
|
Name="VCCustomBuildTool"
|
||||||
Description="Gathering header files"
|
Description="Gathering header files"
|
||||||
CommandLine="msgpack_vc8.postbuild.bat"
|
CommandLine="msgpack_vc.postbuild.bat"
|
||||||
Outputs="include"
|
Outputs="include"
|
||||||
/>
|
/>
|
||||||
<Tool
|
<Tool
|
||||||
@@ -96,7 +96,7 @@
|
|||||||
<Tool
|
<Tool
|
||||||
Name="VCCustomBuildTool"
|
Name="VCCustomBuildTool"
|
||||||
Description="Gathering header files"
|
Description="Gathering header files"
|
||||||
CommandLine="msgpack_vc8.postbuild.bat"
|
CommandLine="msgpack_vc.postbuild.bat"
|
||||||
Outputs="include"
|
Outputs="include"
|
||||||
/>
|
/>
|
||||||
<Tool
|
<Tool
|
||||||
@@ -157,7 +157,7 @@
|
|||||||
UniqueIdentifier="{4FC737F1-C7A5-4376-A066-2A32D752A2FF}"
|
UniqueIdentifier="{4FC737F1-C7A5-4376-A066-2A32D752A2FF}"
|
||||||
>
|
>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\objectc.c"
|
RelativePath=".\src\objectc.c"
|
||||||
>
|
>
|
||||||
<FileConfiguration
|
<FileConfiguration
|
||||||
Name="Debug|Win32"
|
Name="Debug|Win32"
|
||||||
@@ -177,7 +177,7 @@
|
|||||||
</FileConfiguration>
|
</FileConfiguration>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\unpack.c"
|
RelativePath=".\src\unpack.c"
|
||||||
>
|
>
|
||||||
<FileConfiguration
|
<FileConfiguration
|
||||||
Name="Debug|Win32"
|
Name="Debug|Win32"
|
||||||
@@ -197,7 +197,7 @@
|
|||||||
</FileConfiguration>
|
</FileConfiguration>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\vrefbuffer.c"
|
RelativePath=".\src\version.c"
|
||||||
>
|
>
|
||||||
<FileConfiguration
|
<FileConfiguration
|
||||||
Name="Debug|Win32"
|
Name="Debug|Win32"
|
||||||
@@ -217,7 +217,7 @@
|
|||||||
</FileConfiguration>
|
</FileConfiguration>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\zone.c"
|
RelativePath=".\src\vrefbuffer.c"
|
||||||
>
|
>
|
||||||
<FileConfiguration
|
<FileConfiguration
|
||||||
Name="Debug|Win32"
|
Name="Debug|Win32"
|
||||||
@@ -237,7 +237,27 @@
|
|||||||
</FileConfiguration>
|
</FileConfiguration>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\object.cpp"
|
RelativePath=".\src\zone.c"
|
||||||
|
>
|
||||||
|
<FileConfiguration
|
||||||
|
Name="Debug|Win32"
|
||||||
|
>
|
||||||
|
<Tool
|
||||||
|
Name="VCCLCompilerTool"
|
||||||
|
CompileAs="2"
|
||||||
|
/>
|
||||||
|
</FileConfiguration>
|
||||||
|
<FileConfiguration
|
||||||
|
Name="Release|Win32"
|
||||||
|
>
|
||||||
|
<Tool
|
||||||
|
Name="VCCLCompilerTool"
|
||||||
|
CompileAs="2"
|
||||||
|
/>
|
||||||
|
</FileConfiguration>
|
||||||
|
</File>
|
||||||
|
<File
|
||||||
|
RelativePath=".\src\object.cpp"
|
||||||
>
|
>
|
||||||
</File>
|
</File>
|
||||||
</Filter>
|
</Filter>
|
||||||
@@ -247,23 +267,23 @@
|
|||||||
UniqueIdentifier="{93995380-89BD-4b04-88EB-625FBE52EBFB}"
|
UniqueIdentifier="{93995380-89BD-4b04-88EB-625FBE52EBFB}"
|
||||||
>
|
>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\msgpack\pack_define.h"
|
RelativePath=".\src\msgpack\pack_define.h"
|
||||||
>
|
>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\msgpack\pack_template.h"
|
RelativePath=".\src\msgpack\pack_template.h"
|
||||||
>
|
>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\msgpack\sysdep.h"
|
RelativePath=".\src\msgpack\sysdep.h"
|
||||||
>
|
>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\msgpack\unpack_define.h"
|
RelativePath=".\src\msgpack\unpack_define.h"
|
||||||
>
|
>
|
||||||
</File>
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\msgpack\unpack_template.h"
|
RelativePath=".\src\msgpack\unpack_template.h"
|
||||||
>
|
>
|
||||||
</File>
|
</File>
|
||||||
</Filter>
|
</Filter>
|
||||||
|
@@ -6,17 +6,29 @@ preprocess() {
|
|||||||
echo ""
|
echo ""
|
||||||
echo "** preprocess failed **"
|
echo "** preprocess failed **"
|
||||||
echo ""
|
echo ""
|
||||||
|
exit 1
|
||||||
else
|
else
|
||||||
mv $1.tmp $1
|
mv $1.tmp $1
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
preprocess msgpack/type/tuple.hpp
|
if [ "$1" = "clean" ];then
|
||||||
preprocess msgpack/type/define.hpp
|
rm -f src/msgpack/type/tuple.hpp
|
||||||
preprocess msgpack/zone.hpp
|
rm -f src/msgpack/type/define.hpp
|
||||||
cp -f ../msgpack/sysdep.h msgpack/
|
rm -f src/msgpack/zone.hpp
|
||||||
cp -f ../msgpack/pack_define.h msgpack/
|
else
|
||||||
cp -f ../msgpack/pack_template.h msgpack/
|
preprocess src/msgpack/type/tuple.hpp
|
||||||
cp -f ../msgpack/unpack_define.h msgpack/
|
preprocess src/msgpack/type/define.hpp
|
||||||
cp -f ../msgpack/unpack_template.h msgpack/
|
preprocess src/msgpack/zone.hpp
|
||||||
|
fi
|
||||||
|
cp -f ../msgpack/sysdep.h src/msgpack/
|
||||||
|
cp -f ../msgpack/pack_define.h src/msgpack/
|
||||||
|
cp -f ../msgpack/pack_template.h src/msgpack/
|
||||||
|
cp -f ../msgpack/unpack_define.h src/msgpack/
|
||||||
|
cp -f ../msgpack/unpack_template.h src/msgpack/
|
||||||
|
cp -f ../test/cases.mpac test/
|
||||||
|
cp -f ../test/cases_compact.mpac test/
|
||||||
|
|
||||||
|
sed -e 's/8\.00/9.00/' < msgpack_vc8.vcproj > msgpack_vc2008.vcproj
|
||||||
|
sed -e 's/9\.00/10.00/' -e 's/msgpack_vc8/msgpack_vc2008/' < msgpack_vc8.sln > msgpack_vc2008.sln
|
||||||
|
|
||||||
|
101
cpp/src/Makefile.am
Normal file
101
cpp/src/Makefile.am
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
|
||||||
|
lib_LTLIBRARIES = libmsgpack.la
|
||||||
|
|
||||||
|
libmsgpack_la_SOURCES = \
|
||||||
|
unpack.c \
|
||||||
|
objectc.c \
|
||||||
|
version.c \
|
||||||
|
vrefbuffer.c \
|
||||||
|
zone.c
|
||||||
|
|
||||||
|
if ENABLE_CXX
|
||||||
|
libmsgpack_la_SOURCES += \
|
||||||
|
object.cpp
|
||||||
|
endif
|
||||||
|
|
||||||
|
# -version-info CURRENT:REVISION:AGE
|
||||||
|
libmsgpack_la_LDFLAGS = -version-info 3:0:0
|
||||||
|
|
||||||
|
|
||||||
|
# backward compatibility
|
||||||
|
lib_LTLIBRARIES += libmsgpackc.la
|
||||||
|
|
||||||
|
libmsgpackc_la_SOURCES = \
|
||||||
|
unpack.c \
|
||||||
|
objectc.c \
|
||||||
|
version.c \
|
||||||
|
vrefbuffer.c \
|
||||||
|
zone.c
|
||||||
|
|
||||||
|
libmsgpackc_la_LDFLAGS = -version-info 2:0:0
|
||||||
|
|
||||||
|
|
||||||
|
nobase_include_HEADERS = \
|
||||||
|
msgpack/pack_define.h \
|
||||||
|
msgpack/pack_template.h \
|
||||||
|
msgpack/unpack_define.h \
|
||||||
|
msgpack/unpack_template.h \
|
||||||
|
msgpack/sysdep.h \
|
||||||
|
msgpack.h \
|
||||||
|
msgpack/sbuffer.h \
|
||||||
|
msgpack/version.h \
|
||||||
|
msgpack/vrefbuffer.h \
|
||||||
|
msgpack/zbuffer.h \
|
||||||
|
msgpack/pack.h \
|
||||||
|
msgpack/unpack.h \
|
||||||
|
msgpack/object.h \
|
||||||
|
msgpack/zone.h
|
||||||
|
|
||||||
|
if ENABLE_CXX
|
||||||
|
nobase_include_HEADERS += \
|
||||||
|
msgpack.hpp \
|
||||||
|
msgpack/sbuffer.hpp \
|
||||||
|
msgpack/vrefbuffer.hpp \
|
||||||
|
msgpack/zbuffer.hpp \
|
||||||
|
msgpack/pack.hpp \
|
||||||
|
msgpack/unpack.hpp \
|
||||||
|
msgpack/object.hpp \
|
||||||
|
msgpack/zone.hpp \
|
||||||
|
msgpack/type.hpp \
|
||||||
|
msgpack/type/bool.hpp \
|
||||||
|
msgpack/type/deque.hpp \
|
||||||
|
msgpack/type/float.hpp \
|
||||||
|
msgpack/type/fixint.hpp \
|
||||||
|
msgpack/type/int.hpp \
|
||||||
|
msgpack/type/list.hpp \
|
||||||
|
msgpack/type/map.hpp \
|
||||||
|
msgpack/type/nil.hpp \
|
||||||
|
msgpack/type/pair.hpp \
|
||||||
|
msgpack/type/raw.hpp \
|
||||||
|
msgpack/type/set.hpp \
|
||||||
|
msgpack/type/string.hpp \
|
||||||
|
msgpack/type/vector.hpp \
|
||||||
|
msgpack/type/tuple.hpp \
|
||||||
|
msgpack/type/define.hpp \
|
||||||
|
msgpack/type/tr1/unordered_map.hpp \
|
||||||
|
msgpack/type/tr1/unordered_set.hpp
|
||||||
|
endif
|
||||||
|
|
||||||
|
EXTRA_DIST = \
|
||||||
|
msgpack/version.h.in \
|
||||||
|
msgpack/zone.hpp.erb \
|
||||||
|
msgpack/type/define.hpp.erb \
|
||||||
|
msgpack/type/tuple.hpp.erb
|
||||||
|
|
||||||
|
|
||||||
|
doxygen_c:
|
||||||
|
cat ../Doxyfile > Doxyfile_c
|
||||||
|
echo "FILE_PATTERNS = *.h" >> Doxyfile_c
|
||||||
|
echo "OUTPUT_DIRECTORY = doc_c" >> Doxyfile_c
|
||||||
|
echo "PROJECT_NAME = \"MessagePack for C\"" >> Doxyfile_c
|
||||||
|
doxygen Doxyfile_c
|
||||||
|
|
||||||
|
doxygen_cpp:
|
||||||
|
cat ../Doxyfile > Doxyfile_cpp
|
||||||
|
echo "FILE_PATTERNS = *.hpp" >> Doxyfile_cpp
|
||||||
|
echo "OUTPUT_DIRECTORY = doc_cpp" >> Doxyfile_cpp
|
||||||
|
echo "PROJECT_NAME = \"MessagePack for C++\"" >> Doxyfile_cpp
|
||||||
|
doxygen Doxyfile_cpp
|
||||||
|
|
||||||
|
doxygen: doxygen_c doxygen_cpp
|
||||||
|
|
@@ -15,9 +15,16 @@
|
|||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack MessagePack C
|
||||||
|
* @{
|
||||||
|
* @}
|
||||||
|
*/
|
||||||
#include "msgpack/object.h"
|
#include "msgpack/object.h"
|
||||||
#include "msgpack/zone.h"
|
#include "msgpack/zone.h"
|
||||||
#include "msgpack/pack.h"
|
#include "msgpack/pack.h"
|
||||||
#include "msgpack/unpack.h"
|
#include "msgpack/unpack.h"
|
||||||
#include "msgpack/sbuffer.h"
|
#include "msgpack/sbuffer.h"
|
||||||
#include "msgpack/vrefbuffer.h"
|
#include "msgpack/vrefbuffer.h"
|
||||||
|
#include "msgpack/version.h"
|
||||||
|
|
@@ -18,7 +18,7 @@
|
|||||||
#ifndef MSGPACK_OBJECT_H__
|
#ifndef MSGPACK_OBJECT_H__
|
||||||
#define MSGPACK_OBJECT_H__
|
#define MSGPACK_OBJECT_H__
|
||||||
|
|
||||||
#include "msgpack/zone.h"
|
#include "zone.h"
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
@@ -26,6 +26,12 @@ extern "C" {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_object Dynamically typed object
|
||||||
|
* @ingroup msgpack
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
MSGPACK_OBJECT_NIL = 0x00,
|
MSGPACK_OBJECT_NIL = 0x00,
|
||||||
MSGPACK_OBJECT_BOOLEAN = 0x01,
|
MSGPACK_OBJECT_BOOLEAN = 0x01,
|
||||||
@@ -81,6 +87,8 @@ void msgpack_object_print(FILE* out, msgpack_object o);
|
|||||||
|
|
||||||
bool msgpack_object_equal(const msgpack_object x, const msgpack_object y);
|
bool msgpack_object_equal(const msgpack_object x, const msgpack_object y);
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
@@ -18,9 +18,9 @@
|
|||||||
#ifndef MSGPACK_OBJECT_HPP__
|
#ifndef MSGPACK_OBJECT_HPP__
|
||||||
#define MSGPACK_OBJECT_HPP__
|
#define MSGPACK_OBJECT_HPP__
|
||||||
|
|
||||||
#include "msgpack/object.h"
|
#include "object.h"
|
||||||
#include "msgpack/pack.hpp"
|
#include "pack.hpp"
|
||||||
#include "msgpack/zone.hpp"
|
#include "zone.hpp"
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <typeinfo>
|
#include <typeinfo>
|
@@ -18,8 +18,8 @@
|
|||||||
#ifndef MSGPACK_PACK_H__
|
#ifndef MSGPACK_PACK_H__
|
||||||
#define MSGPACK_PACK_H__
|
#define MSGPACK_PACK_H__
|
||||||
|
|
||||||
#include "msgpack/pack_define.h"
|
#include "pack_define.h"
|
||||||
#include "msgpack/object.h"
|
#include "object.h"
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
@@ -27,6 +27,19 @@ extern "C" {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_buffer Buffers
|
||||||
|
* @ingroup msgpack
|
||||||
|
* @{
|
||||||
|
* @}
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_pack Serializer
|
||||||
|
* @ingroup msgpack
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
typedef int (*msgpack_packer_write)(void* data, const char* buf, unsigned int len);
|
typedef int (*msgpack_packer_write)(void* data, const char* buf, unsigned int len);
|
||||||
|
|
||||||
typedef struct msgpack_packer {
|
typedef struct msgpack_packer {
|
||||||
@@ -57,6 +70,15 @@ static int msgpack_pack_int16(msgpack_packer* pk, int16_t d);
|
|||||||
static int msgpack_pack_int32(msgpack_packer* pk, int32_t d);
|
static int msgpack_pack_int32(msgpack_packer* pk, int32_t d);
|
||||||
static int msgpack_pack_int64(msgpack_packer* pk, int64_t d);
|
static int msgpack_pack_int64(msgpack_packer* pk, int64_t d);
|
||||||
|
|
||||||
|
static int msgpack_pack_fix_uint8(msgpack_packer* pk, uint8_t d);
|
||||||
|
static int msgpack_pack_fix_uint16(msgpack_packer* pk, uint16_t d);
|
||||||
|
static int msgpack_pack_fix_uint32(msgpack_packer* pk, uint32_t d);
|
||||||
|
static int msgpack_pack_fix_uint64(msgpack_packer* pk, uint64_t d);
|
||||||
|
static int msgpack_pack_fix_int8(msgpack_packer* pk, int8_t d);
|
||||||
|
static int msgpack_pack_fix_int16(msgpack_packer* pk, int16_t d);
|
||||||
|
static int msgpack_pack_fix_int32(msgpack_packer* pk, int32_t d);
|
||||||
|
static int msgpack_pack_fix_int64(msgpack_packer* pk, int64_t d);
|
||||||
|
|
||||||
static int msgpack_pack_float(msgpack_packer* pk, float d);
|
static int msgpack_pack_float(msgpack_packer* pk, float d);
|
||||||
static int msgpack_pack_double(msgpack_packer* pk, double d);
|
static int msgpack_pack_double(msgpack_packer* pk, double d);
|
||||||
|
|
||||||
@@ -74,6 +96,8 @@ static int msgpack_pack_raw_body(msgpack_packer* pk, const void* b, size_t l);
|
|||||||
int msgpack_pack_object(msgpack_packer* pk, msgpack_object d);
|
int msgpack_pack_object(msgpack_packer* pk, msgpack_object d);
|
||||||
|
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
#define msgpack_pack_inline_func(name) \
|
#define msgpack_pack_inline_func(name) \
|
||||||
inline int msgpack_pack ## name
|
inline int msgpack_pack ## name
|
||||||
@@ -81,12 +105,15 @@ int msgpack_pack_object(msgpack_packer* pk, msgpack_object d);
|
|||||||
#define msgpack_pack_inline_func_cint(name) \
|
#define msgpack_pack_inline_func_cint(name) \
|
||||||
inline int msgpack_pack ## name
|
inline int msgpack_pack ## name
|
||||||
|
|
||||||
|
#define msgpack_pack_inline_func_fixint(name) \
|
||||||
|
inline int msgpack_pack_fix ## name
|
||||||
|
|
||||||
#define msgpack_pack_user msgpack_packer*
|
#define msgpack_pack_user msgpack_packer*
|
||||||
|
|
||||||
#define msgpack_pack_append_buffer(user, buf, len) \
|
#define msgpack_pack_append_buffer(user, buf, len) \
|
||||||
return (*(user)->callback)((user)->data, (const char*)buf, len)
|
return (*(user)->callback)((user)->data, (const char*)buf, len)
|
||||||
|
|
||||||
#include "msgpack/pack_template.h"
|
#include "pack_template.h"
|
||||||
|
|
||||||
inline void msgpack_packer_init(msgpack_packer* pk, void* data, msgpack_packer_write callback)
|
inline void msgpack_packer_init(msgpack_packer* pk, void* data, msgpack_packer_write callback)
|
||||||
{
|
{
|
@@ -18,7 +18,7 @@
|
|||||||
#ifndef MSGPACK_PACK_HPP__
|
#ifndef MSGPACK_PACK_HPP__
|
||||||
#define MSGPACK_PACK_HPP__
|
#define MSGPACK_PACK_HPP__
|
||||||
|
|
||||||
#include "msgpack/pack_define.h"
|
#include "pack_define.h"
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
|
|
||||||
@@ -45,6 +45,15 @@ public:
|
|||||||
packer<Stream>& pack_int32(int32_t d);
|
packer<Stream>& pack_int32(int32_t d);
|
||||||
packer<Stream>& pack_int64(int64_t d);
|
packer<Stream>& pack_int64(int64_t d);
|
||||||
|
|
||||||
|
packer<Stream>& pack_fix_uint8(uint8_t d);
|
||||||
|
packer<Stream>& pack_fix_uint16(uint16_t d);
|
||||||
|
packer<Stream>& pack_fix_uint32(uint32_t d);
|
||||||
|
packer<Stream>& pack_fix_uint64(uint64_t d);
|
||||||
|
packer<Stream>& pack_fix_int8(int8_t d);
|
||||||
|
packer<Stream>& pack_fix_int16(int16_t d);
|
||||||
|
packer<Stream>& pack_fix_int32(int32_t d);
|
||||||
|
packer<Stream>& pack_fix_int64(int64_t d);
|
||||||
|
|
||||||
packer<Stream>& pack_short(short d);
|
packer<Stream>& pack_short(short d);
|
||||||
packer<Stream>& pack_int(int d);
|
packer<Stream>& pack_int(int d);
|
||||||
packer<Stream>& pack_long(long d);
|
packer<Stream>& pack_long(long d);
|
||||||
@@ -78,6 +87,15 @@ private:
|
|||||||
static void _pack_int32(Stream& x, int32_t d);
|
static void _pack_int32(Stream& x, int32_t d);
|
||||||
static void _pack_int64(Stream& x, int64_t d);
|
static void _pack_int64(Stream& x, int64_t d);
|
||||||
|
|
||||||
|
static void _pack_fix_uint8(Stream& x, uint8_t d);
|
||||||
|
static void _pack_fix_uint16(Stream& x, uint16_t d);
|
||||||
|
static void _pack_fix_uint32(Stream& x, uint32_t d);
|
||||||
|
static void _pack_fix_uint64(Stream& x, uint64_t d);
|
||||||
|
static void _pack_fix_int8(Stream& x, int8_t d);
|
||||||
|
static void _pack_fix_int16(Stream& x, int16_t d);
|
||||||
|
static void _pack_fix_int32(Stream& x, int32_t d);
|
||||||
|
static void _pack_fix_int64(Stream& x, int64_t d);
|
||||||
|
|
||||||
static void _pack_short(Stream& x, short d);
|
static void _pack_short(Stream& x, short d);
|
||||||
static void _pack_int(Stream& x, int d);
|
static void _pack_int(Stream& x, int d);
|
||||||
static void _pack_long(Stream& x, long d);
|
static void _pack_long(Stream& x, long d);
|
||||||
@@ -133,11 +151,15 @@ inline void pack(Stream& s, const T& v)
|
|||||||
template <typename Stream> \
|
template <typename Stream> \
|
||||||
inline void packer<Stream>::_pack ## name
|
inline void packer<Stream>::_pack ## name
|
||||||
|
|
||||||
|
#define msgpack_pack_inline_func_fixint(name) \
|
||||||
|
template <typename Stream> \
|
||||||
|
inline void packer<Stream>::_pack_fix ## name
|
||||||
|
|
||||||
#define msgpack_pack_user Stream&
|
#define msgpack_pack_user Stream&
|
||||||
|
|
||||||
#define msgpack_pack_append_buffer append_buffer
|
#define msgpack_pack_append_buffer append_buffer
|
||||||
|
|
||||||
#include "msgpack/pack_template.h"
|
#include "pack_template.h"
|
||||||
|
|
||||||
|
|
||||||
template <typename Stream>
|
template <typename Stream>
|
||||||
@@ -149,6 +171,7 @@ packer<Stream>::packer(Stream& s) : m_stream(s) { }
|
|||||||
template <typename Stream>
|
template <typename Stream>
|
||||||
packer<Stream>::~packer() { }
|
packer<Stream>::~packer() { }
|
||||||
|
|
||||||
|
|
||||||
template <typename Stream>
|
template <typename Stream>
|
||||||
inline packer<Stream>& packer<Stream>::pack_uint8(uint8_t d)
|
inline packer<Stream>& packer<Stream>::pack_uint8(uint8_t d)
|
||||||
{ _pack_uint8(m_stream, d); return *this; }
|
{ _pack_uint8(m_stream, d); return *this; }
|
||||||
@@ -182,6 +205,39 @@ inline packer<Stream>& packer<Stream>::pack_int64(int64_t d)
|
|||||||
{ _pack_int64(m_stream, d); return *this;}
|
{ _pack_int64(m_stream, d); return *this;}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_uint8(uint8_t d)
|
||||||
|
{ _pack_fix_uint8(m_stream, d); return *this; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_uint16(uint16_t d)
|
||||||
|
{ _pack_fix_uint16(m_stream, d); return *this; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_uint32(uint32_t d)
|
||||||
|
{ _pack_fix_uint32(m_stream, d); return *this; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_uint64(uint64_t d)
|
||||||
|
{ _pack_fix_uint64(m_stream, d); return *this; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_int8(int8_t d)
|
||||||
|
{ _pack_fix_int8(m_stream, d); return *this; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_int16(int16_t d)
|
||||||
|
{ _pack_fix_int16(m_stream, d); return *this; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_int32(int32_t d)
|
||||||
|
{ _pack_fix_int32(m_stream, d); return *this; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& packer<Stream>::pack_fix_int64(int64_t d)
|
||||||
|
{ _pack_fix_int64(m_stream, d); return *this;}
|
||||||
|
|
||||||
|
|
||||||
template <typename Stream>
|
template <typename Stream>
|
||||||
inline packer<Stream>& packer<Stream>::pack_short(short d)
|
inline packer<Stream>& packer<Stream>::pack_short(short d)
|
||||||
{ _pack_short(m_stream, d); return *this; }
|
{ _pack_short(m_stream, d); return *this; }
|
@@ -21,15 +21,17 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#ifndef MSGPACK_SBUFFER_INIT_SIZE
|
|
||||||
#define MSGPACK_SBUFFER_INIT_SIZE 8192
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_sbuffer Simple buffer
|
||||||
|
* @ingroup msgpack_buffer
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
typedef struct msgpack_sbuffer {
|
typedef struct msgpack_sbuffer {
|
||||||
size_t size;
|
size_t size;
|
||||||
char* data;
|
char* data;
|
||||||
@@ -46,6 +48,22 @@ static inline void msgpack_sbuffer_destroy(msgpack_sbuffer* sbuf)
|
|||||||
free(sbuf->data);
|
free(sbuf->data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline msgpack_sbuffer* msgpack_sbuffer_new(void)
|
||||||
|
{
|
||||||
|
return (msgpack_sbuffer*)calloc(1, sizeof(msgpack_sbuffer));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void msgpack_sbuffer_free(msgpack_sbuffer* sbuf)
|
||||||
|
{
|
||||||
|
if(sbuf == NULL) { return; }
|
||||||
|
msgpack_sbuffer_destroy(sbuf);
|
||||||
|
free(sbuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef MSGPACK_SBUFFER_INIT_SIZE
|
||||||
|
#define MSGPACK_SBUFFER_INIT_SIZE 8192
|
||||||
|
#endif
|
||||||
|
|
||||||
static inline int msgpack_sbuffer_write(void* data, const char* buf, unsigned int len)
|
static inline int msgpack_sbuffer_write(void* data, const char* buf, unsigned int len)
|
||||||
{
|
{
|
||||||
msgpack_sbuffer* sbuf = (msgpack_sbuffer*)data;
|
msgpack_sbuffer* sbuf = (msgpack_sbuffer*)data;
|
||||||
@@ -77,6 +95,13 @@ static inline char* msgpack_sbuffer_release(msgpack_sbuffer* sbuf)
|
|||||||
return tmp;
|
return tmp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void msgpack_sbuffer_clear(msgpack_sbuffer* sbuf)
|
||||||
|
{
|
||||||
|
sbuf->size = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
@@ -18,7 +18,7 @@
|
|||||||
#ifndef MSGPACK_SBUFFER_HPP__
|
#ifndef MSGPACK_SBUFFER_HPP__
|
||||||
#define MSGPACK_SBUFFER_HPP__
|
#define MSGPACK_SBUFFER_HPP__
|
||||||
|
|
||||||
#include "msgpack/sbuffer.h"
|
#include "sbuffer.h"
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
|
|
||||||
namespace msgpack {
|
namespace msgpack {
|
||||||
@@ -28,9 +28,13 @@ class sbuffer : public msgpack_sbuffer {
|
|||||||
public:
|
public:
|
||||||
sbuffer(size_t initsz = MSGPACK_SBUFFER_INIT_SIZE)
|
sbuffer(size_t initsz = MSGPACK_SBUFFER_INIT_SIZE)
|
||||||
{
|
{
|
||||||
base::data = (char*)::malloc(initsz);
|
if(initsz == 0) {
|
||||||
if(!base::data) {
|
base::data = NULL;
|
||||||
throw std::bad_alloc();
|
} else {
|
||||||
|
base::data = (char*)::malloc(initsz);
|
||||||
|
if(!base::data) {
|
||||||
|
throw std::bad_alloc();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
base::size = 0;
|
base::size = 0;
|
||||||
@@ -72,10 +76,15 @@ public:
|
|||||||
return msgpack_sbuffer_release(this);
|
return msgpack_sbuffer_release(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void clear()
|
||||||
|
{
|
||||||
|
msgpack_sbuffer_clear(this);
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void expand_buffer(size_t len)
|
void expand_buffer(size_t len)
|
||||||
{
|
{
|
||||||
size_t nsize = (base::alloc) ?
|
size_t nsize = (base::alloc > 0) ?
|
||||||
base::alloc * 2 : MSGPACK_SBUFFER_INIT_SIZE;
|
base::alloc * 2 : MSGPACK_SBUFFER_INIT_SIZE;
|
||||||
|
|
||||||
while(nsize < base::size + len) { nsize *= 2; }
|
while(nsize < base::size + len) { nsize *= 2; }
|
16
cpp/src/msgpack/type.hpp
Normal file
16
cpp/src/msgpack/type.hpp
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
#include "type/bool.hpp"
|
||||||
|
#include "type/deque.hpp"
|
||||||
|
#include "type/fixint.hpp"
|
||||||
|
#include "type/float.hpp"
|
||||||
|
#include "type/int.hpp"
|
||||||
|
#include "type/list.hpp"
|
||||||
|
#include "type/map.hpp"
|
||||||
|
#include "type/nil.hpp"
|
||||||
|
#include "type/pair.hpp"
|
||||||
|
#include "type/raw.hpp"
|
||||||
|
#include "type/set.hpp"
|
||||||
|
#include "type/string.hpp"
|
||||||
|
#include "type/vector.hpp"
|
||||||
|
#include "type/tuple.hpp"
|
||||||
|
#include "type/define.hpp"
|
||||||
|
|
172
cpp/src/msgpack/type/fixint.hpp
Normal file
172
cpp/src/msgpack/type/fixint.hpp
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
//
|
||||||
|
// MessagePack for C++ static resolution routine
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 FURUHASHI Sadayuki
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
//
|
||||||
|
#ifndef MSGPACK_TYPE_FIXINT_HPP__
|
||||||
|
#define MSGPACK_TYPE_FIXINT_HPP__
|
||||||
|
|
||||||
|
#include "msgpack/object.hpp"
|
||||||
|
#include "msgpack/type/int.hpp"
|
||||||
|
|
||||||
|
namespace msgpack {
|
||||||
|
|
||||||
|
namespace type {
|
||||||
|
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct fix_int {
|
||||||
|
fix_int() : value(0) { }
|
||||||
|
fix_int(T value) : value(value) { }
|
||||||
|
|
||||||
|
operator T() const { return value; }
|
||||||
|
|
||||||
|
T get() const { return value; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
T value;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
typedef fix_int<uint8_t> fix_uint8;
|
||||||
|
typedef fix_int<uint16_t> fix_uint16;
|
||||||
|
typedef fix_int<uint32_t> fix_uint32;
|
||||||
|
typedef fix_int<uint64_t> fix_uint64;
|
||||||
|
|
||||||
|
typedef fix_int<int8_t> fix_int8;
|
||||||
|
typedef fix_int<int16_t> fix_int16;
|
||||||
|
typedef fix_int<int32_t> fix_int32;
|
||||||
|
typedef fix_int<int64_t> fix_int64;
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace type
|
||||||
|
|
||||||
|
|
||||||
|
inline type::fix_int8& operator>> (object o, type::fix_int8& v)
|
||||||
|
{ v = type::detail::convert_integer<int8_t>(o); return v; }
|
||||||
|
|
||||||
|
inline type::fix_int16& operator>> (object o, type::fix_int16& v)
|
||||||
|
{ v = type::detail::convert_integer<int16_t>(o); return v; }
|
||||||
|
|
||||||
|
inline type::fix_int32& operator>> (object o, type::fix_int32& v)
|
||||||
|
{ v = type::detail::convert_integer<int32_t>(o); return v; }
|
||||||
|
|
||||||
|
inline type::fix_int64& operator>> (object o, type::fix_int64& v)
|
||||||
|
{ v = type::detail::convert_integer<int64_t>(o); return v; }
|
||||||
|
|
||||||
|
|
||||||
|
inline type::fix_uint8& operator>> (object o, type::fix_uint8& v)
|
||||||
|
{ v = type::detail::convert_integer<uint8_t>(o); return v; }
|
||||||
|
|
||||||
|
inline type::fix_uint16& operator>> (object o, type::fix_uint16& v)
|
||||||
|
{ v = type::detail::convert_integer<uint16_t>(o); return v; }
|
||||||
|
|
||||||
|
inline type::fix_uint32& operator>> (object o, type::fix_uint32& v)
|
||||||
|
{ v = type::detail::convert_integer<uint32_t>(o); return v; }
|
||||||
|
|
||||||
|
inline type::fix_uint64& operator>> (object o, type::fix_uint64& v)
|
||||||
|
{ v = type::detail::convert_integer<uint64_t>(o); return v; }
|
||||||
|
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int8& v)
|
||||||
|
{ o.pack_fix_int8(v); return o; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int16& v)
|
||||||
|
{ o.pack_fix_int16(v); return o; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int32& v)
|
||||||
|
{ o.pack_fix_int32(v); return o; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_int64& v)
|
||||||
|
{ o.pack_fix_int64(v); return o; }
|
||||||
|
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint8& v)
|
||||||
|
{ o.pack_fix_uint8(v); return o; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint16& v)
|
||||||
|
{ o.pack_fix_uint16(v); return o; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint32& v)
|
||||||
|
{ o.pack_fix_uint32(v); return o; }
|
||||||
|
|
||||||
|
template <typename Stream>
|
||||||
|
inline packer<Stream>& operator<< (packer<Stream>& o, const type::fix_uint64& v)
|
||||||
|
{ o.pack_fix_uint64(v); return o; }
|
||||||
|
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_int8 v)
|
||||||
|
{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_int16 v)
|
||||||
|
{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_int32 v)
|
||||||
|
{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_int64 v)
|
||||||
|
{ v.get() < 0 ? o.type = type::NEGATIVE_INTEGER, o.via.i64 = v.get() : o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_uint8 v)
|
||||||
|
{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_uint16 v)
|
||||||
|
{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_uint32 v)
|
||||||
|
{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
inline void operator<< (object& o, type::fix_uint64 v)
|
||||||
|
{ o.type = type::POSITIVE_INTEGER, o.via.u64 = v.get(); }
|
||||||
|
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_int8 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_int16 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_int32 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_int64 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_uint8 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_uint16 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_uint32 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
inline void operator<< (object::with_zone& o, type::fix_uint64 v)
|
||||||
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace msgpack
|
||||||
|
|
||||||
|
#endif /* msgpack/type/fixint.hpp */
|
||||||
|
|
@@ -30,7 +30,7 @@ namespace msgpack {
|
|||||||
inline float& operator>> (object o, float& v)
|
inline float& operator>> (object o, float& v)
|
||||||
{
|
{
|
||||||
if(o.type != type::DOUBLE) { throw type_error(); }
|
if(o.type != type::DOUBLE) { throw type_error(); }
|
||||||
v = o.via.dec;
|
v = (float)o.via.dec;
|
||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -60,7 +60,7 @@ inline packer<Stream>& operator<< (packer<Stream>& o, const double& v)
|
|||||||
inline void operator<< (object& o, float v)
|
inline void operator<< (object& o, float v)
|
||||||
{
|
{
|
||||||
o.type = type::DOUBLE;
|
o.type = type::DOUBLE;
|
||||||
o.via.dec = v;
|
o.via.dec = (double)v;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void operator<< (object& o, double v)
|
inline void operator<< (object& o, double v)
|
@@ -35,11 +35,11 @@ namespace detail {
|
|||||||
if(o.type == type::POSITIVE_INTEGER) {
|
if(o.type == type::POSITIVE_INTEGER) {
|
||||||
if(o.via.u64 > (uint64_t)std::numeric_limits<T>::max())
|
if(o.via.u64 > (uint64_t)std::numeric_limits<T>::max())
|
||||||
{ throw type_error(); }
|
{ throw type_error(); }
|
||||||
return o.via.u64;
|
return (T)o.via.u64;
|
||||||
} else if(o.type == type::NEGATIVE_INTEGER) {
|
} else if(o.type == type::NEGATIVE_INTEGER) {
|
||||||
if(o.via.i64 < (int64_t)std::numeric_limits<T>::min())
|
if(o.via.i64 < (int64_t)std::numeric_limits<T>::min())
|
||||||
{ throw type_error(); }
|
{ throw type_error(); }
|
||||||
return o.via.i64;
|
return (T)o.via.i64;
|
||||||
}
|
}
|
||||||
throw type_error();
|
throw type_error();
|
||||||
}
|
}
|
||||||
@@ -51,7 +51,7 @@ namespace detail {
|
|||||||
if(o.type == type::POSITIVE_INTEGER) {
|
if(o.type == type::POSITIVE_INTEGER) {
|
||||||
if(o.via.u64 > (uint64_t)std::numeric_limits<T>::max())
|
if(o.via.u64 > (uint64_t)std::numeric_limits<T>::max())
|
||||||
{ throw type_error(); }
|
{ throw type_error(); }
|
||||||
return o.via.u64;
|
return (T)o.via.u64;
|
||||||
}
|
}
|
||||||
throw type_error();
|
throw type_error();
|
||||||
}
|
}
|
@@ -51,6 +51,14 @@ inline void operator<< (object::with_zone& o, type::nil v)
|
|||||||
{ static_cast<object&>(o) << v; }
|
{ static_cast<object&>(o) << v; }
|
||||||
|
|
||||||
|
|
||||||
|
template <>
|
||||||
|
inline void object::as<void>() const
|
||||||
|
{
|
||||||
|
msgpack::type::nil v;
|
||||||
|
convert(&v);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
} // namespace msgpack
|
} // namespace msgpack
|
||||||
|
|
||||||
#endif /* msgpack/type/nil.hpp */
|
#endif /* msgpack/type/nil.hpp */
|
@@ -33,25 +33,25 @@ struct raw_ref {
|
|||||||
uint32_t size;
|
uint32_t size;
|
||||||
const char* ptr;
|
const char* ptr;
|
||||||
|
|
||||||
std::string str() { return std::string(ptr, size); }
|
std::string str() const { return std::string(ptr, size); }
|
||||||
|
|
||||||
bool operator== (const raw_ref& x)
|
bool operator== (const raw_ref& x) const
|
||||||
{
|
{
|
||||||
return size == x.size && memcmp(ptr, x.ptr, size) == 0;
|
return size == x.size && memcmp(ptr, x.ptr, size) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator!= (const raw_ref& x)
|
bool operator!= (const raw_ref& x) const
|
||||||
{
|
{
|
||||||
return !(*this != x);
|
return !(*this != x);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator< (const raw_ref& x)
|
bool operator< (const raw_ref& x) const
|
||||||
{
|
{
|
||||||
if(size == x.size) { return memcmp(ptr, x.ptr, size) < 0; }
|
if(size == x.size) { return memcmp(ptr, x.ptr, size) < 0; }
|
||||||
else { return size < x.size; }
|
else { return size < x.size; }
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator> (const raw_ref& x)
|
bool operator> (const raw_ref& x) const
|
||||||
{
|
{
|
||||||
if(size == x.size) { return memcmp(ptr, x.ptr, size) > 0; }
|
if(size == x.size) { return memcmp(ptr, x.ptr, size) > 0; }
|
||||||
else { return size > x.size; }
|
else { return size > x.size; }
|
@@ -44,7 +44,7 @@ inline void operator<< (object::with_zone& o, const std::string& v)
|
|||||||
o.type = type::RAW;
|
o.type = type::RAW;
|
||||||
char* ptr = (char*)o.zone->malloc(v.size());
|
char* ptr = (char*)o.zone->malloc(v.size());
|
||||||
o.via.raw.ptr = ptr;
|
o.via.raw.ptr = ptr;
|
||||||
o.via.raw.size = v.size();
|
o.via.raw.size = (uint32_t)v.size();
|
||||||
memcpy(ptr, v.data(), v.size());
|
memcpy(ptr, v.data(), v.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -52,7 +52,7 @@ inline void operator<< (object& o, const std::string& v)
|
|||||||
{
|
{
|
||||||
o.type = type::RAW;
|
o.type = type::RAW;
|
||||||
o.via.raw.ptr = v.data();
|
o.via.raw.ptr = v.data();
|
||||||
o.via.raw.size = v.size();
|
o.via.raw.size = (uint32_t)v.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@@ -141,7 +141,7 @@ type::tuple<A0<%1.upto(i) {|j|%>, A<%=j%><%}%>>& operator>> (
|
|||||||
if(o.type != type::ARRAY) { throw type_error(); }
|
if(o.type != type::ARRAY) { throw type_error(); }
|
||||||
if(o.via.array.size < <%=i+1%>) { throw type_error(); }
|
if(o.via.array.size < <%=i+1%>) { throw type_error(); }
|
||||||
<%0.upto(i) {|j|%>
|
<%0.upto(i) {|j|%>
|
||||||
o.via.array.ptr[<%=j%>].convert<A<%=j%>>(&v.template get<<%=j%>>());<%}%>
|
o.via.array.ptr[<%=j%>].convert<typename type::tuple_type<A<%=j%>>::type>(&v.template get<<%=j%>>());<%}%>
|
||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
<%}%>
|
<%}%>
|
||||||
@@ -187,5 +187,20 @@ inline void operator<< (
|
|||||||
|
|
||||||
} // namespace msgpack
|
} // namespace msgpack
|
||||||
|
|
||||||
|
|
||||||
|
//inline std::ostream& operator<< (std::ostream& o, const msgpack::type::tuple<>& v) {
|
||||||
|
// return o << "[]";
|
||||||
|
//}
|
||||||
|
//<%0.upto(GENERATION_LIMIT) {|i|%>
|
||||||
|
//template <typename A0<%1.upto(i) {|j|%>, typename A<%=j%><%}%>>
|
||||||
|
//inline std::ostream& operator<< (std::ostream& o,
|
||||||
|
// const msgpack::type::tuple<A0<%1.upto(i) {|j|%>, A<%=j%><%}%>>& v) {
|
||||||
|
// return o << "["
|
||||||
|
// <%0.upto(i) {|j|%>
|
||||||
|
// <<<%if j != 0 then%> ", " <<<%end%> v.template get<<%=j%>>()<%}%>
|
||||||
|
// << "]";
|
||||||
|
//}
|
||||||
|
//<%}%>
|
||||||
|
|
||||||
#endif /* msgpack/type/tuple.hpp */
|
#endif /* msgpack/type/tuple.hpp */
|
||||||
|
|
260
cpp/src/msgpack/unpack.h
Normal file
260
cpp/src/msgpack/unpack.h
Normal file
@@ -0,0 +1,260 @@
|
|||||||
|
/*
|
||||||
|
* MessagePack for C unpacking routine
|
||||||
|
*
|
||||||
|
* Copyright (C) 2008-2009 FURUHASHI Sadayuki
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
#ifndef MSGPACK_UNPACKER_H__
|
||||||
|
#define MSGPACK_UNPACKER_H__
|
||||||
|
|
||||||
|
#include "zone.h"
|
||||||
|
#include "object.h"
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_unpack Deserializer
|
||||||
|
* @ingroup msgpack
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
|
typedef struct msgpack_unpacked {
|
||||||
|
msgpack_zone* zone;
|
||||||
|
msgpack_object data;
|
||||||
|
} msgpack_unpacked;
|
||||||
|
|
||||||
|
bool msgpack_unpack_next(msgpack_unpacked* result,
|
||||||
|
const char* data, size_t len, size_t* off);
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_unpacker Streaming deserializer
|
||||||
|
* @ingroup msgpack
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
|
typedef struct msgpack_unpacker {
|
||||||
|
char* buffer;
|
||||||
|
size_t used;
|
||||||
|
size_t free;
|
||||||
|
size_t off;
|
||||||
|
size_t parsed;
|
||||||
|
msgpack_zone* z;
|
||||||
|
size_t initial_buffer_size;
|
||||||
|
void* ctx;
|
||||||
|
} msgpack_unpacker;
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef MSGPACK_UNPACKER_INIT_BUFFER_SIZE
|
||||||
|
#define MSGPACK_UNPACKER_INIT_BUFFER_SIZE (64*1024)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes a streaming deserializer.
|
||||||
|
* The initialized deserializer must be destroyed by msgpack_unpacker_destroy(msgpack_unpacker*).
|
||||||
|
*/
|
||||||
|
bool msgpack_unpacker_init(msgpack_unpacker* mpac, size_t initial_buffer_size);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Destroys a streaming deserializer initialized by msgpack_unpacker_init(msgpack_unpacker*, size_t).
|
||||||
|
*/
|
||||||
|
void msgpack_unpacker_destroy(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a streaming deserializer.
|
||||||
|
* The created deserializer must be destroyed by msgpack_unpacker_free(msgpack_unpacker*).
|
||||||
|
*/
|
||||||
|
msgpack_unpacker* msgpack_unpacker_new(size_t initial_buffer_size);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Frees a streaming deserializer created by msgpack_unpacker_new(size_t).
|
||||||
|
*/
|
||||||
|
void msgpack_unpacker_free(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef MSGPACK_UNPACKER_RESERVE_SIZE
|
||||||
|
#define MSGPACK_UNPACKER_RESERVE_SIZE (32*1024)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reserves free space of the internal buffer.
|
||||||
|
* Use this function to fill the internal buffer with
|
||||||
|
* msgpack_unpacker_buffer(msgpack_unpacker*),
|
||||||
|
* msgpack_unpacker_buffer_capacity(const msgpack_unpacker*) and
|
||||||
|
* msgpack_unpacker_buffer_consumed(msgpack_unpacker*).
|
||||||
|
*/
|
||||||
|
static inline bool msgpack_unpacker_reserve_buffer(msgpack_unpacker* mpac, size_t size);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets pointer to the free space of the internal buffer.
|
||||||
|
* Use this function to fill the internal buffer with
|
||||||
|
* msgpack_unpacker_reserve_buffer(msgpack_unpacker*, size_t),
|
||||||
|
* msgpack_unpacker_buffer_capacity(const msgpack_unpacker*) and
|
||||||
|
* msgpack_unpacker_buffer_consumed(msgpack_unpacker*).
|
||||||
|
*/
|
||||||
|
static inline char* msgpack_unpacker_buffer(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets size of the free space of the internal buffer.
|
||||||
|
* Use this function to fill the internal buffer with
|
||||||
|
* msgpack_unpacker_reserve_buffer(msgpack_unpacker*, size_t),
|
||||||
|
* msgpack_unpacker_buffer(const msgpack_unpacker*) and
|
||||||
|
* msgpack_unpacker_buffer_consumed(msgpack_unpacker*).
|
||||||
|
*/
|
||||||
|
static inline size_t msgpack_unpacker_buffer_capacity(const msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Notifies the deserializer that the internal buffer filled.
|
||||||
|
* Use this function to fill the internal buffer with
|
||||||
|
* msgpack_unpacker_reserve_buffer(msgpack_unpacker*, size_t),
|
||||||
|
* msgpack_unpacker_buffer(msgpack_unpacker*) and
|
||||||
|
* msgpack_unpacker_buffer_capacity(const msgpack_unpacker*).
|
||||||
|
*/
|
||||||
|
static inline void msgpack_unpacker_buffer_consumed(msgpack_unpacker* mpac, size_t size);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Deserializes one object.
|
||||||
|
* Returns true if it successes. Otherwise false is returned.
|
||||||
|
* @param pac pointer to an initialized msgpack_unpacked object.
|
||||||
|
*/
|
||||||
|
bool msgpack_unpacker_next(msgpack_unpacker* mpac, msgpack_unpacked* pac);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes a msgpack_unpacked object.
|
||||||
|
* The initialized object must be destroyed by msgpack_unpacked_destroy(msgpack_unpacker*).
|
||||||
|
* Use the object with msgpack_unpacker_next(msgpack_unpacker*, msgpack_unpacked*) or
|
||||||
|
* msgpack_unpack_next(msgpack_unpacked*, const char*, size_t, size_t*).
|
||||||
|
*/
|
||||||
|
static inline void msgpack_unpacked_init(msgpack_unpacked* result);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Destroys a streaming deserializer initialized by msgpack_unpacked().
|
||||||
|
*/
|
||||||
|
static inline void msgpack_unpacked_destroy(msgpack_unpacked* result);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Releases the memory zone from msgpack_unpacked object.
|
||||||
|
* The released zone must be freed by msgpack_zone_free(msgpack_zone*).
|
||||||
|
*/
|
||||||
|
static inline msgpack_zone* msgpack_unpacked_release_zone(msgpack_unpacked* result);
|
||||||
|
|
||||||
|
|
||||||
|
int msgpack_unpacker_execute(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
msgpack_object msgpack_unpacker_data(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
msgpack_zone* msgpack_unpacker_release_zone(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
void msgpack_unpacker_reset_zone(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
void msgpack_unpacker_reset(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
static inline size_t msgpack_unpacker_message_size(const msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
|
// obsolete
|
||||||
|
typedef enum {
|
||||||
|
MSGPACK_UNPACK_SUCCESS = 2,
|
||||||
|
MSGPACK_UNPACK_EXTRA_BYTES = 1,
|
||||||
|
MSGPACK_UNPACK_CONTINUE = 0,
|
||||||
|
MSGPACK_UNPACK_PARSE_ERROR = -1,
|
||||||
|
} msgpack_unpack_return;
|
||||||
|
|
||||||
|
// obsolete
|
||||||
|
msgpack_unpack_return
|
||||||
|
msgpack_unpack(const char* data, size_t len, size_t* off,
|
||||||
|
msgpack_zone* result_zone, msgpack_object* result);
|
||||||
|
|
||||||
|
|
||||||
|
static inline size_t msgpack_unpacker_parsed_size(const msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
bool msgpack_unpacker_flush_zone(msgpack_unpacker* mpac);
|
||||||
|
|
||||||
|
bool msgpack_unpacker_expand_buffer(msgpack_unpacker* mpac, size_t size);
|
||||||
|
|
||||||
|
bool msgpack_unpacker_reserve_buffer(msgpack_unpacker* mpac, size_t size)
|
||||||
|
{
|
||||||
|
if(mpac->free >= size) { return true; }
|
||||||
|
return msgpack_unpacker_expand_buffer(mpac, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
char* msgpack_unpacker_buffer(msgpack_unpacker* mpac)
|
||||||
|
{
|
||||||
|
return mpac->buffer + mpac->used;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t msgpack_unpacker_buffer_capacity(const msgpack_unpacker* mpac)
|
||||||
|
{
|
||||||
|
return mpac->free;
|
||||||
|
}
|
||||||
|
|
||||||
|
void msgpack_unpacker_buffer_consumed(msgpack_unpacker* mpac, size_t size)
|
||||||
|
{
|
||||||
|
mpac->used += size;
|
||||||
|
mpac->free -= size;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t msgpack_unpacker_message_size(const msgpack_unpacker* mpac)
|
||||||
|
{
|
||||||
|
return mpac->parsed - mpac->off + mpac->used;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t msgpack_unpacker_parsed_size(const msgpack_unpacker* mpac)
|
||||||
|
{
|
||||||
|
return mpac->parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void msgpack_unpacked_init(msgpack_unpacked* result)
|
||||||
|
{
|
||||||
|
memset(result, 0, sizeof(msgpack_unpacked));
|
||||||
|
}
|
||||||
|
|
||||||
|
void msgpack_unpacked_destroy(msgpack_unpacked* result)
|
||||||
|
{
|
||||||
|
if(result->zone != NULL) {
|
||||||
|
msgpack_zone_free(result->zone);
|
||||||
|
result->zone = NULL;
|
||||||
|
memset(&result->data, 0, sizeof(msgpack_object));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
msgpack_zone* msgpack_unpacked_release_zone(msgpack_unpacked* result)
|
||||||
|
{
|
||||||
|
if(result->zone != NULL) {
|
||||||
|
msgpack_zone* z = result->zone;
|
||||||
|
result->zone = NULL;
|
||||||
|
return z;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* msgpack/unpack.h */
|
||||||
|
|
@@ -18,14 +18,15 @@
|
|||||||
#ifndef MSGPACK_UNPACK_HPP__
|
#ifndef MSGPACK_UNPACK_HPP__
|
||||||
#define MSGPACK_UNPACK_HPP__
|
#define MSGPACK_UNPACK_HPP__
|
||||||
|
|
||||||
#include "msgpack/unpack.h"
|
#include "unpack.h"
|
||||||
#include "msgpack/object.hpp"
|
#include "object.hpp"
|
||||||
#include "msgpack/zone.hpp"
|
#include "zone.hpp"
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
|
|
||||||
|
// backward compatibility
|
||||||
#ifndef MSGPACK_UNPACKER_DEFAULT_INITIAL_BUFFER_SIZE
|
#ifndef MSGPACK_UNPACKER_DEFAULT_INITIAL_BUFFER_SIZE
|
||||||
#define MSGPACK_UNPACKER_DEFAULT_INITIAL_BUFFER_SIZE (32*1024)
|
#define MSGPACK_UNPACKER_DEFAULT_INITIAL_BUFFER_SIZE MSGPACK_UNPACKER_INIT_BUFFER_SIZE
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace msgpack {
|
namespace msgpack {
|
||||||
@@ -64,12 +65,12 @@ private:
|
|||||||
|
|
||||||
class unpacker : public msgpack_unpacker {
|
class unpacker : public msgpack_unpacker {
|
||||||
public:
|
public:
|
||||||
unpacker(size_t init_buffer_size = MSGPACK_UNPACKER_DEFAULT_INITIAL_BUFFER_SIZE);
|
unpacker(size_t init_buffer_size = MSGPACK_UNPACKER_INIT_BUFFER_SIZE);
|
||||||
~unpacker();
|
~unpacker();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/*! 1. reserve buffer. at least `size' bytes of capacity will be ready */
|
/*! 1. reserve buffer. at least `size' bytes of capacity will be ready */
|
||||||
void reserve_buffer(size_t size);
|
void reserve_buffer(size_t size = MSGPACK_UNPACKER_RESERVE_SIZE);
|
||||||
|
|
||||||
/*! 2. read data to the buffer() up to buffer_capacity() bytes */
|
/*! 2. read data to the buffer() up to buffer_capacity() bytes */
|
||||||
char* buffer();
|
char* buffer();
|
||||||
@@ -160,7 +161,7 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
static bool unpack(unpacked* result,
|
static void unpack(unpacked* result,
|
||||||
const char* data, size_t len, size_t* offset = NULL);
|
const char* data, size_t len, size_t* offset = NULL);
|
||||||
|
|
||||||
|
|
||||||
@@ -257,17 +258,7 @@ inline object unpacker::data()
|
|||||||
|
|
||||||
inline zone* unpacker::release_zone()
|
inline zone* unpacker::release_zone()
|
||||||
{
|
{
|
||||||
if(!msgpack_unpacker_flush_zone(this)) {
|
return static_cast<msgpack::zone*>(msgpack_unpacker_release_zone(static_cast<msgpack_unpacker*>(this)));
|
||||||
throw std::bad_alloc();
|
|
||||||
}
|
|
||||||
|
|
||||||
zone* r = new zone();
|
|
||||||
|
|
||||||
msgpack_zone old = *base::z;
|
|
||||||
*base::z = *r;
|
|
||||||
*static_cast<msgpack_zone*>(r) = old;
|
|
||||||
|
|
||||||
return r;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void unpacker::reset_zone()
|
inline void unpacker::reset_zone()
|
||||||
@@ -312,7 +303,7 @@ inline void unpacker::remove_nonparsed_buffer()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
inline bool unpack(unpacked* result,
|
inline void unpack(unpacked* result,
|
||||||
const char* data, size_t len, size_t* offset)
|
const char* data, size_t len, size_t* offset)
|
||||||
{
|
{
|
||||||
msgpack::object obj;
|
msgpack::object obj;
|
||||||
@@ -326,12 +317,12 @@ inline bool unpack(unpacked* result,
|
|||||||
case UNPACK_SUCCESS:
|
case UNPACK_SUCCESS:
|
||||||
result->get() = obj;
|
result->get() = obj;
|
||||||
result->zone() = z;
|
result->zone() = z;
|
||||||
return false;
|
return;
|
||||||
|
|
||||||
case UNPACK_EXTRA_BYTES:
|
case UNPACK_EXTRA_BYTES:
|
||||||
result->get() = obj;
|
result->get() = obj;
|
||||||
result->zone() = z;
|
result->zone() = z;
|
||||||
return true;
|
return;
|
||||||
|
|
||||||
case UNPACK_CONTINUE:
|
case UNPACK_CONTINUE:
|
||||||
throw unpack_error("insufficient bytes");
|
throw unpack_error("insufficient bytes");
|
40
cpp/src/msgpack/version.h.in
Normal file
40
cpp/src/msgpack/version.h.in
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
* MessagePack for C version information
|
||||||
|
*
|
||||||
|
* Copyright (C) 2008-2009 FURUHASHI Sadayuki
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
#ifndef MSGPACK_VERSION_H__
|
||||||
|
#define MSGPACK_VERSION_H__
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
const char* msgpack_version(void);
|
||||||
|
int msgpack_version_major(void);
|
||||||
|
int msgpack_version_minor(void);
|
||||||
|
|
||||||
|
#define MSGPACK_VERSION "@VERSION@"
|
||||||
|
#define MSGPACK_VERSION_MAJOR @VERSION_MAJOR@
|
||||||
|
#define MSGPACK_VERSION_MINOR @VERSION_MINOR@
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* msgpack/version.h */
|
||||||
|
|
@@ -18,7 +18,8 @@
|
|||||||
#ifndef MSGPACK_VREFBUFFER_H__
|
#ifndef MSGPACK_VREFBUFFER_H__
|
||||||
#define MSGPACK_VREFBUFFER_H__
|
#define MSGPACK_VREFBUFFER_H__
|
||||||
|
|
||||||
#include "msgpack/zone.h"
|
#include "zone.h"
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
#ifndef _WIN32
|
#ifndef _WIN32
|
||||||
#include <sys/uio.h>
|
#include <sys/uio.h>
|
||||||
@@ -29,19 +30,17 @@ struct iovec {
|
|||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef MSGPACK_VREFBUFFER_REF_SIZE
|
|
||||||
#define MSGPACK_VREFBUFFER_REF_SIZE 32
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef MSGPACK_VREFBUFFER_CHUNK_SIZE
|
|
||||||
#define MSGPACK_VREFBUFFER_CHUNK_SIZE 8192
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_vrefbuffer Vectored Referencing buffer
|
||||||
|
* @ingroup msgpack_buffer
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
struct msgpack_vrefbuffer_chunk;
|
struct msgpack_vrefbuffer_chunk;
|
||||||
typedef struct msgpack_vrefbuffer_chunk msgpack_vrefbuffer_chunk;
|
typedef struct msgpack_vrefbuffer_chunk msgpack_vrefbuffer_chunk;
|
||||||
|
|
||||||
@@ -63,10 +62,21 @@ typedef struct msgpack_vrefbuffer {
|
|||||||
} msgpack_vrefbuffer;
|
} msgpack_vrefbuffer;
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef MSGPACK_VREFBUFFER_REF_SIZE
|
||||||
|
#define MSGPACK_VREFBUFFER_REF_SIZE 32
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef MSGPACK_VREFBUFFER_CHUNK_SIZE
|
||||||
|
#define MSGPACK_VREFBUFFER_CHUNK_SIZE 8192
|
||||||
|
#endif
|
||||||
|
|
||||||
bool msgpack_vrefbuffer_init(msgpack_vrefbuffer* vbuf,
|
bool msgpack_vrefbuffer_init(msgpack_vrefbuffer* vbuf,
|
||||||
size_t ref_size, size_t chunk_size);
|
size_t ref_size, size_t chunk_size);
|
||||||
void msgpack_vrefbuffer_destroy(msgpack_vrefbuffer* vbuf);
|
void msgpack_vrefbuffer_destroy(msgpack_vrefbuffer* vbuf);
|
||||||
|
|
||||||
|
static inline msgpack_vrefbuffer* msgpack_vrefbuffer_new(size_t ref_size, size_t chunk_size);
|
||||||
|
static inline void msgpack_vrefbuffer_free(msgpack_vrefbuffer* vbuf);
|
||||||
|
|
||||||
static inline int msgpack_vrefbuffer_write(void* data, const char* buf, unsigned int len);
|
static inline int msgpack_vrefbuffer_write(void* data, const char* buf, unsigned int len);
|
||||||
|
|
||||||
static inline const struct iovec* msgpack_vrefbuffer_vec(const msgpack_vrefbuffer* vref);
|
static inline const struct iovec* msgpack_vrefbuffer_vec(const msgpack_vrefbuffer* vref);
|
||||||
@@ -80,6 +90,28 @@ int msgpack_vrefbuffer_append_ref(msgpack_vrefbuffer* vbuf,
|
|||||||
|
|
||||||
int msgpack_vrefbuffer_migrate(msgpack_vrefbuffer* vbuf, msgpack_vrefbuffer* to);
|
int msgpack_vrefbuffer_migrate(msgpack_vrefbuffer* vbuf, msgpack_vrefbuffer* to);
|
||||||
|
|
||||||
|
void msgpack_vrefbuffer_clear(msgpack_vrefbuffer* vref);
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
|
msgpack_vrefbuffer* msgpack_vrefbuffer_new(size_t ref_size, size_t chunk_size)
|
||||||
|
{
|
||||||
|
msgpack_vrefbuffer* vbuf = (msgpack_vrefbuffer*)malloc(sizeof(msgpack_vrefbuffer));
|
||||||
|
if(!msgpack_vrefbuffer_init(vbuf, ref_size, chunk_size)) {
|
||||||
|
free(vbuf);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
return vbuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
void msgpack_vrefbuffer_free(msgpack_vrefbuffer* vbuf)
|
||||||
|
{
|
||||||
|
if(vbuf == NULL) { return; }
|
||||||
|
msgpack_vrefbuffer_destroy(vbuf);
|
||||||
|
free(vbuf);
|
||||||
|
}
|
||||||
|
|
||||||
int msgpack_vrefbuffer_write(void* data, const char* buf, unsigned int len)
|
int msgpack_vrefbuffer_write(void* data, const char* buf, unsigned int len)
|
||||||
{
|
{
|
||||||
msgpack_vrefbuffer* vbuf = (msgpack_vrefbuffer*)data;
|
msgpack_vrefbuffer* vbuf = (msgpack_vrefbuffer*)data;
|
@@ -18,7 +18,7 @@
|
|||||||
#ifndef MSGPACK_VREFBUFFER_HPP__
|
#ifndef MSGPACK_VREFBUFFER_HPP__
|
||||||
#define MSGPACK_VREFBUFFER_HPP__
|
#define MSGPACK_VREFBUFFER_HPP__
|
||||||
|
|
||||||
#include "msgpack/vrefbuffer.h"
|
#include "vrefbuffer.h"
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
|
|
||||||
namespace msgpack {
|
namespace msgpack {
|
||||||
@@ -78,6 +78,11 @@ public:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void clear()
|
||||||
|
{
|
||||||
|
msgpack_vrefbuffer_clear(this);
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
typedef msgpack_vrefbuffer base;
|
typedef msgpack_vrefbuffer base;
|
||||||
|
|
@@ -18,35 +18,39 @@
|
|||||||
#ifndef MSGPACK_ZBUFFER_H__
|
#ifndef MSGPACK_ZBUFFER_H__
|
||||||
#define MSGPACK_ZBUFFER_H__
|
#define MSGPACK_ZBUFFER_H__
|
||||||
|
|
||||||
#include "msgpack/sysdep.h"
|
#include "sysdep.h"
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <zlib.h>
|
#include <zlib.h>
|
||||||
|
|
||||||
#ifndef MSGPACK_ZBUFFER_INIT_SIZE
|
|
||||||
#define MSGPACK_ZBUFFER_INIT_SIZE 8192
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef MSGPACK_ZBUFFER_RESERVE_SIZE
|
|
||||||
#define MSGPACK_ZBUFFER_RESERVE_SIZE 512
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_zbuffer Compressed buffer
|
||||||
|
* @ingroup msgpack_buffer
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
typedef struct msgpack_zbuffer {
|
typedef struct msgpack_zbuffer {
|
||||||
z_stream stream;
|
z_stream stream;
|
||||||
char* data;
|
char* data;
|
||||||
size_t init_size;
|
size_t init_size;
|
||||||
} msgpack_zbuffer;
|
} msgpack_zbuffer;
|
||||||
|
|
||||||
|
#ifndef MSGPACK_ZBUFFER_INIT_SIZE
|
||||||
|
#define MSGPACK_ZBUFFER_INIT_SIZE 8192
|
||||||
|
#endif
|
||||||
|
|
||||||
static inline bool msgpack_zbuffer_init(msgpack_zbuffer* zbuf,
|
static inline bool msgpack_zbuffer_init(msgpack_zbuffer* zbuf,
|
||||||
int level, size_t init_size);
|
int level, size_t init_size);
|
||||||
static inline void msgpack_zbuffer_destroy(msgpack_zbuffer* zbuf);
|
static inline void msgpack_zbuffer_destroy(msgpack_zbuffer* zbuf);
|
||||||
|
|
||||||
|
static inline msgpack_zbuffer* msgpack_zbuffer_new(int level, size_t init_size);
|
||||||
|
static inline void msgpack_zbuffer_free(msgpack_zbuffer* zbuf);
|
||||||
|
|
||||||
static inline char* msgpack_zbuffer_flush(msgpack_zbuffer* zbuf);
|
static inline char* msgpack_zbuffer_flush(msgpack_zbuffer* zbuf);
|
||||||
|
|
||||||
static inline const char* msgpack_zbuffer_data(const msgpack_zbuffer* zbuf);
|
static inline const char* msgpack_zbuffer_data(const msgpack_zbuffer* zbuf);
|
||||||
@@ -57,6 +61,10 @@ static inline void msgpack_zbuffer_reset_buffer(msgpack_zbuffer* zbuf);
|
|||||||
static inline char* msgpack_zbuffer_release_buffer(msgpack_zbuffer* zbuf);
|
static inline char* msgpack_zbuffer_release_buffer(msgpack_zbuffer* zbuf);
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef MSGPACK_ZBUFFER_RESERVE_SIZE
|
||||||
|
#define MSGPACK_ZBUFFER_RESERVE_SIZE 512
|
||||||
|
#endif
|
||||||
|
|
||||||
static inline int msgpack_zbuffer_write(void* data, const char* buf, unsigned int len);
|
static inline int msgpack_zbuffer_write(void* data, const char* buf, unsigned int len);
|
||||||
|
|
||||||
static inline bool msgpack_zbuffer_expand(msgpack_zbuffer* zbuf);
|
static inline bool msgpack_zbuffer_expand(msgpack_zbuffer* zbuf);
|
||||||
@@ -80,6 +88,23 @@ void msgpack_zbuffer_destroy(msgpack_zbuffer* zbuf)
|
|||||||
free(zbuf->data);
|
free(zbuf->data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
msgpack_zbuffer* msgpack_zbuffer_new(int level, size_t init_size)
|
||||||
|
{
|
||||||
|
msgpack_zbuffer* zbuf = (msgpack_zbuffer*)malloc(sizeof(msgpack_zbuffer));
|
||||||
|
if(!msgpack_zbuffer_init(zbuf, level, init_size)) {
|
||||||
|
free(zbuf);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
return zbuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
void msgpack_zbuffer_free(msgpack_zbuffer* zbuf)
|
||||||
|
{
|
||||||
|
if(zbuf == NULL) { return; }
|
||||||
|
msgpack_zbuffer_destroy(zbuf);
|
||||||
|
free(zbuf);
|
||||||
|
}
|
||||||
|
|
||||||
bool msgpack_zbuffer_expand(msgpack_zbuffer* zbuf)
|
bool msgpack_zbuffer_expand(msgpack_zbuffer* zbuf)
|
||||||
{
|
{
|
||||||
size_t used = (char*)zbuf->stream.next_out - zbuf->data;
|
size_t used = (char*)zbuf->stream.next_out - zbuf->data;
|
||||||
@@ -171,6 +196,8 @@ char* msgpack_zbuffer_release_buffer(msgpack_zbuffer* zbuf)
|
|||||||
return tmp;
|
return tmp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
@@ -18,7 +18,7 @@
|
|||||||
#ifndef MSGPACK_ZBUFFER_HPP__
|
#ifndef MSGPACK_ZBUFFER_HPP__
|
||||||
#define MSGPACK_ZBUFFER_HPP__
|
#define MSGPACK_ZBUFFER_HPP__
|
||||||
|
|
||||||
#include "msgpack/zbuffer.h"
|
#include "zbuffer.h"
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
|
|
||||||
namespace msgpack {
|
namespace msgpack {
|
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* MessagePack for C memory pool implementation
|
* MessagePack for C memory pool implementation
|
||||||
*
|
*
|
||||||
* Copyright (C) 2008-2009 FURUHASHI Sadayuki
|
* Copyright (C) 2008-2010 FURUHASHI Sadayuki
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
@@ -18,13 +18,19 @@
|
|||||||
#ifndef MSGPACK_ZONE_H__
|
#ifndef MSGPACK_ZONE_H__
|
||||||
#define MSGPACK_ZONE_H__
|
#define MSGPACK_ZONE_H__
|
||||||
|
|
||||||
#include "msgpack/sysdep.h"
|
#include "sysdep.h"
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @defgroup msgpack_zone Memory zone
|
||||||
|
* @ingroup msgpack
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
typedef struct msgpack_zone_finalizer {
|
typedef struct msgpack_zone_finalizer {
|
||||||
void (*func)(void* data);
|
void (*func)(void* data);
|
||||||
void* data;
|
void* data;
|
||||||
@@ -67,10 +73,13 @@ static inline void* msgpack_zone_malloc_no_align(msgpack_zone* zone, size_t size
|
|||||||
static inline bool msgpack_zone_push_finalizer(msgpack_zone* zone,
|
static inline bool msgpack_zone_push_finalizer(msgpack_zone* zone,
|
||||||
void (*func)(void* data), void* data);
|
void (*func)(void* data), void* data);
|
||||||
|
|
||||||
|
static inline void msgpack_zone_swap(msgpack_zone* a, msgpack_zone* b);
|
||||||
|
|
||||||
bool msgpack_zone_is_empty(msgpack_zone* zone);
|
bool msgpack_zone_is_empty(msgpack_zone* zone);
|
||||||
|
|
||||||
void msgpack_zone_clear(msgpack_zone* zone);
|
void msgpack_zone_clear(msgpack_zone* zone);
|
||||||
|
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
#ifndef MSGPACK_ZONE_ALIGN
|
#ifndef MSGPACK_ZONE_ALIGN
|
||||||
@@ -122,6 +131,13 @@ bool msgpack_zone_push_finalizer(msgpack_zone* zone,
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void msgpack_zone_swap(msgpack_zone* a, msgpack_zone* b)
|
||||||
|
{
|
||||||
|
msgpack_zone tmp = *a;
|
||||||
|
*a = *b;
|
||||||
|
*b = tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
@@ -1,7 +1,7 @@
|
|||||||
//
|
//
|
||||||
// MessagePack for C++ memory pool
|
// MessagePack for C++ memory pool
|
||||||
//
|
//
|
||||||
// Copyright (C) 2008-2009 FURUHASHI Sadayuki
|
// Copyright (C) 2008-2010 FURUHASHI Sadayuki
|
||||||
//
|
//
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
// you may not use this file except in compliance with the License.
|
// you may not use this file except in compliance with the License.
|
||||||
@@ -18,7 +18,7 @@
|
|||||||
#ifndef MSGPACK_ZONE_HPP__
|
#ifndef MSGPACK_ZONE_HPP__
|
||||||
#define MSGPACK_ZONE_HPP__
|
#define MSGPACK_ZONE_HPP__
|
||||||
|
|
||||||
#include "msgpack/zone.h"
|
#include "zone.h"
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
@@ -43,6 +43,8 @@ public:
|
|||||||
|
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
|
void swap(zone& o);
|
||||||
|
|
||||||
<%0.upto(GENERATION_LIMIT) {|i|%>
|
<%0.upto(GENERATION_LIMIT) {|i|%>
|
||||||
template <typename T<%1.upto(i) {|j|%>, typename A<%=j%><%}%>>
|
template <typename T<%1.upto(i) {|j|%>, typename A<%=j%><%}%>>
|
||||||
T* allocate(<%=(1..i).map{|j|"A#{j} a#{j}"}.join(', ')%>);
|
T* allocate(<%=(1..i).map{|j|"A#{j} a#{j}"}.join(', ')%>);
|
||||||
@@ -111,6 +113,11 @@ inline void zone::clear()
|
|||||||
msgpack_zone_clear(this);
|
msgpack_zone_clear(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline void zone::swap(zone& o)
|
||||||
|
{
|
||||||
|
msgpack_zone_swap(this, &o);
|
||||||
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void zone::object_destructor(void* obj)
|
void zone::object_destructor(void* obj)
|
||||||
{
|
{
|
@@ -164,7 +164,7 @@ void msgpack_object_print(FILE* out, msgpack_object o)
|
|||||||
|
|
||||||
default:
|
default:
|
||||||
// FIXME
|
// FIXME
|
||||||
fprintf(out, "#<UNKNOWN %hu %"PRIu64">", o.type, o.via.u64);
|
fprintf(out, "#<UNKNOWN %i %"PRIu64">", o.type, o.via.u64);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@@ -19,6 +19,10 @@
|
|||||||
#include "msgpack/unpack_define.h"
|
#include "msgpack/unpack_define.h"
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#ifdef _msgpack_atomic_counter_header
|
||||||
|
#include _msgpack_atomic_counter_header
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
msgpack_zone* z;
|
msgpack_zone* z;
|
||||||
@@ -323,7 +327,7 @@ msgpack_object msgpack_unpacker_data(msgpack_unpacker* mpac)
|
|||||||
msgpack_zone* msgpack_unpacker_release_zone(msgpack_unpacker* mpac)
|
msgpack_zone* msgpack_unpacker_release_zone(msgpack_unpacker* mpac)
|
||||||
{
|
{
|
||||||
if(!msgpack_unpacker_flush_zone(mpac)) {
|
if(!msgpack_unpacker_flush_zone(mpac)) {
|
||||||
return false;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
msgpack_zone* r = msgpack_zone_new(MSGPACK_ZONE_CHUNK_SIZE);
|
msgpack_zone* r = msgpack_zone_new(MSGPACK_ZONE_CHUNK_SIZE);
|
||||||
@@ -333,6 +337,7 @@ msgpack_zone* msgpack_unpacker_release_zone(msgpack_unpacker* mpac)
|
|||||||
|
|
||||||
msgpack_zone* old = mpac->z;
|
msgpack_zone* old = mpac->z;
|
||||||
mpac->z = r;
|
mpac->z = r;
|
||||||
|
CTX_CAST(mpac->ctx)->user.z = mpac->z;
|
||||||
|
|
||||||
return old;
|
return old;
|
||||||
}
|
}
|
||||||
@@ -363,20 +368,46 @@ void msgpack_unpacker_reset(msgpack_unpacker* mpac)
|
|||||||
mpac->parsed = 0;
|
mpac->parsed = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool msgpack_unpacker_next(msgpack_unpacker* mpac, msgpack_unpacked* result)
|
||||||
|
{
|
||||||
|
if(result->zone != NULL) {
|
||||||
|
msgpack_zone_free(result->zone);
|
||||||
|
}
|
||||||
|
|
||||||
|
int ret = msgpack_unpacker_execute(mpac);
|
||||||
|
|
||||||
|
if(ret <= 0) {
|
||||||
|
result->zone = NULL;
|
||||||
|
memset(&result->data, 0, sizeof(msgpack_object));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
result->zone = msgpack_unpacker_release_zone(mpac);
|
||||||
|
result->data = msgpack_unpacker_data(mpac);
|
||||||
|
msgpack_unpacker_reset(mpac);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
msgpack_unpack_return
|
msgpack_unpack_return
|
||||||
msgpack_unpack(const char* data, size_t len, size_t* off,
|
msgpack_unpack(const char* data, size_t len, size_t* off,
|
||||||
msgpack_zone* z, msgpack_object* result)
|
msgpack_zone* result_zone, msgpack_object* result)
|
||||||
{
|
{
|
||||||
|
size_t noff = 0;
|
||||||
|
if(off != NULL) { noff = *off; }
|
||||||
|
|
||||||
|
if(len <= noff) {
|
||||||
|
// FIXME
|
||||||
|
return MSGPACK_UNPACK_CONTINUE;
|
||||||
|
}
|
||||||
|
|
||||||
template_context ctx;
|
template_context ctx;
|
||||||
template_init(&ctx);
|
template_init(&ctx);
|
||||||
|
|
||||||
ctx.user.z = z;
|
ctx.user.z = result_zone;
|
||||||
ctx.user.referenced = false;
|
ctx.user.referenced = false;
|
||||||
|
|
||||||
size_t noff = 0;
|
|
||||||
if(off != NULL) { noff = *off; }
|
|
||||||
|
|
||||||
int e = template_execute(&ctx, data, len, &noff);
|
int e = template_execute(&ctx, data, len, &noff);
|
||||||
if(e < 0) {
|
if(e < 0) {
|
||||||
return MSGPACK_UNPACK_PARSE_ERROR;
|
return MSGPACK_UNPACK_PARSE_ERROR;
|
||||||
@@ -397,3 +428,37 @@ msgpack_unpack(const char* data, size_t len, size_t* off,
|
|||||||
return MSGPACK_UNPACK_SUCCESS;
|
return MSGPACK_UNPACK_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool msgpack_unpack_next(msgpack_unpacked* result,
|
||||||
|
const char* data, size_t len, size_t* off)
|
||||||
|
{
|
||||||
|
msgpack_unpacked_destroy(result);
|
||||||
|
|
||||||
|
size_t noff = 0;
|
||||||
|
if(off != NULL) { noff = *off; }
|
||||||
|
|
||||||
|
if(len <= noff) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
msgpack_zone* z = msgpack_zone_new(MSGPACK_ZONE_CHUNK_SIZE);
|
||||||
|
|
||||||
|
template_context ctx;
|
||||||
|
template_init(&ctx);
|
||||||
|
|
||||||
|
ctx.user.z = z;
|
||||||
|
ctx.user.referenced = false;
|
||||||
|
|
||||||
|
int e = template_execute(&ctx, data, len, &noff);
|
||||||
|
if(e <= 0) {
|
||||||
|
msgpack_zone_free(z);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(off != NULL) { *off = noff; }
|
||||||
|
|
||||||
|
result->zone = z;
|
||||||
|
result->data = template_data(&ctx);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
17
cpp/src/version.c
Normal file
17
cpp/src/version.c
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
#include "msgpack.h"
|
||||||
|
|
||||||
|
const char* msgpack_version(void)
|
||||||
|
{
|
||||||
|
return MSGPACK_VERSION;
|
||||||
|
}
|
||||||
|
|
||||||
|
int msgpack_version_major(void)
|
||||||
|
{
|
||||||
|
return MSGPACK_VERSION_MAJOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
int msgpack_version_minor(void)
|
||||||
|
{
|
||||||
|
return MSGPACK_VERSION_MINOR;
|
||||||
|
}
|
||||||
|
|
@@ -75,6 +75,25 @@ void msgpack_vrefbuffer_destroy(msgpack_vrefbuffer* vbuf)
|
|||||||
free(vbuf->array);
|
free(vbuf->array);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void msgpack_vrefbuffer_clear(msgpack_vrefbuffer* vbuf)
|
||||||
|
{
|
||||||
|
msgpack_vrefbuffer_chunk* c = vbuf->inner_buffer.head->next;
|
||||||
|
msgpack_vrefbuffer_chunk* n;
|
||||||
|
while(c != NULL) {
|
||||||
|
n = c->next;
|
||||||
|
free(c);
|
||||||
|
c = n;
|
||||||
|
}
|
||||||
|
|
||||||
|
msgpack_vrefbuffer_inner_buffer* const ib = &vbuf->inner_buffer;
|
||||||
|
msgpack_vrefbuffer_chunk* chunk = ib->head;
|
||||||
|
chunk->next = NULL;
|
||||||
|
ib->free = vbuf->chunk_size;
|
||||||
|
ib->ptr = ((char*)chunk) + sizeof(msgpack_vrefbuffer_chunk);
|
||||||
|
|
||||||
|
vbuf->tail = vbuf->array;
|
||||||
|
}
|
||||||
|
|
||||||
int msgpack_vrefbuffer_append_ref(msgpack_vrefbuffer* vbuf,
|
int msgpack_vrefbuffer_append_ref(msgpack_vrefbuffer* vbuf,
|
||||||
const char* buf, unsigned int len)
|
const char* buf, unsigned int len)
|
||||||
{
|
{
|
@@ -204,7 +204,7 @@ msgpack_zone* msgpack_zone_new(size_t chunk_size)
|
|||||||
|
|
||||||
if(!init_chunk_list(&zone->chunk_list, chunk_size)) {
|
if(!init_chunk_list(&zone->chunk_list, chunk_size)) {
|
||||||
free(zone);
|
free(zone);
|
||||||
return false;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
init_finalizer_array(&zone->finalizer_array);
|
init_finalizer_array(&zone->finalizer_array);
|
||||||
@@ -214,6 +214,7 @@ msgpack_zone* msgpack_zone_new(size_t chunk_size)
|
|||||||
|
|
||||||
void msgpack_zone_free(msgpack_zone* zone)
|
void msgpack_zone_free(msgpack_zone* zone)
|
||||||
{
|
{
|
||||||
|
if(zone == NULL) { return; }
|
||||||
msgpack_zone_destroy(zone);
|
msgpack_zone_destroy(zone);
|
||||||
free(zone);
|
free(zone);
|
||||||
}
|
}
|
@@ -1,9 +0,0 @@
|
|||||||
|
|
||||||
CXXFLAGS += -Wall -g -I. -I.. -O4
|
|
||||||
LDFLAGS +=
|
|
||||||
|
|
||||||
all: test
|
|
||||||
|
|
||||||
test: test.o unpack.o zone.o object.o pack.hpp unpack.hpp zone.hpp object.hpp
|
|
||||||
$(CXX) test.o unpack.o zone.o object.o $(CXXFLAGS) $(LDFLAGS) -o $@
|
|
||||||
|
|
@@ -1,15 +1,23 @@
|
|||||||
|
|
||||||
AM_CPPFLAGS = -I..
|
AM_CPPFLAGS = -I../src
|
||||||
AM_C_CPPFLAGS = -I..
|
AM_C_CPPFLAGS = -I../src
|
||||||
AM_LDFLAGS = ../libmsgpack.la -lgtest_main
|
AM_LDFLAGS = ../src/libmsgpack.la -lgtest_main -pthread
|
||||||
|
|
||||||
check_PROGRAMS = \
|
check_PROGRAMS = \
|
||||||
zone \
|
zone \
|
||||||
pack_unpack \
|
pack_unpack \
|
||||||
|
pack_unpack_c \
|
||||||
streaming \
|
streaming \
|
||||||
|
streaming_c \
|
||||||
object \
|
object \
|
||||||
convert \
|
convert \
|
||||||
buffer
|
buffer \
|
||||||
|
cases \
|
||||||
|
fixint \
|
||||||
|
fixint_c \
|
||||||
|
version \
|
||||||
|
msgpackc_test \
|
||||||
|
msgpack_test
|
||||||
|
|
||||||
TESTS = $(check_PROGRAMS)
|
TESTS = $(check_PROGRAMS)
|
||||||
|
|
||||||
@@ -17,8 +25,12 @@ zone_SOURCES = zone.cc
|
|||||||
|
|
||||||
pack_unpack_SOURCES = pack_unpack.cc
|
pack_unpack_SOURCES = pack_unpack.cc
|
||||||
|
|
||||||
|
pack_unpack_c_SOURCES = pack_unpack_c.cc
|
||||||
|
|
||||||
streaming_SOURCES = streaming.cc
|
streaming_SOURCES = streaming.cc
|
||||||
|
|
||||||
|
streaming_c_SOURCES = streaming_c.cc
|
||||||
|
|
||||||
object_SOURCES = object.cc
|
object_SOURCES = object.cc
|
||||||
|
|
||||||
convert_SOURCES = convert.cc
|
convert_SOURCES = convert.cc
|
||||||
@@ -26,3 +38,17 @@ convert_SOURCES = convert.cc
|
|||||||
buffer_SOURCES = buffer.cc
|
buffer_SOURCES = buffer.cc
|
||||||
buffer_LDADD = -lz
|
buffer_LDADD = -lz
|
||||||
|
|
||||||
|
cases_SOURCES = cases.cc
|
||||||
|
|
||||||
|
fixint_SOURCES = fixint.cc
|
||||||
|
|
||||||
|
fixint_c_SOURCES = fixint_c.cc
|
||||||
|
|
||||||
|
version_SOURCES = version.cc
|
||||||
|
|
||||||
|
msgpackc_test_SOURCES = msgpackc_test.cpp
|
||||||
|
|
||||||
|
msgpack_test_SOURCES = msgpack_test.cpp
|
||||||
|
|
||||||
|
EXTRA_DIST = cases.mpac cases_compact.mpac
|
||||||
|
|
||||||
|
@@ -12,6 +12,14 @@ TEST(buffer, sbuffer)
|
|||||||
|
|
||||||
EXPECT_EQ(3, sbuf.size());
|
EXPECT_EQ(3, sbuf.size());
|
||||||
EXPECT_TRUE( memcmp(sbuf.data(), "aaa", 3) == 0 );
|
EXPECT_TRUE( memcmp(sbuf.data(), "aaa", 3) == 0 );
|
||||||
|
|
||||||
|
sbuf.clear();
|
||||||
|
sbuf.write("a", 1);
|
||||||
|
sbuf.write("a", 1);
|
||||||
|
sbuf.write("a", 1);
|
||||||
|
|
||||||
|
EXPECT_EQ(3, sbuf.size());
|
||||||
|
EXPECT_TRUE( memcmp(sbuf.data(), "aaa", 3) == 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -32,6 +40,23 @@ TEST(buffer, vrefbuffer)
|
|||||||
|
|
||||||
EXPECT_EQ(3, sbuf.size());
|
EXPECT_EQ(3, sbuf.size());
|
||||||
EXPECT_TRUE( memcmp(sbuf.data(), "aaa", 3) == 0 );
|
EXPECT_TRUE( memcmp(sbuf.data(), "aaa", 3) == 0 );
|
||||||
|
|
||||||
|
|
||||||
|
vbuf.clear();
|
||||||
|
vbuf.write("a", 1);
|
||||||
|
vbuf.write("a", 1);
|
||||||
|
vbuf.write("a", 1);
|
||||||
|
|
||||||
|
vec = vbuf.vector();
|
||||||
|
veclen = vbuf.vector_size();
|
||||||
|
|
||||||
|
sbuf.clear();
|
||||||
|
for(size_t i=0; i < veclen; ++i) {
|
||||||
|
sbuf.write((const char*)vec[i].iov_base, vec[i].iov_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
EXPECT_EQ(3, sbuf.size());
|
||||||
|
EXPECT_TRUE( memcmp(sbuf.data(), "aaa", 3) == 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
38
cpp/test/cases.cc
Normal file
38
cpp/test/cases.cc
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
#include <msgpack.hpp>
|
||||||
|
#include <fstream>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
static void feed_file(msgpack::unpacker& pac, const char* path)
|
||||||
|
{
|
||||||
|
std::ifstream fin(path);
|
||||||
|
while(true) {
|
||||||
|
pac.reserve_buffer(32*1024);
|
||||||
|
fin.read(pac.buffer(), pac.buffer_capacity());
|
||||||
|
if(fin.bad()) {
|
||||||
|
throw std::runtime_error("read failed");
|
||||||
|
}
|
||||||
|
pac.buffer_consumed(fin.gcount());
|
||||||
|
if(fin.fail()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(cases, format)
|
||||||
|
{
|
||||||
|
msgpack::unpacker pac;
|
||||||
|
msgpack::unpacker pac_compact;
|
||||||
|
|
||||||
|
feed_file(pac, "cases.mpac");
|
||||||
|
feed_file(pac_compact, "cases_compact.mpac");
|
||||||
|
|
||||||
|
msgpack::unpacked result;
|
||||||
|
while(pac.next(&result)) {
|
||||||
|
msgpack::unpacked result_compact;
|
||||||
|
EXPECT_TRUE( pac_compact.next(&result_compact) );
|
||||||
|
EXPECT_EQ(result_compact.get(), result.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
EXPECT_FALSE( pac_compact.next(&result) );
|
||||||
|
}
|
||||||
|
|
55
cpp/test/fixint.cc
Normal file
55
cpp/test/fixint.cc
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
#include <msgpack.hpp>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
void check_size(size_t size) {
|
||||||
|
T v(0);
|
||||||
|
msgpack::sbuffer sbuf;
|
||||||
|
msgpack::pack(sbuf, v);
|
||||||
|
EXPECT_EQ(size, sbuf.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(fixint, size)
|
||||||
|
{
|
||||||
|
check_size<msgpack::type::fix_int8>(2);
|
||||||
|
check_size<msgpack::type::fix_int16>(3);
|
||||||
|
check_size<msgpack::type::fix_int32>(5);
|
||||||
|
check_size<msgpack::type::fix_int64>(9);
|
||||||
|
|
||||||
|
check_size<msgpack::type::fix_uint8>(2);
|
||||||
|
check_size<msgpack::type::fix_uint16>(3);
|
||||||
|
check_size<msgpack::type::fix_uint32>(5);
|
||||||
|
check_size<msgpack::type::fix_uint64>(9);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
void check_convert() {
|
||||||
|
T v1(-11);
|
||||||
|
msgpack::sbuffer sbuf;
|
||||||
|
msgpack::pack(sbuf, v1);
|
||||||
|
|
||||||
|
msgpack::unpacked msg;
|
||||||
|
msgpack::unpack(&msg, sbuf.data(), sbuf.size());
|
||||||
|
|
||||||
|
T v2;
|
||||||
|
msg.get().convert(&v2);
|
||||||
|
|
||||||
|
EXPECT_EQ(v1.get(), v2.get());
|
||||||
|
|
||||||
|
EXPECT_EQ(msg.get(), msgpack::object(T(v1.get())));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(fixint, convert)
|
||||||
|
{
|
||||||
|
check_convert<msgpack::type::fix_int8>();
|
||||||
|
check_convert<msgpack::type::fix_int16>();
|
||||||
|
check_convert<msgpack::type::fix_int32>();
|
||||||
|
check_convert<msgpack::type::fix_int64>();
|
||||||
|
|
||||||
|
check_convert<msgpack::type::fix_uint8>();
|
||||||
|
check_convert<msgpack::type::fix_uint16>();
|
||||||
|
check_convert<msgpack::type::fix_uint32>();
|
||||||
|
check_convert<msgpack::type::fix_uint64>();
|
||||||
|
}
|
||||||
|
|
32
cpp/test/fixint_c.cc
Normal file
32
cpp/test/fixint_c.cc
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
#include <msgpack.hpp>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
TEST(fixint, size)
|
||||||
|
{
|
||||||
|
msgpack_sbuffer* sbuf = msgpack_sbuffer_new();
|
||||||
|
msgpack_packer* pk = msgpack_packer_new(sbuf, msgpack_sbuffer_write);
|
||||||
|
|
||||||
|
size_t sum = 0;
|
||||||
|
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_int8(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=2, sbuf->size);
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_int16(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=3, sbuf->size);
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_int32(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=5, sbuf->size);
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_int64(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=9, sbuf->size);
|
||||||
|
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_uint8(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=2, sbuf->size);
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_uint16(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=3, sbuf->size);
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_uint32(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=5, sbuf->size);
|
||||||
|
EXPECT_EQ(0, msgpack_pack_fix_uint64(pk, 0));
|
||||||
|
EXPECT_EQ(sum+=9, sbuf->size);
|
||||||
|
|
||||||
|
msgpack_sbuffer_free(sbuf);
|
||||||
|
msgpack_packer_free(pk);
|
||||||
|
}
|
||||||
|
|
@@ -77,21 +77,17 @@ TEST(unpack, sequence)
|
|||||||
msgpack::pack(sbuf, 2);
|
msgpack::pack(sbuf, 2);
|
||||||
msgpack::pack(sbuf, 3);
|
msgpack::pack(sbuf, 3);
|
||||||
|
|
||||||
bool cont;
|
|
||||||
size_t offset = 0;
|
size_t offset = 0;
|
||||||
|
|
||||||
msgpack::unpacked msg;
|
msgpack::unpacked msg;
|
||||||
|
|
||||||
cont = msgpack::unpack(&msg, sbuf.data(), sbuf.size(), &offset);
|
msgpack::unpack(&msg, sbuf.data(), sbuf.size(), &offset);
|
||||||
EXPECT_TRUE(cont);
|
|
||||||
EXPECT_EQ(1, msg.get().as<int>());
|
EXPECT_EQ(1, msg.get().as<int>());
|
||||||
|
|
||||||
cont = msgpack::unpack(&msg, sbuf.data(), sbuf.size(), &offset);
|
msgpack::unpack(&msg, sbuf.data(), sbuf.size(), &offset);
|
||||||
EXPECT_TRUE(cont);
|
|
||||||
EXPECT_EQ(2, msg.get().as<int>());
|
EXPECT_EQ(2, msg.get().as<int>());
|
||||||
|
|
||||||
cont = msgpack::unpack(&msg, sbuf.data(), sbuf.size(), &offset);
|
msgpack::unpack(&msg, sbuf.data(), sbuf.size(), &offset);
|
||||||
EXPECT_FALSE(cont);
|
|
||||||
EXPECT_EQ(3, msg.get().as<int>());
|
EXPECT_EQ(3, msg.get().as<int>());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
70
cpp/test/pack_unpack_c.cc
Normal file
70
cpp/test/pack_unpack_c.cc
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
#include <msgpack.h>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
TEST(pack, num)
|
||||||
|
{
|
||||||
|
msgpack_sbuffer* sbuf = msgpack_sbuffer_new();
|
||||||
|
msgpack_packer* pk = msgpack_packer_new(sbuf, msgpack_sbuffer_write);
|
||||||
|
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 1));
|
||||||
|
|
||||||
|
msgpack_sbuffer_free(sbuf);
|
||||||
|
msgpack_packer_free(pk);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TEST(pack, array)
|
||||||
|
{
|
||||||
|
msgpack_sbuffer* sbuf = msgpack_sbuffer_new();
|
||||||
|
msgpack_packer* pk = msgpack_packer_new(sbuf, msgpack_sbuffer_write);
|
||||||
|
|
||||||
|
EXPECT_EQ(0, msgpack_pack_array(pk, 3));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 1));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 2));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 3));
|
||||||
|
|
||||||
|
msgpack_sbuffer_free(sbuf);
|
||||||
|
msgpack_packer_free(pk);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TEST(unpack, sequence)
|
||||||
|
{
|
||||||
|
msgpack_sbuffer* sbuf = msgpack_sbuffer_new();
|
||||||
|
msgpack_packer* pk = msgpack_packer_new(sbuf, msgpack_sbuffer_write);
|
||||||
|
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 1));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 2));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 3));
|
||||||
|
|
||||||
|
msgpack_packer_free(pk);
|
||||||
|
|
||||||
|
bool success;
|
||||||
|
size_t offset = 0;
|
||||||
|
|
||||||
|
msgpack_unpacked msg;
|
||||||
|
msgpack_unpacked_init(&msg);
|
||||||
|
|
||||||
|
success = msgpack_unpack_next(&msg, sbuf->data, sbuf->size, &offset);
|
||||||
|
EXPECT_TRUE(success);
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_POSITIVE_INTEGER, msg.data.type);
|
||||||
|
EXPECT_EQ(1, msg.data.via.u64);
|
||||||
|
|
||||||
|
success = msgpack_unpack_next(&msg, sbuf->data, sbuf->size, &offset);
|
||||||
|
EXPECT_TRUE(success);
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_POSITIVE_INTEGER, msg.data.type);
|
||||||
|
EXPECT_EQ(2, msg.data.via.u64);
|
||||||
|
|
||||||
|
success = msgpack_unpack_next(&msg, sbuf->data, sbuf->size, &offset);
|
||||||
|
EXPECT_TRUE(success);
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_POSITIVE_INTEGER, msg.data.type);
|
||||||
|
EXPECT_EQ(3, msg.data.via.u64);
|
||||||
|
|
||||||
|
success = msgpack_unpack_next(&msg, sbuf->data, sbuf->size, &offset);
|
||||||
|
EXPECT_FALSE(success);
|
||||||
|
|
||||||
|
msgpack_sbuffer_free(sbuf);
|
||||||
|
msgpack_unpacked_destroy(&msg);
|
||||||
|
}
|
||||||
|
|
@@ -2,28 +2,33 @@
|
|||||||
#include <gtest/gtest.h>
|
#include <gtest/gtest.h>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
|
|
||||||
TEST(streaming, basic)
|
TEST(streaming, basic)
|
||||||
{
|
{
|
||||||
std::ostringstream stream;
|
msgpack::sbuffer buffer;
|
||||||
msgpack::packer<std::ostream> pk(&stream);
|
|
||||||
|
|
||||||
|
msgpack::packer<msgpack::sbuffer> pk(&buffer);
|
||||||
pk.pack(1);
|
pk.pack(1);
|
||||||
pk.pack(2);
|
pk.pack(2);
|
||||||
pk.pack(3);
|
pk.pack(3);
|
||||||
|
|
||||||
std::istringstream input(stream.str());
|
const char* input = buffer.data();
|
||||||
|
const char* const eof = input + buffer.size();
|
||||||
|
|
||||||
msgpack::unpacker pac;
|
msgpack::unpacker pac;
|
||||||
|
msgpack::unpacked result;
|
||||||
|
|
||||||
int count = 0;
|
int count = 0;
|
||||||
while(count < 3) {
|
while(count < 3) {
|
||||||
pac.reserve_buffer(32*1024);
|
pac.reserve_buffer(32*1024);
|
||||||
|
|
||||||
size_t len = input.readsome(pac.buffer(), pac.buffer_capacity());
|
// read buffer into pac.buffer() upto
|
||||||
|
// pac.buffer_capacity() bytes.
|
||||||
|
size_t len = 1;
|
||||||
|
memcpy(pac.buffer(), input, len);
|
||||||
|
input += len;
|
||||||
|
|
||||||
pac.buffer_consumed(len);
|
pac.buffer_consumed(len);
|
||||||
|
|
||||||
msgpack::unpacked result;
|
|
||||||
while(pac.next(&result)) {
|
while(pac.next(&result)) {
|
||||||
msgpack::object obj = result.get();
|
msgpack::object obj = result.get();
|
||||||
switch(count++) {
|
switch(count++) {
|
||||||
@@ -38,6 +43,8 @@ TEST(streaming, basic)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
EXPECT_TRUE(input < eof);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
98
cpp/test/streaming_c.cc
Normal file
98
cpp/test/streaming_c.cc
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
#include <msgpack.h>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
TEST(streaming, basic)
|
||||||
|
{
|
||||||
|
msgpack_sbuffer* buffer = msgpack_sbuffer_new();
|
||||||
|
|
||||||
|
msgpack_packer* pk = msgpack_packer_new(buffer, msgpack_sbuffer_write);
|
||||||
|
|
||||||
|
// 1, 2, 3, "raw", ["data"], {0.3: 0.4}
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 1));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 2));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_int(pk, 3));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_raw(pk, 3));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_raw_body(pk, "raw", 3));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_array(pk, 1));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_raw(pk, 4));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_raw_body(pk, "data", 4));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_map(pk, 1));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_float(pk, 0.4));
|
||||||
|
EXPECT_EQ(0, msgpack_pack_double(pk, 0.8));
|
||||||
|
int max_count = 6;
|
||||||
|
|
||||||
|
msgpack_packer_free(pk);
|
||||||
|
|
||||||
|
const char* input = buffer->data;
|
||||||
|
const char* const eof = input + buffer->size;
|
||||||
|
|
||||||
|
msgpack_unpacker pac;
|
||||||
|
msgpack_unpacker_init(&pac, MSGPACK_UNPACKER_INIT_BUFFER_SIZE);
|
||||||
|
|
||||||
|
msgpack_unpacked result;
|
||||||
|
msgpack_unpacked_init(&result);
|
||||||
|
|
||||||
|
int count = 0;
|
||||||
|
while(count < max_count) {
|
||||||
|
bool unpacked = false;
|
||||||
|
|
||||||
|
msgpack_unpacker_reserve_buffer(&pac, 32*1024);
|
||||||
|
|
||||||
|
while(!unpacked) {
|
||||||
|
/* read buffer into msgpack_unapcker_buffer(&pac) upto
|
||||||
|
* msgpack_unpacker_buffer_capacity(&pac) bytes. */
|
||||||
|
memcpy(msgpack_unpacker_buffer(&pac), input, 1);
|
||||||
|
input += 1;
|
||||||
|
|
||||||
|
EXPECT_TRUE(input <= eof);
|
||||||
|
|
||||||
|
msgpack_unpacker_buffer_consumed(&pac, 1);
|
||||||
|
|
||||||
|
while(msgpack_unpacker_next(&pac, &result)) {
|
||||||
|
unpacked = 1;
|
||||||
|
msgpack_object obj = result.data;
|
||||||
|
msgpack_object e;
|
||||||
|
switch(count++) {
|
||||||
|
case 0:
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_POSITIVE_INTEGER, obj.type);
|
||||||
|
EXPECT_EQ(1, obj.via.u64);
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_POSITIVE_INTEGER, obj.type);
|
||||||
|
EXPECT_EQ(2, obj.via.u64);
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_POSITIVE_INTEGER, obj.type);
|
||||||
|
EXPECT_EQ(3, obj.via.u64);
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_RAW, obj.type);
|
||||||
|
EXPECT_EQ(std::string("raw",3), std::string(obj.via.raw.ptr, obj.via.raw.size));
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_ARRAY, obj.type);
|
||||||
|
EXPECT_EQ(1, obj.via.array.size);
|
||||||
|
e = obj.via.array.ptr[0];
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_RAW, e.type);
|
||||||
|
EXPECT_EQ(std::string("data",4), std::string(e.via.raw.ptr, e.via.raw.size));
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_MAP, obj.type);
|
||||||
|
EXPECT_EQ(1, obj.via.map.size);
|
||||||
|
e = obj.via.map.ptr[0].key;
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_DOUBLE, e.type);
|
||||||
|
ASSERT_FLOAT_EQ(0.4, (float)e.via.dec);
|
||||||
|
e = obj.via.map.ptr[0].val;
|
||||||
|
EXPECT_EQ(MSGPACK_OBJECT_DOUBLE, e.type);
|
||||||
|
ASSERT_DOUBLE_EQ(0.8, e.via.dec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
msgpack_unpacker_destroy(&pac);
|
||||||
|
msgpack_unpacked_destroy(&result);
|
||||||
|
}
|
||||||
|
|
13
cpp/test/version.cc
Normal file
13
cpp/test/version.cc
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
#include <msgpack.hpp>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
TEST(version, print)
|
||||||
|
{
|
||||||
|
printf("MSGPACK_VERSION : %s\n", MSGPACK_VERSION);
|
||||||
|
printf("MSGPACK_VERSION_MAJOR : %d\n", MSGPACK_VERSION_MAJOR);
|
||||||
|
printf("MSGPACK_VERSION_MINOR : %d\n", MSGPACK_VERSION_MINOR);
|
||||||
|
printf("msgpack_version() : %s\n", msgpack_version());
|
||||||
|
printf("msgpack_version_major() : %d\n", msgpack_version_major());
|
||||||
|
printf("msgpack_version_minor() : %d\n", msgpack_version_minor());
|
||||||
|
}
|
||||||
|
|
5
erlang/.gitignore
vendored
Normal file
5
erlang/.gitignore
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
MANIFEST
|
||||||
|
*.beam
|
||||||
|
.omakedb*
|
||||||
|
*.omc
|
||||||
|
*~
|
51
erlang/OMakefile
Normal file
51
erlang/OMakefile
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
########################################################################
|
||||||
|
# Permission is hereby granted, free of charge, to any person
|
||||||
|
# obtaining a copy of this file, to deal in the File without
|
||||||
|
# restriction, including without limitation the rights to use,
|
||||||
|
# copy, modify, merge, publish, distribute, sublicense, and/or
|
||||||
|
# sell copies of the File, and to permit persons to whom the
|
||||||
|
# File is furnished to do so, subject to the following condition:
|
||||||
|
#
|
||||||
|
# THE FILE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
||||||
|
# OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
# IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
||||||
|
# DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
||||||
|
# OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE FILE OR
|
||||||
|
# THE USE OR OTHER DEALINGS IN THE FILE.
|
||||||
|
|
||||||
|
########################################################################
|
||||||
|
# The standard OMakefile.
|
||||||
|
# You will usually need to modify this file for your project.
|
||||||
|
|
||||||
|
########################################################################
|
||||||
|
# Phony targets are scoped, so you probably want to declare them first.
|
||||||
|
#
|
||||||
|
|
||||||
|
.PHONY: all clean test edoc dialyzer #install
|
||||||
|
|
||||||
|
########################################################################
|
||||||
|
# Subdirectories.
|
||||||
|
# You may want to include some subdirectories in this project.
|
||||||
|
# If so, define the subdirectory targets and uncomment this section.
|
||||||
|
#
|
||||||
|
|
||||||
|
.DEFAULT: msgpack.beam
|
||||||
|
|
||||||
|
msgpack.beam: msgpack.erl
|
||||||
|
erlc -Wall +debug_info $<
|
||||||
|
|
||||||
|
msgpack.html: msgpack.erl
|
||||||
|
erl -noshell -run edoc_run file $<
|
||||||
|
|
||||||
|
test: msgpack.beam
|
||||||
|
erl -noshell -s msgpack test -s init stop
|
||||||
|
|
||||||
|
edoc: msgpack.erl
|
||||||
|
erl -noshell -eval 'ok=edoc:files(["msgpack.erl"], [{dir, "edoc"}]).' -s init stop
|
||||||
|
|
||||||
|
dialyzer: msgpack.erl
|
||||||
|
dialyzer --src $<
|
||||||
|
|
||||||
|
clean:
|
||||||
|
-rm -f *.beam *.html
|
45
erlang/OMakeroot
Normal file
45
erlang/OMakeroot
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
########################################################################
|
||||||
|
# Permission is hereby granted, free of charge, to any person
|
||||||
|
# obtaining a copy of this file, to deal in the File without
|
||||||
|
# restriction, including without limitation the rights to use,
|
||||||
|
# copy, modify, merge, publish, distribute, sublicense, and/or
|
||||||
|
# sell copies of the File, and to permit persons to whom the
|
||||||
|
# File is furnished to do so, subject to the following condition:
|
||||||
|
#
|
||||||
|
# THE FILE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
||||||
|
# OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
# IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
||||||
|
# DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
||||||
|
# OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE FILE OR
|
||||||
|
# THE USE OR OTHER DEALINGS IN THE FILE.
|
||||||
|
|
||||||
|
########################################################################
|
||||||
|
# The standard OMakeroot file.
|
||||||
|
# You will not normally need to modify this file.
|
||||||
|
# By default, your changes should be placed in the
|
||||||
|
# OMakefile in this directory.
|
||||||
|
#
|
||||||
|
# If you decide to modify this file, note that it uses exactly
|
||||||
|
# the same syntax as the OMakefile.
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# Include the standard installed configuration files.
|
||||||
|
# Any of these can be deleted if you are not using them,
|
||||||
|
# but you probably want to keep the Common file.
|
||||||
|
#
|
||||||
|
open build/C
|
||||||
|
open build/OCaml
|
||||||
|
open build/LaTeX
|
||||||
|
|
||||||
|
#
|
||||||
|
# The command-line variables are defined *after* the
|
||||||
|
# standard configuration has been loaded.
|
||||||
|
#
|
||||||
|
DefineCommandVars()
|
||||||
|
|
||||||
|
#
|
||||||
|
# Include the OMakefile in this directory.
|
||||||
|
#
|
||||||
|
.SUBDIRS: .
|
9
erlang/README.md
Normal file
9
erlang/README.md
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
MessagePack for Erlang
|
||||||
|
======================
|
||||||
|
Binary-based efficient object serialization library.
|
||||||
|
|
||||||
|
see wiki ( http://redmine.msgpack.org/projects/msgpack/wiki/QuickStartErlang ) for details
|
||||||
|
|
||||||
|
# Status
|
||||||
|
|
||||||
|
0.1.0 released.
|
4
erlang/edoc/.gitignore
vendored
Normal file
4
erlang/edoc/.gitignore
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
*.html
|
||||||
|
*.css
|
||||||
|
*.png
|
||||||
|
edoc-info
|
395
erlang/msgpack.erl
Normal file
395
erlang/msgpack.erl
Normal file
@@ -0,0 +1,395 @@
|
|||||||
|
%%
|
||||||
|
%% MessagePack for Erlang
|
||||||
|
%%
|
||||||
|
%% Copyright (C) 2009-2010 UENISHI Kota
|
||||||
|
%%
|
||||||
|
%% Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
%% you may not use this file except in compliance with the License.
|
||||||
|
%% You may obtain a copy of the License at
|
||||||
|
%%
|
||||||
|
%% http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
%%
|
||||||
|
%% Unless required by applicable law or agreed to in writing, software
|
||||||
|
%% distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
%% See the License for the specific language governing permissions and
|
||||||
|
%% limitations under the License.
|
||||||
|
|
||||||
|
|
||||||
|
%% @doc <a href="http://msgpack.org/">MessagePack</a> codec for Erlang.
|
||||||
|
%%
|
||||||
|
%% APIs are almost compatible with <a href="http://redmine.msgpack.org/projects/msgpack/wiki/QuickStartC">C API</a>
|
||||||
|
%% except for buffering functions (both copying and zero-copying), which are unavailable.
|
||||||
|
%%
|
||||||
|
%% <table border="1">
|
||||||
|
%% <caption>Equivalence between Erlang and <a href="http://msgpack.sourceforge.jp/spec">Msgpack type</a> :</caption>
|
||||||
|
%% <tr><th> erlang </th><th> msgpack </th></tr>
|
||||||
|
%% <tr><td> integer() </td><td> pos_fixnum/neg_fixnum/uint8/uint16/uint32/uint64/int8/int16/int32/int64 </td></tr>
|
||||||
|
%% <tr><td> float() </td><td> float/double </td></tr>
|
||||||
|
%% <tr><td> nil </td><td> nil </td></tr>
|
||||||
|
%% <tr><td> boolean() </td><td> boolean </td></tr>
|
||||||
|
%% <tr><td> binary() </td><td> fix_raw/raw16/raw32 </td></tr>
|
||||||
|
%% <tr><td> list() </td><td> fix_array/array16/array32 </td></tr>
|
||||||
|
%% <tr><td> {proplist()} </td><td> fix_map/map16/map32 </td></tr>
|
||||||
|
%% </table>
|
||||||
|
%% @end
|
||||||
|
|
||||||
|
-module(msgpack).
|
||||||
|
-author('kuenishi+msgpack@gmail.com').
|
||||||
|
|
||||||
|
-export([pack/1, unpack/1, unpack_all/1]).
|
||||||
|
|
||||||
|
% @type msgpack_term() = [msgpack_term()]
|
||||||
|
% | {[{msgpack_term(),msgpack_term()}]}
|
||||||
|
% | integer() | float() | binary().
|
||||||
|
% Erlang representation of msgpack data.
|
||||||
|
-type msgpack_term() :: [msgpack_term()]
|
||||||
|
| {[{msgpack_term(),msgpack_term()}]}
|
||||||
|
| integer() | float() | binary().
|
||||||
|
|
||||||
|
|
||||||
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||||
|
% external APIs
|
||||||
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||||
|
|
||||||
|
% @doc Encode an erlang term into an msgpack binary.
|
||||||
|
% Returns {error, {badarg, term()}} if the input is illegal.
|
||||||
|
% @spec pack(Term::msgpack_term()) -> binary() | {error, {badarg, term()}}
|
||||||
|
-spec pack(Term::msgpack_term()) -> binary() | {error, {badarg, term()}}.
|
||||||
|
pack(Term)->
|
||||||
|
try
|
||||||
|
pack_(Term)
|
||||||
|
catch
|
||||||
|
throw:Exception ->
|
||||||
|
{error, Exception}
|
||||||
|
end.
|
||||||
|
|
||||||
|
% @doc Decode an msgpack binary into an erlang term.
|
||||||
|
% It only decodes the first msgpack packet contained in the binary; the rest is returned as is.
|
||||||
|
% Returns {error, {badarg, term()}} if the input is corrupted.
|
||||||
|
% Returns {error, incomplete} if the input is not a full msgpack packet (caller should gather more data and try again).
|
||||||
|
% @spec unpack(Bin::binary()) -> {msgpack_term(), binary()} | {error, incomplete} | {error, {badarg, term()}}
|
||||||
|
-spec unpack(Bin::binary()) -> {msgpack_term(), binary()} | {error, incomplete} | {error, {badarg, term()}}.
|
||||||
|
unpack(Bin) when is_binary(Bin) ->
|
||||||
|
try
|
||||||
|
unpack_(Bin)
|
||||||
|
catch
|
||||||
|
throw:Exception ->
|
||||||
|
{error, Exception}
|
||||||
|
end;
|
||||||
|
unpack(Other) ->
|
||||||
|
{error, {badarg, Other}}.
|
||||||
|
|
||||||
|
% @doc Decode an msgpack binary into an erlang terms.
|
||||||
|
% It only decodes ALL msgpack packets contained in the binary. No packets should not remain.
|
||||||
|
% Returns {error, {badarg, term()}} if the input is corrupted.
|
||||||
|
% Returns {error, incomplete} if the input is not a full msgpack packet (caller should gather more data and try again).
|
||||||
|
% @spec unpack_all(binary()) -> [msgpack_term()] | {error, incomplete} | {error, {badarg, term()}}
|
||||||
|
-spec unpack_all(binary()) -> [msgpack_term()] | {error, incomplete} | {error, {badarg, term()}}.
|
||||||
|
unpack_all(Data)->
|
||||||
|
try
|
||||||
|
unpack_all_(Data)
|
||||||
|
catch
|
||||||
|
throw:Exception ->
|
||||||
|
{error, Exception}
|
||||||
|
end.
|
||||||
|
unpack_all_(Data)->
|
||||||
|
case unpack_(Data) of
|
||||||
|
{ Term, <<>> } ->
|
||||||
|
[Term];
|
||||||
|
{ Term, Binary } when is_binary(Binary) ->
|
||||||
|
[Term|unpack_all_(Binary)]
|
||||||
|
end.
|
||||||
|
|
||||||
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||||
|
% internal APIs
|
||||||
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||||
|
|
||||||
|
% pack them all
|
||||||
|
-spec pack_(msgpack_term()) -> binary() | no_return().
|
||||||
|
pack_(I) when is_integer(I) andalso I < 0 ->
|
||||||
|
pack_int_(I);
|
||||||
|
pack_(I) when is_integer(I) ->
|
||||||
|
pack_uint_(I);
|
||||||
|
pack_(F) when is_float(F) ->
|
||||||
|
pack_double(F);
|
||||||
|
pack_(nil) ->
|
||||||
|
<< 16#C0:8 >>;
|
||||||
|
pack_(true) ->
|
||||||
|
<< 16#C3:8 >>;
|
||||||
|
pack_(false) ->
|
||||||
|
<< 16#C2:8 >>;
|
||||||
|
pack_(Bin) when is_binary(Bin) ->
|
||||||
|
pack_raw(Bin);
|
||||||
|
pack_(List) when is_list(List) ->
|
||||||
|
pack_array(List);
|
||||||
|
pack_({Map}) when is_list(Map) ->
|
||||||
|
pack_map(Map);
|
||||||
|
pack_(Other) ->
|
||||||
|
throw({badarg, Other}).
|
||||||
|
|
||||||
|
|
||||||
|
-spec pack_uint_(non_neg_integer()) -> binary().
|
||||||
|
% positive fixnum
|
||||||
|
pack_uint_(N) when N < 128 ->
|
||||||
|
<< 2#0:1, N:7 >>;
|
||||||
|
% uint 8
|
||||||
|
pack_uint_(N) when N < 256 ->
|
||||||
|
<< 16#CC:8, N:8 >>;
|
||||||
|
% uint 16
|
||||||
|
pack_uint_(N) when N < 65536 ->
|
||||||
|
<< 16#CD:8, N:16/big-unsigned-integer-unit:1 >>;
|
||||||
|
% uint 32
|
||||||
|
pack_uint_(N) when N < 16#FFFFFFFF->
|
||||||
|
<< 16#CE:8, N:32/big-unsigned-integer-unit:1 >>;
|
||||||
|
% uint 64
|
||||||
|
pack_uint_(N) ->
|
||||||
|
<< 16#CF:8, N:64/big-unsigned-integer-unit:1 >>.
|
||||||
|
|
||||||
|
-spec pack_int_(integer()) -> binary().
|
||||||
|
% negative fixnum
|
||||||
|
pack_int_(N) when N >= -32->
|
||||||
|
<< 2#111:3, N:5 >>;
|
||||||
|
% int 8
|
||||||
|
pack_int_(N) when N > -128 ->
|
||||||
|
<< 16#D0:8, N:8/big-signed-integer-unit:1 >>;
|
||||||
|
% int 16
|
||||||
|
pack_int_(N) when N > -32768 ->
|
||||||
|
<< 16#D1:8, N:16/big-signed-integer-unit:1 >>;
|
||||||
|
% int 32
|
||||||
|
pack_int_(N) when N > -16#FFFFFFFF ->
|
||||||
|
<< 16#D2:8, N:32/big-signed-integer-unit:1 >>;
|
||||||
|
% int 64
|
||||||
|
pack_int_(N) ->
|
||||||
|
<< 16#D3:8, N:64/big-signed-integer-unit:1 >>.
|
||||||
|
|
||||||
|
|
||||||
|
-spec pack_double(float()) -> binary().
|
||||||
|
% float : erlang's float is always IEEE 754 64bit format.
|
||||||
|
% pack_float(F) when is_float(F)->
|
||||||
|
% << 16#CA:8, F:32/big-float-unit:1 >>.
|
||||||
|
% pack_double(F).
|
||||||
|
% double
|
||||||
|
pack_double(F) ->
|
||||||
|
<< 16#CB:8, F:64/big-float-unit:1 >>.
|
||||||
|
|
||||||
|
|
||||||
|
-spec pack_raw(binary()) -> binary().
|
||||||
|
% raw bytes
|
||||||
|
pack_raw(Bin) ->
|
||||||
|
case byte_size(Bin) of
|
||||||
|
Len when Len < 6->
|
||||||
|
<< 2#101:3, Len:5, Bin/binary >>;
|
||||||
|
Len when Len < 16#10000 -> % 65536
|
||||||
|
<< 16#DA:8, Len:16/big-unsigned-integer-unit:1, Bin/binary >>;
|
||||||
|
Len ->
|
||||||
|
<< 16#DB:8, Len:32/big-unsigned-integer-unit:1, Bin/binary >>
|
||||||
|
end.
|
||||||
|
|
||||||
|
|
||||||
|
-spec pack_array([msgpack_term()]) -> binary() | no_return().
|
||||||
|
% list
|
||||||
|
pack_array(L) ->
|
||||||
|
case length(L) of
|
||||||
|
Len when Len < 16 ->
|
||||||
|
<< 2#1001:4, Len:4/integer-unit:1, (pack_array_(L, <<>>))/binary >>;
|
||||||
|
Len when Len < 16#10000 -> % 65536
|
||||||
|
<< 16#DC:8, Len:16/big-unsigned-integer-unit:1, (pack_array_(L, <<>>))/binary >>;
|
||||||
|
Len ->
|
||||||
|
<< 16#DD:8, Len:32/big-unsigned-integer-unit:1, (pack_array_(L, <<>>))/binary >>
|
||||||
|
end.
|
||||||
|
|
||||||
|
pack_array_([], Acc) -> Acc;
|
||||||
|
pack_array_([Head|Tail], Acc) ->
|
||||||
|
pack_array_(Tail, <<Acc/binary, (pack_(Head))/binary>>).
|
||||||
|
|
||||||
|
% Users SHOULD NOT send too long list: this uses lists:reverse/1
|
||||||
|
-spec unpack_array_(binary(), non_neg_integer(), [msgpack_term()]) -> {[msgpack_term()], binary()} | no_return().
|
||||||
|
unpack_array_(Bin, 0, Acc) -> {lists:reverse(Acc), Bin};
|
||||||
|
unpack_array_(Bin, Len, Acc) ->
|
||||||
|
{Term, Rest} = unpack_(Bin),
|
||||||
|
unpack_array_(Rest, Len-1, [Term|Acc]).
|
||||||
|
|
||||||
|
|
||||||
|
-spec pack_map(M::[{msgpack_term(),msgpack_term()}]) -> binary() | no_return().
|
||||||
|
pack_map(M)->
|
||||||
|
case length(M) of
|
||||||
|
Len when Len < 16 ->
|
||||||
|
<< 2#1000:4, Len:4/integer-unit:1, (pack_map_(M, <<>>))/binary >>;
|
||||||
|
Len when Len < 16#10000 -> % 65536
|
||||||
|
<< 16#DE:8, Len:16/big-unsigned-integer-unit:1, (pack_map_(M, <<>>))/binary >>;
|
||||||
|
Len ->
|
||||||
|
<< 16#DF:8, Len:32/big-unsigned-integer-unit:1, (pack_map_(M, <<>>))/binary >>
|
||||||
|
end.
|
||||||
|
|
||||||
|
pack_map_([], Acc) -> Acc;
|
||||||
|
pack_map_([{Key,Value}|Tail], Acc) ->
|
||||||
|
pack_map_(Tail, << Acc/binary, (pack_(Key))/binary, (pack_(Value))/binary>>).
|
||||||
|
|
||||||
|
% Users SHOULD NOT send too long list: this uses lists:reverse/1
|
||||||
|
-spec unpack_map_(binary(), non_neg_integer(), [{msgpack_term(), msgpack_term()}]) ->
|
||||||
|
{{[{msgpack_term(), msgpack_term()}]}, binary()} | no_return().
|
||||||
|
unpack_map_(Bin, 0, Acc) -> {{lists:reverse(Acc)}, Bin};
|
||||||
|
unpack_map_(Bin, Len, Acc) ->
|
||||||
|
{Key, Rest} = unpack_(Bin),
|
||||||
|
{Value, Rest2} = unpack_(Rest),
|
||||||
|
unpack_map_(Rest2, Len-1, [{Key,Value}|Acc]).
|
||||||
|
|
||||||
|
% unpack them all
|
||||||
|
-spec unpack_(Bin::binary()) -> {msgpack_term(), binary()} | no_return().
|
||||||
|
unpack_(Bin) ->
|
||||||
|
case Bin of
|
||||||
|
% ATOMS
|
||||||
|
<<16#C0, Rest/binary>> -> {nil, Rest};
|
||||||
|
<<16#C2, Rest/binary>> -> {false, Rest};
|
||||||
|
<<16#C3, Rest/binary>> -> {true, Rest};
|
||||||
|
% Floats
|
||||||
|
<<16#CA, V:32/float-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#CB, V:64/float-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
% Unsigned integers
|
||||||
|
<<16#CC, V:8/unsigned-integer, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#CD, V:16/big-unsigned-integer-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#CE, V:32/big-unsigned-integer-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#CF, V:64/big-unsigned-integer-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
% Signed integers
|
||||||
|
<<16#D0, V:8/signed-integer, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#D1, V:16/big-signed-integer-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#D2, V:32/big-signed-integer-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#D3, V:64/big-signed-integer-unit:1, Rest/binary>> -> {V, Rest};
|
||||||
|
% Raw bytes
|
||||||
|
<<16#DA, L:16/unsigned-integer-unit:1, V:L/binary, Rest/binary>> -> {V, Rest};
|
||||||
|
<<16#DB, L:32/unsigned-integer-unit:1, V:L/binary, Rest/binary>> -> {V, Rest};
|
||||||
|
% Arrays
|
||||||
|
<<16#DC, L:16/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_array_(Rest, L, []);
|
||||||
|
<<16#DD, L:32/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_array_(Rest, L, []);
|
||||||
|
% Maps
|
||||||
|
<<16#DE, L:16/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_map_(Rest, L, []);
|
||||||
|
<<16#DF, L:32/big-unsigned-integer-unit:1, Rest/binary>> -> unpack_map_(Rest, L, []);
|
||||||
|
|
||||||
|
% Tag-encoded lengths (kept last, for speed)
|
||||||
|
<<0:1, V:7, Rest/binary>> -> {V, Rest}; % positive int
|
||||||
|
<<2#111:3, V:5, Rest/binary>> -> {V - 2#100000, Rest}; % negative int
|
||||||
|
<<2#101:3, L:5, V:L/binary, Rest/binary>> -> {V, Rest}; % raw bytes
|
||||||
|
<<2#1001:4, L:4, Rest/binary>> -> unpack_array_(Rest, L, []); % array
|
||||||
|
<<2#1000:4, L:4, Rest/binary>> -> unpack_map_(Rest, L, []); % map
|
||||||
|
|
||||||
|
% Invalid data
|
||||||
|
<<F, R/binary>> when F==16#C1;
|
||||||
|
F==16#C4; F==16#C5; F==16#C6; F==16#C7; F==16#C8; F==16#C9;
|
||||||
|
F==16#D4; F==16#D5; F==16#D6; F==16#D7; F==16#D8; F==16#D9 ->
|
||||||
|
throw({badarg, <<F, R/binary>>});
|
||||||
|
% Incomplete data (we've covered every complete/invalid case; anything left is incomplete)
|
||||||
|
_ ->
|
||||||
|
throw(incomplete)
|
||||||
|
end.
|
||||||
|
|
||||||
|
|
||||||
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||||
|
% unit tests
|
||||||
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||||
|
-include_lib("eunit/include/eunit.hrl").
|
||||||
|
-ifdef(EUNIT).
|
||||||
|
|
||||||
|
compare_all([], [])-> ok;
|
||||||
|
compare_all([], R)-> {toomuchrhs, R};
|
||||||
|
compare_all(L, [])-> {toomuchlhs, L};
|
||||||
|
compare_all([LH|LTL], [RH|RTL]) ->
|
||||||
|
?assertEqual(LH, RH),
|
||||||
|
compare_all(LTL, RTL).
|
||||||
|
|
||||||
|
port_receive(Port) ->
|
||||||
|
port_receive(Port, <<>>).
|
||||||
|
port_receive(Port, Acc) ->
|
||||||
|
receive
|
||||||
|
{Port, {data, Data}} -> port_receive(Port, <<Acc/binary, Data/binary>>);
|
||||||
|
{Port, eof} -> Acc
|
||||||
|
after 1000 -> Acc
|
||||||
|
end.
|
||||||
|
|
||||||
|
test_([]) -> 0;
|
||||||
|
test_([Term|Rest])->
|
||||||
|
Pack = msgpack:pack(Term),
|
||||||
|
?assertEqual({Term, <<>>}, msgpack:unpack( Pack )),
|
||||||
|
1+test_(Rest).
|
||||||
|
|
||||||
|
test_data()->
|
||||||
|
[true, false, nil,
|
||||||
|
0, 1, 2, 123, 512, 1230, 678908, 16#FFFFFFFFFF,
|
||||||
|
-1, -23, -512, -1230, -567898, -16#FFFFFFFFFF,
|
||||||
|
123.123, -234.4355, 1.0e-34, 1.0e64,
|
||||||
|
[23, 234, 0.23],
|
||||||
|
<<"hogehoge">>, <<"243546rf7g68h798j", 0, 23, 255>>,
|
||||||
|
<<"hoasfdafdas][">>,
|
||||||
|
[0,42, <<"sum">>, [1,2]], [1,42, nil, [3]],
|
||||||
|
-234, -40000, -16#10000000, -16#100000000,
|
||||||
|
42
|
||||||
|
].
|
||||||
|
|
||||||
|
basic_test()->
|
||||||
|
Tests = test_data(),
|
||||||
|
Passed = test_(Tests),
|
||||||
|
Passed = length(Tests).
|
||||||
|
|
||||||
|
port_test()->
|
||||||
|
Tests = test_data(),
|
||||||
|
?assertEqual({[Tests],<<>>}, msgpack:unpack(msgpack:pack([Tests]))),
|
||||||
|
|
||||||
|
Port = open_port({spawn, "ruby ../test/crosslang.rb"}, [binary, eof]),
|
||||||
|
true = port_command(Port, msgpack:pack(Tests)),
|
||||||
|
?assertEqual({Tests, <<>>}, msgpack:unpack(port_receive(Port))),
|
||||||
|
port_close(Port).
|
||||||
|
|
||||||
|
test_p(Len,Term,OrigBin,Len) ->
|
||||||
|
{Term, <<>>}=msgpack:unpack(OrigBin);
|
||||||
|
test_p(I,_,OrigBin,Len) when I < Len->
|
||||||
|
<<Bin:I/binary, _/binary>> = OrigBin,
|
||||||
|
?assertEqual({error,incomplete}, msgpack:unpack(Bin)).
|
||||||
|
|
||||||
|
partial_test()-> % error handling test.
|
||||||
|
Term = lists:seq(0, 45),
|
||||||
|
Bin=msgpack:pack(Term),
|
||||||
|
BinLen = byte_size(Bin),
|
||||||
|
[test_p(X, Term, Bin, BinLen) || X <- lists:seq(0,BinLen)].
|
||||||
|
|
||||||
|
long_test()->
|
||||||
|
Longer = lists:seq(0, 655),
|
||||||
|
{Longer, <<>>} = msgpack:unpack(msgpack:pack(Longer)).
|
||||||
|
|
||||||
|
map_test()->
|
||||||
|
Ints = lists:seq(0, 65),
|
||||||
|
Map = {[ {X, X*2} || X <- Ints ] ++ [{<<"hage">>, 324}, {43542, [nil, true, false]}]},
|
||||||
|
{Map2, <<>>} = msgpack:unpack(msgpack:pack(Map)),
|
||||||
|
?assertEqual(Map, Map2),
|
||||||
|
ok.
|
||||||
|
|
||||||
|
unknown_test()->
|
||||||
|
Port = open_port({spawn, "ruby testcase_generator.rb"}, [binary, eof]),
|
||||||
|
Tests = [0, 1, 2, 123, 512, 1230, 678908,
|
||||||
|
-1, -23, -512, -1230, -567898,
|
||||||
|
<<"hogehoge">>, <<"243546rf7g68h798j">>,
|
||||||
|
123.123,
|
||||||
|
-234.4355, 1.0e-34, 1.0e64,
|
||||||
|
[23, 234, 0.23],
|
||||||
|
[0,42,<<"sum">>, [1,2]], [1,42, nil, [3]],
|
||||||
|
{[{1,2},{<<"hoge">>,nil}]}, % map
|
||||||
|
-234, -50000,
|
||||||
|
42
|
||||||
|
],
|
||||||
|
?assertEqual(ok, compare_all(Tests, msgpack:unpack_all(port_receive(Port)))),
|
||||||
|
port_close(Port).
|
||||||
|
|
||||||
|
other_test()->
|
||||||
|
?assertEqual({error,incomplete},msgpack:unpack(<<>>)).
|
||||||
|
|
||||||
|
benchmark_test()->
|
||||||
|
Data=[test_data() || _ <- lists:seq(0, 10000)],
|
||||||
|
S=?debugTime(" serialize", msgpack:pack(Data)),
|
||||||
|
{Data,<<>>}=?debugTime("deserialize", msgpack:unpack(S)),
|
||||||
|
?debugFmt("for ~p KB test data.", [byte_size(S) div 1024]).
|
||||||
|
|
||||||
|
error_test()->
|
||||||
|
?assertEqual({error,{badarg, atom}}, msgpack:pack(atom)),
|
||||||
|
Term = {"hoge", "hage", atom},
|
||||||
|
?assertEqual({error,{badarg, Term}}, msgpack:pack(Term)).
|
||||||
|
|
||||||
|
-endif.
|
65
erlang/testcase_generator.rb
Normal file
65
erlang/testcase_generator.rb
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
begin
|
||||||
|
require 'rubygems'
|
||||||
|
rescue LoadError
|
||||||
|
end
|
||||||
|
require 'msgpack'
|
||||||
|
|
||||||
|
def usage
|
||||||
|
puts <<EOF
|
||||||
|
Usage: #{$0} [out-file]
|
||||||
|
|
||||||
|
This tool is for testing of accepting MessagePack random-term.
|
||||||
|
This does following behavior:
|
||||||
|
|
||||||
|
1. serializes the objects in this file, using Ruby implementation
|
||||||
|
of MessagePack (Note that Ruby implementation is considered valid)
|
||||||
|
2. Writes the serialized binaries into <out-file> (default: stdout)
|
||||||
|
|
||||||
|
EOF
|
||||||
|
exit 1
|
||||||
|
end
|
||||||
|
|
||||||
|
code = 1
|
||||||
|
outio = $stdout
|
||||||
|
|
||||||
|
if ARGV.length > 2
|
||||||
|
usage
|
||||||
|
end
|
||||||
|
|
||||||
|
if fname = ARGV[0]
|
||||||
|
unless fname == "-"
|
||||||
|
begin
|
||||||
|
outio = File.open(fname, "w")
|
||||||
|
rescue
|
||||||
|
puts "can't open output file: #{$!}"
|
||||||
|
exit 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
objs = [0, 1, 2, 123, 512, 1230, 678908,
|
||||||
|
-1, -23, -512, -1230, -567898,
|
||||||
|
"hogehoge", "243546rf7g68h798j",
|
||||||
|
123.123,
|
||||||
|
-234.4355, 1.0e-34, 1.0e64,
|
||||||
|
[23, 234, 0.23],
|
||||||
|
[0,42,"sum", [1,2]], [1,42, nil, [3]],
|
||||||
|
{ 1 => 2, "hoge" => nil },
|
||||||
|
-234, -50000,
|
||||||
|
42
|
||||||
|
]
|
||||||
|
begin
|
||||||
|
objs.each do |obj|
|
||||||
|
outio.write MessagePack.pack(obj)
|
||||||
|
outio.flush
|
||||||
|
end
|
||||||
|
rescue EOFError
|
||||||
|
code=0
|
||||||
|
rescue
|
||||||
|
$stderr.puts $!
|
||||||
|
code=1
|
||||||
|
end
|
||||||
|
|
||||||
|
outio.close
|
||||||
|
exit code
|
||||||
|
|
11
go/Makefile
Normal file
11
go/Makefile
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
include $(GOROOT)/src/Make.inc
|
||||||
|
|
||||||
|
TARG=msgpack
|
||||||
|
|
||||||
|
GOFILES=pack.go unpack.go
|
||||||
|
|
||||||
|
include $(GOROOT)/src/Make.pkg
|
||||||
|
|
||||||
|
%: install %.go
|
||||||
|
$(GC) $*.go
|
||||||
|
$(LD) -o $@ $*.$O
|
218
go/msgpack_test.go
Normal file
218
go/msgpack_test.go
Normal file
@@ -0,0 +1,218 @@
|
|||||||
|
package msgpack_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
. "msgpack"
|
||||||
|
"testing"
|
||||||
|
"bytes"
|
||||||
|
"reflect"
|
||||||
|
"math"
|
||||||
|
);
|
||||||
|
|
||||||
|
func equal(lhs reflect.Value, rhs reflect.Value) bool {
|
||||||
|
{
|
||||||
|
_rhs, ok := rhs.(*reflect.InterfaceValue)
|
||||||
|
if ok { return equal(lhs, _rhs.Elem()) }
|
||||||
|
}
|
||||||
|
switch _lhs := lhs.(type) {
|
||||||
|
case *reflect.InterfaceValue:
|
||||||
|
return equal(_lhs.Elem(), rhs)
|
||||||
|
case *reflect.BoolValue:
|
||||||
|
_rhs, ok := rhs.(*reflect.BoolValue)
|
||||||
|
return ok && _lhs.Get() == _rhs.Get()
|
||||||
|
case *reflect.UintValue:
|
||||||
|
_rhs, ok := rhs.(*reflect.UintValue)
|
||||||
|
return ok && _lhs.Get() == _rhs.Get()
|
||||||
|
case *reflect.IntValue:
|
||||||
|
_rhs, ok := rhs.(*reflect.IntValue)
|
||||||
|
return ok && _lhs.Get() == _rhs.Get()
|
||||||
|
case *reflect.FloatValue:
|
||||||
|
_rhs, ok := rhs.(*reflect.FloatValue)
|
||||||
|
return ok && _lhs.Get() == _rhs.Get()
|
||||||
|
case reflect.ArrayOrSliceValue:
|
||||||
|
_rhs := rhs.(reflect.ArrayOrSliceValue)
|
||||||
|
if _lhs.Len() != _rhs.Len() { return false; }
|
||||||
|
for i := 0; i < _lhs.Len(); i++ {
|
||||||
|
if !equal(_lhs.Elem(i), _rhs.Elem(i)) { return false; }
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
case *reflect.MapValue:
|
||||||
|
_rhs := rhs.(*reflect.MapValue)
|
||||||
|
if _lhs.Len() != _rhs.Len() { return false; }
|
||||||
|
for _, k := range _lhs.Keys() {
|
||||||
|
lv, rv := _lhs.Elem(k), _rhs.Elem(k)
|
||||||
|
if lv == nil || rv == nil || !equal(lv, rv) { return false; }
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackUint8(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []uint8 { 0, 1, 2, 125, 126, 127, 128, 253, 254, 255 } {
|
||||||
|
_, err := PackUint8(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0x00, 0x01, 0x02, 0x7d, 0x7e, 0x7f, 0xcc, 0x80, 0xcc, 0xfd, 0xcc, 0xfe, 0xcc, 0xff }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackUint16(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []uint16 { 0, 1, 2, 125, 126, 127, 128, 253, 254, 255, 256, 65533, 65534, 65535 } {
|
||||||
|
_, err := PackUint16(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0x00, 0x01, 0x02, 0x7d, 0x7e, 0x7f, 0xcc, 0x80, 0xcc, 0xfd, 0xcc, 0xfe, 0xcc, 0xff, 0xcd, 0x01, 0x00, 0xcd, 0xff, 0xfd, 0xcd, 0xff, 0xfe, 0xcd, 0xff, 0xff }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackUint32(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []uint32 { 0, 1, 2, 125, 126, 127, 128, 253, 254, 255, 256, 65533, 65534, 65535, 65536, 4294967293, 4294967294, 4294967295 } {
|
||||||
|
_, err := PackUint32(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0x00, 0x01, 0x02, 0x7d, 0x7e, 0x7f, 0xcc, 0x80, 0xcc, 0xfd, 0xcc, 0xfe, 0xcc, 0xff, 0xcd, 0x01, 0x00, 0xcd, 0xff, 0xfd, 0xcd, 0xff, 0xfe, 0xcd, 0xff, 0xff, 0xce, 0x00, 0x01, 0x00, 0x00, 0xce, 0xff, 0xff, 0xff, 0xfd, 0xce, 0xff, 0xff, 0xff, 0xfe, 0xce, 0xff, 0xff, 0xff, 0xff }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackUint64(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []uint64 { 0, 1, 2, 125, 126, 127, 128, 253, 254, 255, 256, 65533, 65534, 65535, 65536, 4294967293, 4294967294, 4294967295, 4294967296, 18446744073709551613, 18446744073709551614, 18446744073709551615 } {
|
||||||
|
_, err := PackUint64(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0x00, 0x01, 0x02, 0x7d, 0x7e, 0x7f, 0xcc, 0x80, 0xcc, 0xfd, 0xcc, 0xfe, 0xcc, 0xff, 0xcd, 0x01, 0x00, 0xcd, 0xff, 0xfd, 0xcd, 0xff, 0xfe, 0xcd, 0xff, 0xff, 0xce, 0x00, 0x01, 0x00, 0x00, 0xce, 0xff, 0xff, 0xff, 0xfd, 0xce, 0xff, 0xff, 0xff, 0xfe, 0xce, 0xff, 0xff, 0xff, 0xff, 0xcf, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0xcf, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfd, 0xcf, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe, 0xcf, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackInt8(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []int8 { -128, -127, -34, -33, -32, -31, 0, 1, 126, 127 } {
|
||||||
|
_, err := PackInt8(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0xd0, 0x80, 0xd0, 0x81, 0xd0, 0xde, 0xd0, 0xdf, 0xe0, 0xe1, 0x00, 0x01, 0x7e, 0x7f }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackInt16(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []int16 { -32768, -32767, -131, -130, -129, -128, -127, -34, -33, -32, -31, 0, 1, 126, 127, 128, 129, 130, 32765, 32766, 32767 } {
|
||||||
|
_, err := PackInt16(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0xd1, 0x80, 0x00, 0xd1, 0x80, 0x01, 0xd1, 0xff, 0x7d, 0xd1, 0xff, 0x7e, 0xd1, 0xff, 0x7f, 0xd0, 0x80, 0xd0, 0x81, 0xd0, 0xde, 0xd0, 0xdf, 0xe0, 0xe1, 0x00, 0x01, 0x7e, 0x7f, 0xd1, 0x00, 0x80, 0xd1, 0x00, 0x81, 0xd1, 0x00, 0x82, 0xd1, 0x7f, 0xfd, 0xd1, 0x7f, 0xfe, 0xd1, 0x7f, 0xff }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackInt32(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []int32 { -2147483648, -2147483647, -2147483646, -32771, -32770, -32769, -32768, -32767, -131, -130, -129, -128, -127, -34, -33, -32, -31, 0, 1, 126, 127, 128, 129, 130, 32765, 32766, 32767, 32768, 32769, 32770, 2147483645, 2147483646, 2147483647 } {
|
||||||
|
_, err := PackInt32(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0xd2, 0x80, 0x00, 0x00, 0x00, 0xd2, 0x80, 0x00, 0x00, 0x01, 0xd2, 0x80, 0x00, 0x00, 0x02, 0xd2, 0xff, 0xff, 0x7f, 0xfd, 0xd2, 0xff, 0xff, 0x7f, 0xfe, 0xd2, 0xff, 0xff, 0x7f, 0xff, 0xd1, 0x80, 0x00, 0xd1, 0x80, 0x01, 0xd1, 0xff, 0x7d, 0xd1, 0xff, 0x7e, 0xd1, 0xff, 0x7f, 0xd0, 0x80, 0xd0, 0x81, 0xd0, 0xde, 0xd0, 0xdf, 0xe0, 0xe1, 0x00, 0x01, 0x7e, 0x7f, 0xd1, 0x00, 0x80, 0xd1, 0x00, 0x81, 0xd1, 0x00, 0x82, 0xd1, 0x7f, 0xfd, 0xd1, 0x7f, 0xfe, 0xd1, 0x7f, 0xff, 0xd2, 0x00, 0x00, 0x80, 0x00, 0xd2, 0x00, 0x00, 0x80, 0x01, 0xd2, 0x00, 0x00, 0x80, 0x02, 0xd2, 0x7f, 0xff, 0xff, 0xfd, 0xd2, 0x7f, 0xff, 0xff, 0xfe, 0xd2, 0x7f, 0xff, 0xff, 0xff }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackInt64(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []int64 { -9223372036854775808, -9223372036854775807, -9223372036854775806, -2147483651, -2147483650, -2147483649, -2147483648, -2147483647, -2147483646, -32771, -32770, -32769, -32768, -32767, -131, -130, -129, -128, -127, -34, -33, -32, -31, 0, 1, 126, 127, 128, 129, 130, 32765, 32766, 32767, 32768, 32769, 32770, 2147483645, 2147483646, 2147483647, 2147483648, 2147483649, 2147483650, 4294967296, 4294967297, 4294967298 } {
|
||||||
|
_, err := PackInt64(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0xd3, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xd3, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xd3, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xd3, 0xff, 0xff, 0xff, 0xff, 0x7f, 0xff, 0xff, 0xfd, 0xd3, 0xff, 0xff, 0xff, 0xff, 0x7f, 0xff, 0xff, 0xfe, 0xd3, 0xff, 0xff, 0xff, 0xff, 0x7f, 0xff, 0xff, 0xff, 0xd2, 0x80, 0x00, 0x00, 0x00, 0xd2, 0x80, 0x00, 0x00, 0x01, 0xd2, 0x80, 0x00, 0x00, 0x02, 0xd2, 0xff, 0xff, 0x7f, 0xfd, 0xd2, 0xff, 0xff, 0x7f, 0xfe, 0xd2, 0xff, 0xff, 0x7f, 0xff, 0xd1, 0x80, 0x00, 0xd1, 0x80, 0x01, 0xd1, 0xff, 0x7d, 0xd1, 0xff, 0x7e, 0xd1, 0xff, 0x7f, 0xd0, 0x80, 0xd0, 0x81, 0xd0, 0xde, 0xd0, 0xdf, 0xe0, 0xe1, 0x00, 0x01, 0x7e, 0x7f, 0xd1, 0x00, 0x80, 0xd1, 0x00, 0x81, 0xd1, 0x00, 0x82, 0xd1, 0x7f, 0xfd, 0xd1, 0x7f, 0xfe, 0xd1, 0x7f, 0xff, 0xd2, 0x00, 0x00, 0x80, 0x00, 0xd2, 0x00, 0x00, 0x80, 0x01, 0xd2, 0x00, 0x00, 0x80, 0x02, 0xd2, 0x7f, 0xff, 0xff, 0xfd, 0xd2, 0x7f, 0xff, 0xff, 0xfe, 0xd2, 0x7f, 0xff, 0xff, 0xff, 0xd3, 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x00, 0xd3, 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x01, 0xd3, 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x02, 0xd3, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0xd3, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x01, 0xd3, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x02 }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackNil(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
nbyteswrite, err := PackNil(b)
|
||||||
|
if nbyteswrite != 1 { t.Error("nbyteswrite != 1") }
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0xc0 }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackBool(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range []bool { false, true } {
|
||||||
|
nbyteswrite, err := PackBool(b, i)
|
||||||
|
if nbyteswrite != 1 { t.Error("nbyteswrite != 1") }
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0xc2, 0xc3 }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackInt32Array(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
_, err := PackInt32Array(b, []int32 {})
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
_, err = PackInt32Array(b, []int32 { 0 })
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
_, err = PackInt32Array(b, []int32 { 0, 1 })
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
_, err = PackInt32Array(b, []int32 { 0, 1, 2 })
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0x90, 0x91, 0x00, 0x92, 0x00, 0x01, 0x93, 0x00, 0x01, 0x02 }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackArray(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
_, err := PackArray(b, reflect.NewValue([]int32 {}).(reflect.ArrayOrSliceValue))
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
_, err = PackArray(b, reflect.NewValue([]int32 { 0 }).(reflect.ArrayOrSliceValue))
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
_, err = PackArray(b, reflect.NewValue([]int32 { 0, 1 }).(reflect.ArrayOrSliceValue))
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
_, err = PackArray(b, reflect.NewValue([]int32 { 0, 1, 2 }).(reflect.ArrayOrSliceValue))
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0x90, 0x91, 0x00, 0x92, 0x00, 0x01, 0x93, 0x00, 0x01, 0x02 }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPackMap(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
_, err := PackMap(b, reflect.NewValue(map[int] int { 0: 1, 2: 3, 4: 5 }).(*reflect.MapValue))
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0x83, 0x00, 0x01, 0x04, 0x05, 0x02, 0x03 }) != 0 { t.Error("wrong output", b.Bytes()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPack(t *testing.T) {
|
||||||
|
b := &bytes.Buffer{}
|
||||||
|
for _, i := range [](interface{}) { nil, false, true, 0, 1, 2, 3, 127, -32, -1, -33, 128 } {
|
||||||
|
_, err := Pack(b, i)
|
||||||
|
if err != nil { t.Error("err != nil") }
|
||||||
|
}
|
||||||
|
if bytes.Compare(b.Bytes(), []byte { 0xc0, 0xc2, 0xc3, 0x00, 0x01, 0x02, 0x03, 0x7f, 0xf0, 0xff, 0xd0, 0xef, 0xd1, 0x00, 0x80 }) == 0 { t.Error("wrong output") }
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestUnpackArray(t *testing.T) {
|
||||||
|
b := bytes.NewBuffer([]byte { 0x90, 0x91, 0x00, 0x92, 0x00, 0x01, 0x93, 0xd1, 0x00, 0x00, 0xd2, 0x00, 0x00, 0x00, 0x01, 0xd3, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xdc, 0x00, 0x02, 0x00, 0x01, 0xdd, 0x00, 0x00, 0x00, 0x04, 0x00, 0x01, 0x02, 0x03 })
|
||||||
|
for _, v := range [](interface{}) { [](interface{}) {}, [](interface{}) { int8(0) }, [](interface{}) { int8(0), int8(1) }, [](interface{}) { int16(0), int32(1), int64(2) }, [](interface{}){ int8(0), int8(1) }, [](interface{}) { int8(0), int8(1), int8(2), int8(3) } } {
|
||||||
|
retval, _, e := Unpack(b)
|
||||||
|
if e != nil { t.Error("err != nil") }
|
||||||
|
if !equal(reflect.NewValue(retval.Interface()), reflect.NewValue(v)) { t.Errorf("%s != %s", retval.Interface(), v) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestUnpackInt(t *testing.T) {
|
||||||
|
b := bytes.NewBuffer([]byte { 0xff, 0xe0, 0x00, 0x01, 0x02, 0x7d, 0x7e, 0x7f, 0xd0, 0x01, 0xd0, 0x80, 0xd0, 0xff, 0xcc, 0x80, 0xcc, 0xfd, 0xcc, 0xfe, 0xcc, 0xff, 0xd1, 0x00, 0x00, 0xd1, 0x7f, 0xff, 0xd1, 0xff, 0xff, 0xcd, 0x80, 0x00, 0xcd, 0xff, 0xff, 0xd2, 0x7f, 0xff, 0xff, 0xff, 0xce, 0x7f, 0xff, 0xff, 0xff, 0xd3, 0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xcf, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff })
|
||||||
|
for _, v := range [](interface{}) {
|
||||||
|
int8(-1), int8(-32), int8(0), int8(1), int8(2), int8(125), int8(126), int8(127), int8(1), int8(-128), int8(-1), uint8(128), uint8(253), uint8(254), uint8(255), int16(0), int16(32767), int16(-1), uint16(32768), uint16(65535), int32(2147483647), uint32(2147483647), int64(9223372036854775807), uint64(18446744073709551615) } {
|
||||||
|
retval, _, e := Unpack(b)
|
||||||
|
if e != nil { t.Error("err != nil") }
|
||||||
|
if retval.Interface() != v { t.Errorf("%u != %u", retval.Interface(), v) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestUnpackFloat(t *testing.T) {
|
||||||
|
b := bytes.NewBuffer([]byte { 0xca, 0x3d, 0xcc, 0xcc, 0xcd, 0xca, 0x3e, 0x4c, 0xcc, 0xcd, 0xca, 0xbd, 0xcc, 0xcc, 0xcd, 0xca, 0xbe, 0x4c, 0xcc, 0xcd, 0xca, 0x7f, 0x80, 0x00, 0x00, 0xca, 0xff, 0x80, 0x00, 0x00, 0xca, 0xff, 0xc0, 0x00, 0x0, 0xcb, 0x3f, 0xb9, 0x99, 0x99, 0x99, 0x99, 0x99, 0x9a, 0xcb, 0x3f, 0xc9, 0x99, 0x99, 0x99, 0x99, 0x99, 0x9a, 0xcb, 0xbf, 0xb9, 0x99, 0x99, 0x99, 0x99, 0x99, 0x9a, 0xcb, 0xbf, 0xc9, 0x99, 0x99, 0x99, 0x99, 0x99, 0x9a, 0xcb, 0x7f, 0xf0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xcb, 0xcb, 0xff, 0xf0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xcb, 0x7f, 0xf8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 })
|
||||||
|
for _, v := range [](interface{}) { float32(.1), float32(.2), float32(-.1), float32(-.2), float32(math.Inf(1)), float32(math.Inf(-1)), float32(math.NaN()), float64(.1), float64(.2), float64(-.1), float64(-.2) } {
|
||||||
|
retval, _, e := Unpack(b)
|
||||||
|
if e != nil { t.Error("err != nil") }
|
||||||
|
isnan := false
|
||||||
|
if _v, ok := v.(float64); ok {
|
||||||
|
isnan = math.IsNaN(_v)
|
||||||
|
} else if _v, ok := v.(float32); ok {
|
||||||
|
isnan = math.IsNaN(float64(_v))
|
||||||
|
}
|
||||||
|
if isnan {
|
||||||
|
if retval.Interface() == v { t.Errorf("[NaN] %u == %u", retval.Interface(), v) }
|
||||||
|
} else {
|
||||||
|
if retval.Interface() != v { t.Errorf("%u != %u", retval.Interface(), v) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
591
go/pack.go
Normal file
591
go/pack.go
Normal file
@@ -0,0 +1,591 @@
|
|||||||
|
package msgpack
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"unsafe"
|
||||||
|
"reflect"
|
||||||
|
);
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUint8(writer io.Writer, value uint8) (n int, err os.Error) {
|
||||||
|
if value < 128 {
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { 0xcc, byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUint16(writer io.Writer, value uint16) (n int, err os.Error) {
|
||||||
|
if value < 128 {
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
} else if value < 256 {
|
||||||
|
return writer.Write([]byte { 0xcc, byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { 0xcd, byte(value >> 8), byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUint32(writer io.Writer, value uint32) (n int, err os.Error) {
|
||||||
|
if value < 128 {
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
} else if value < 256 {
|
||||||
|
return writer.Write([]byte { 0xcc, byte(value) })
|
||||||
|
} else if value < 65536 {
|
||||||
|
return writer.Write([]byte { 0xcd, byte(value >> 8), byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { 0xce, byte(value >> 24), byte(value >> 16), byte(value >> 8), byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUint64(writer io.Writer, value uint64) (n int, err os.Error) {
|
||||||
|
if value < 128 {
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
} else if value < 256 {
|
||||||
|
return writer.Write([]byte { 0xcc, byte(value) })
|
||||||
|
} else if value < 65536 {
|
||||||
|
return writer.Write([]byte { 0xcd, byte(value >> 8), byte(value) })
|
||||||
|
} else if value < 4294967296 {
|
||||||
|
return writer.Write([]byte { 0xce, byte(value >> 24), byte(value >> 16), byte(value >> 8), byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { 0xcf, byte(value >> 56), byte(value >> 48), byte(value >> 40), byte(value >> 32), byte(value >> 24), byte(value >> 16), byte(value >> 8), byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
func PackUint(writer io.Writer, value uint) (n int, err os.Error) {
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
switch unsafe.Sizeof(value) {
|
||||||
|
case 4:
|
||||||
|
return PackUint32(writer, *(*uint32)(unsafe.Pointer(&value)))
|
||||||
|
case 8:
|
||||||
|
return PackUint64(writer, *(*uint64)(unsafe.Pointer(&value)))
|
||||||
|
}
|
||||||
|
return 0, os.ENOENT // never get here
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt8(writer io.Writer, value int8) (n int, err os.Error) {
|
||||||
|
if value < -32 {
|
||||||
|
return writer.Write([]byte { 0xd0, byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt16(writer io.Writer, value int16) (n int, err os.Error) {
|
||||||
|
if value < -128 || value >= 128 {
|
||||||
|
return writer.Write([]byte { 0xd1, byte(uint16(value) >> 8), byte(value) })
|
||||||
|
} else if value < -32 {
|
||||||
|
return writer.Write([]byte { 0xd0, byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt32(writer io.Writer, value int32) (n int, err os.Error) {
|
||||||
|
if value < -32768 || value >= 32768 {
|
||||||
|
return writer.Write([]byte { 0xd2, byte(uint32(value) >> 24), byte(uint32(value) >> 16), byte(uint32(value) >> 8), byte(value) })
|
||||||
|
} else if value < -128 {
|
||||||
|
return writer.Write([]byte { 0xd1, byte(uint32(value) >> 8), byte(value) })
|
||||||
|
} else if value < -32 {
|
||||||
|
return writer.Write([]byte { 0xd0, byte(value) })
|
||||||
|
} else if value >= 128 {
|
||||||
|
return writer.Write([]byte { 0xd1, byte(uint32(value) >> 8), byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt64(writer io.Writer, value int64) (n int, err os.Error) {
|
||||||
|
if value < -2147483648 || value >= 2147483648 {
|
||||||
|
return writer.Write([]byte { 0xd3, byte(uint64(value) >> 56), byte(uint64(value) >> 48), byte(uint64(value) >> 40), byte(uint64(value) >> 32), byte(uint64(value) >> 24), byte(uint64(value) >> 16), byte(uint64(value) >> 8), byte(value) })
|
||||||
|
} else if value < -32768 || value >= 32768 {
|
||||||
|
return writer.Write([]byte { 0xd2, byte(uint64(value) >> 24), byte(uint64(value) >> 16), byte(uint64(value) >> 8), byte(value) })
|
||||||
|
} else if value < -128 || value >= 128 {
|
||||||
|
return writer.Write([]byte { 0xd1, byte(uint64(value) >> 8), byte(value) })
|
||||||
|
} else if value < -32 {
|
||||||
|
return writer.Write([]byte { 0xd0, byte(value) })
|
||||||
|
}
|
||||||
|
return writer.Write([]byte { byte(value) })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt(writer io.Writer, value int) (n int, err os.Error) {
|
||||||
|
switch unsafe.Sizeof(value) {
|
||||||
|
case 4:
|
||||||
|
return PackInt32(writer, *(*int32)(unsafe.Pointer(&value)))
|
||||||
|
case 8:
|
||||||
|
return PackInt64(writer, *(*int64)(unsafe.Pointer(&value)))
|
||||||
|
}
|
||||||
|
return 0, os.ENOENT // never get here
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackNil(writer io.Writer) (n int, err os.Error) {
|
||||||
|
return writer.Write([]byte{ 0xc0 })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackBool(writer io.Writer, value bool) (n int, err os.Error) {
|
||||||
|
var code byte;
|
||||||
|
if value {
|
||||||
|
code = 0xc3
|
||||||
|
} else {
|
||||||
|
code = 0xc2
|
||||||
|
}
|
||||||
|
return writer.Write([]byte{ code })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackFloat32(writer io.Writer, value float32) (n int, err os.Error) {
|
||||||
|
return PackUint32(writer, *(*uint32)(unsafe.Pointer(&value)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackFloat64(writer io.Writer, value float64) (n int, err os.Error) {
|
||||||
|
return PackUint64(writer, *(*uint64)(unsafe.Pointer(&value)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackBytes(writer io.Writer, value []byte) (n int, err os.Error) {
|
||||||
|
if len(value) < 32 {
|
||||||
|
n1, err := writer.Write([]byte { 0xa0 | uint8(len(value)) })
|
||||||
|
if err != nil { return n1, err }
|
||||||
|
n2, err := writer.Write(value)
|
||||||
|
return n1 + n2, err
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n1, err := writer.Write([]byte { 0xda, byte(len(value) >> 16), byte(len(value)) })
|
||||||
|
if err != nil { return n1, err }
|
||||||
|
n2, err := writer.Write(value)
|
||||||
|
return n1 + n2, err
|
||||||
|
}
|
||||||
|
n1, err := writer.Write([]byte { 0xdb, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n1, err }
|
||||||
|
n2, err := writer.Write(value)
|
||||||
|
return n1 + n2, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUint16Array(writer io.Writer, value []uint16) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint16(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint16(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint16(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUint32Array(writer io.Writer, value []uint32) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUint64Array(writer io.Writer, value []uint64) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackUint64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackUintArray(writer io.Writer, value []uint) (n int, err os.Error) {
|
||||||
|
switch unsafe.Sizeof(0) {
|
||||||
|
case 4:
|
||||||
|
return PackUint32Array(writer, *(*[]uint32)(unsafe.Pointer(&value)))
|
||||||
|
case 8:
|
||||||
|
return PackUint64Array(writer, *(*[]uint64)(unsafe.Pointer(&value)))
|
||||||
|
}
|
||||||
|
return 0, os.ENOENT // never get here
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt8Array(writer io.Writer, value []int8) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt8(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt8(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt8(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt16Array(writer io.Writer, value []int16) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt16(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt16(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt16(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt32Array(writer io.Writer, value []int32) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackInt64Array(writer io.Writer, value []int64) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackInt64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackIntArray(writer io.Writer, value []int) (n int, err os.Error) {
|
||||||
|
switch unsafe.Sizeof(0) {
|
||||||
|
case 4:
|
||||||
|
return PackInt32Array(writer, *(*[]int32)(unsafe.Pointer(&value)))
|
||||||
|
case 8:
|
||||||
|
return PackInt64Array(writer, *(*[]int64)(unsafe.Pointer(&value)))
|
||||||
|
}
|
||||||
|
return 0, os.ENOENT // never get here
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackFloat32Array(writer io.Writer, value []float32) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackFloat32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackFloat32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackFloat32(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackFloat64Array(writer io.Writer, value []float64) (n int, err os.Error) {
|
||||||
|
if len(value) < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackFloat64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if len(value) < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackFloat64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(len(value) >> 24), byte(len(value) >> 16), byte(len(value) >> 8), byte(len(value)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, i := range value {
|
||||||
|
_n, err := PackFloat64(writer, i)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackArray(writer io.Writer, value reflect.ArrayOrSliceValue) (n int, err os.Error) {
|
||||||
|
{
|
||||||
|
elemType, ok := value.Type().(reflect.ArrayOrSliceType).Elem().(*reflect.UintType)
|
||||||
|
if ok && elemType.Kind() == reflect.Uint8 {
|
||||||
|
return PackBytes(writer, value.Interface().([]byte))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
l := value.Len()
|
||||||
|
if l < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x90 | byte(l) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for i := 0; i < l; i++ {
|
||||||
|
_n, err := PackValue(writer, value.Elem(i))
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if l < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xdc, byte(l >> 8), byte(l) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for i := 0; i < l; i++ {
|
||||||
|
_n, err := PackValue(writer, value.Elem(i))
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdd, byte(l >> 24), byte(l >> 16), byte(l >> 8), byte(l) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for i := 0; i < l; i++ {
|
||||||
|
_n, err := PackValue(writer, value.Elem(i))
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackMap(writer io.Writer, value *reflect.MapValue) (n int, err os.Error) {
|
||||||
|
keys := value.Keys()
|
||||||
|
if value.Len() < 16 {
|
||||||
|
n, err := writer.Write([]byte { 0x80 | byte(len(keys)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, k := range keys {
|
||||||
|
_n, err := PackValue(writer, k)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
_n, err = PackValue(writer, value.Elem(k))
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else if value.Len() < 65536 {
|
||||||
|
n, err := writer.Write([]byte { 0xde, byte(len(keys) >> 8), byte(len(keys)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, k := range keys {
|
||||||
|
_n, err := PackValue(writer, k)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
_n, err = PackValue(writer, value.Elem(k))
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
n, err := writer.Write([]byte { 0xdf, byte(len(keys) >> 24), byte(len(keys) >> 16), byte(len(keys) >> 8), byte(len(keys)) })
|
||||||
|
if err != nil { return n, err }
|
||||||
|
for _, k := range keys {
|
||||||
|
_n, err := PackValue(writer, k)
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
_n, err = PackValue(writer, value.Elem(k))
|
||||||
|
if err != nil { return n, err }
|
||||||
|
n += _n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func PackValue(writer io.Writer, value reflect.Value) (n int, err os.Error) {
|
||||||
|
if value == nil || value.Type() == nil { return PackNil(writer) }
|
||||||
|
switch _value := value.(type) {
|
||||||
|
case *reflect.BoolValue: return PackBool(writer, _value.Get())
|
||||||
|
case *reflect.UintValue: return PackUint64(writer, _value.Get())
|
||||||
|
case *reflect.IntValue: return PackInt64(writer, _value.Get())
|
||||||
|
case *reflect.FloatValue: return PackFloat64(writer, _value.Get())
|
||||||
|
case *reflect.ArrayValue: return PackArray(writer, _value)
|
||||||
|
case *reflect.SliceValue: return PackArray(writer, _value)
|
||||||
|
case *reflect.MapValue: return PackMap(writer, _value)
|
||||||
|
case *reflect.InterfaceValue:
|
||||||
|
__value := reflect.NewValue(_value.Interface())
|
||||||
|
_, ok := __value.(*reflect.InterfaceValue)
|
||||||
|
if !ok {
|
||||||
|
return PackValue(writer, __value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panic("unsupported type: " + value.Type().String())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packs a given value and writes it into the specified writer.
|
||||||
|
func Pack(writer io.Writer, value interface{}) (n int, err os.Error) {
|
||||||
|
if value == nil { return PackNil(writer) }
|
||||||
|
switch _value := value.(type) {
|
||||||
|
case bool: return PackBool(writer, _value)
|
||||||
|
case uint8: return PackUint8(writer, _value)
|
||||||
|
case uint16: return PackUint16(writer, _value)
|
||||||
|
case uint32: return PackUint32(writer, _value)
|
||||||
|
case uint64: return PackUint64(writer, _value)
|
||||||
|
case uint: return PackUint(writer, _value)
|
||||||
|
case int8: return PackInt8(writer, _value)
|
||||||
|
case int16: return PackInt16(writer, _value)
|
||||||
|
case int32: return PackInt32(writer, _value)
|
||||||
|
case int64: return PackInt64(writer, _value)
|
||||||
|
case int: return PackInt(writer, _value)
|
||||||
|
case float32: return PackFloat32(writer, _value)
|
||||||
|
case float64: return PackFloat64(writer, _value)
|
||||||
|
case []byte: return PackBytes(writer, _value)
|
||||||
|
case []uint16: return PackUint16Array(writer, _value)
|
||||||
|
case []uint32: return PackUint32Array(writer, _value)
|
||||||
|
case []uint64: return PackUint64Array(writer, _value)
|
||||||
|
case []uint: return PackUintArray(writer, _value)
|
||||||
|
case []int8: return PackInt8Array(writer, _value)
|
||||||
|
case []int16: return PackInt16Array(writer, _value)
|
||||||
|
case []int32: return PackInt32Array(writer, _value)
|
||||||
|
case []int64: return PackInt64Array(writer, _value)
|
||||||
|
case []int: return PackIntArray(writer, _value)
|
||||||
|
case []float32: return PackFloat32Array(writer, _value)
|
||||||
|
case []float64: return PackFloat64Array(writer, _value)
|
||||||
|
default:
|
||||||
|
return PackValue(writer, reflect.NewValue(value))
|
||||||
|
}
|
||||||
|
return 0, nil // never get here
|
||||||
|
}
|
288
go/unpack.go
Normal file
288
go/unpack.go
Normal file
@@ -0,0 +1,288 @@
|
|||||||
|
package msgpack
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"unsafe"
|
||||||
|
"strconv"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
|
func readByte(reader io.Reader) (v uint8, err os.Error) {
|
||||||
|
data := [1]byte{}
|
||||||
|
_, e := reader.Read(data[0:])
|
||||||
|
if e != nil { return 0, e }
|
||||||
|
return data[0], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readUint16(reader io.Reader) (v uint16, n int, err os.Error) {
|
||||||
|
data := [2]byte{}
|
||||||
|
n, e := reader.Read(data[0:])
|
||||||
|
if e != nil { return 0, n, e }
|
||||||
|
return (uint16(data[0]) << 8) | uint16(data[1]), n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readUint32(reader io.Reader) (v uint32, n int, err os.Error) {
|
||||||
|
data := [4]byte{}
|
||||||
|
n, e := reader.Read(data[0:])
|
||||||
|
if e != nil { return 0, n, e }
|
||||||
|
return (uint32(data[0]) << 24) | (uint32(data[1]) << 16) | (uint32(data[2]) << 8) | uint32(data[3]), n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readUint64(reader io.Reader) (v uint64, n int, err os.Error) {
|
||||||
|
data := [8]byte{}
|
||||||
|
n, e := reader.Read(data[0:])
|
||||||
|
if e != nil { return 0, n, e }
|
||||||
|
return (uint64(data[0]) << 56) | (uint64(data[1]) << 48) | (uint64(data[2]) << 40) | (uint64(data[3]) << 32) | (uint64(data[4]) << 24) | (uint64(data[5]) << 16) | (uint64(data[6]) << 8) | uint64(data[7]), n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readInt16(reader io.Reader) (v int16, n int, err os.Error) {
|
||||||
|
data := [2]byte{}
|
||||||
|
n, e := reader.Read(data[0:])
|
||||||
|
if e != nil { return 0, n, e }
|
||||||
|
return (int16(data[0]) << 8) | int16(data[1]), n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readInt32(reader io.Reader) (v int32, n int, err os.Error) {
|
||||||
|
data := [4]byte{}
|
||||||
|
n, e := reader.Read(data[0:])
|
||||||
|
if e != nil { return 0, n, e }
|
||||||
|
return (int32(data[0]) << 24) | (int32(data[1]) << 16) | (int32(data[2]) << 8) | int32(data[3]), n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readInt64(reader io.Reader) (v int64, n int, err os.Error) {
|
||||||
|
data := [8]byte{}
|
||||||
|
n, e := reader.Read(data[0:])
|
||||||
|
if e != nil { return 0, n, e }
|
||||||
|
return (int64(data[0]) << 56) | (int64(data[1]) << 48) | (int64(data[2]) << 40) | (int64(data[3]) << 32) | (int64(data[4]) << 24) | (int64(data[5]) << 16) | (int64(data[6]) << 8) | int64(data[7]), n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unpackArray(reader io.Reader, nelems uint) (v reflect.Value, n int, err os.Error) {
|
||||||
|
retval := make([]interface{}, nelems)
|
||||||
|
nbytesread := 0
|
||||||
|
var i uint
|
||||||
|
for i = 0; i < nelems; i++ {
|
||||||
|
v, n, e := Unpack(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval[i] = v.Interface()
|
||||||
|
}
|
||||||
|
return reflect.NewValue(retval), nbytesread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unpackArrayReflected(reader io.Reader, nelems uint) (v reflect.Value, n int, err os.Error) {
|
||||||
|
retval := make([]reflect.Value, nelems)
|
||||||
|
nbytesread := 0
|
||||||
|
var i uint
|
||||||
|
for i = 0; i < nelems; i++ {
|
||||||
|
v, n, e := UnpackReflected(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval[i] = v
|
||||||
|
}
|
||||||
|
return reflect.NewValue(retval), nbytesread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unpackMap(reader io.Reader, nelems uint) (v reflect.Value, n int, err os.Error) {
|
||||||
|
retval := make(map [interface{}] interface{})
|
||||||
|
nbytesread := 0
|
||||||
|
var i uint
|
||||||
|
for i = 0; i < nelems; i++ {
|
||||||
|
k, n, e := Unpack(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
v, n, e := Unpack(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval[k.Interface()] = v.Interface()
|
||||||
|
}
|
||||||
|
return reflect.NewValue(retval), nbytesread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unpackMapReflected(reader io.Reader, nelems uint) (v reflect.Value, n int, err os.Error) {
|
||||||
|
retval := make(map [reflect.Value] reflect.Value)
|
||||||
|
nbytesread := 0
|
||||||
|
var i uint
|
||||||
|
for i = 0; i < nelems; i++ {
|
||||||
|
k, n, e := UnpackReflected(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
v, n, e := UnpackReflected(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval[k] = v
|
||||||
|
}
|
||||||
|
return reflect.NewValue(retval), nbytesread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unpack(reader io.Reader, reflected bool) (v reflect.Value, n int, err os.Error) {
|
||||||
|
var retval reflect.Value
|
||||||
|
var nbytesread int = 0
|
||||||
|
|
||||||
|
c, e := readByte(reader)
|
||||||
|
if e != nil { return nil, 0, e }
|
||||||
|
nbytesread += 1
|
||||||
|
if c < 0x80 || c >= 0xe0 {
|
||||||
|
retval = reflect.NewValue(int8(c))
|
||||||
|
} else if c >= 0x80 && c <= 0x8f {
|
||||||
|
if reflected {
|
||||||
|
retval, n, e = unpackMapReflected(reader, uint(c & 0xf))
|
||||||
|
} else {
|
||||||
|
retval, n, e = unpackMap(reader, uint(c & 0xf))
|
||||||
|
}
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
nbytesread += n
|
||||||
|
} else if c >= 0x90 && c <= 0x9f {
|
||||||
|
if reflected {
|
||||||
|
retval, n, e = unpackArrayReflected(reader, uint(c & 0xf))
|
||||||
|
} else {
|
||||||
|
retval, n, e = unpackArray(reader, uint(c & 0xf))
|
||||||
|
}
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
nbytesread += n
|
||||||
|
} else if c >= 0xa0 && c <= 0xbf {
|
||||||
|
data := make([]byte, c & 0xf);
|
||||||
|
n, e := reader.Read(data)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
} else {
|
||||||
|
switch c {
|
||||||
|
case 0xc0: retval = reflect.NewValue(nil)
|
||||||
|
case 0xc2: retval = reflect.NewValue(false)
|
||||||
|
case 0xc3: retval = reflect.NewValue(true)
|
||||||
|
case 0xca:
|
||||||
|
data, n, e := readUint32(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(*(*float32)(unsafe.Pointer(&data)))
|
||||||
|
case 0xcb:
|
||||||
|
data, n, e := readUint64(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(*(*float64)(unsafe.Pointer(&data)))
|
||||||
|
case 0xcc:
|
||||||
|
data, e := readByte(reader)
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(uint8(data))
|
||||||
|
nbytesread += 1
|
||||||
|
case 0xcd:
|
||||||
|
data, n, e := readUint16(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xce:
|
||||||
|
data, n, e := readUint32(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xcf:
|
||||||
|
data, n, e := readUint64(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xd0:
|
||||||
|
data, e := readByte(reader)
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(int8(data))
|
||||||
|
nbytesread += 1
|
||||||
|
case 0xd1:
|
||||||
|
data, n, e := readInt16(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xd2:
|
||||||
|
data, n, e := readInt32(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xd3:
|
||||||
|
data, n, e := readInt64(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xda:
|
||||||
|
nbytestoread, n, e := readUint16(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
data := make([]byte, nbytestoread)
|
||||||
|
n, e = reader.Read(data)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xdb:
|
||||||
|
nbytestoread, n, e := readUint32(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
data := make([]byte, nbytestoread)
|
||||||
|
n, e = reader.Read(data)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
retval = reflect.NewValue(data)
|
||||||
|
case 0xdc:
|
||||||
|
nelemstoread, n, e := readUint16(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
if reflected {
|
||||||
|
retval, n, e = unpackArrayReflected(reader, uint(nelemstoread))
|
||||||
|
} else {
|
||||||
|
retval, n, e = unpackArray(reader, uint(nelemstoread))
|
||||||
|
}
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
case 0xdd:
|
||||||
|
nelemstoread, n, e := readUint32(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
if reflected {
|
||||||
|
retval, n, e = unpackArrayReflected(reader, uint(nelemstoread))
|
||||||
|
} else {
|
||||||
|
retval, n, e = unpackArray(reader, uint(nelemstoread))
|
||||||
|
}
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
case 0xde:
|
||||||
|
nelemstoread, n, e := readUint16(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
if reflected {
|
||||||
|
retval, n, e = unpackMapReflected(reader, uint(nelemstoread))
|
||||||
|
} else {
|
||||||
|
retval, n, e = unpackMap(reader, uint(nelemstoread))
|
||||||
|
}
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
case 0xdf:
|
||||||
|
nelemstoread, n, e := readUint32(reader)
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
if reflected {
|
||||||
|
retval, n, e = unpackMapReflected(reader, uint(nelemstoread))
|
||||||
|
} else {
|
||||||
|
retval, n, e = unpackMap(reader, uint(nelemstoread))
|
||||||
|
}
|
||||||
|
nbytesread += n
|
||||||
|
if e != nil { return nil, nbytesread, e }
|
||||||
|
default:
|
||||||
|
panic("unsupported code: " + strconv.Itoa(int(c)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return retval, nbytesread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reads a value from the reader, unpack and returns it.
|
||||||
|
func Unpack(reader io.Reader) (v reflect.Value, n int, err os.Error) {
|
||||||
|
return unpack(reader, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reads unpack a value from the reader, unpack and returns it. When the
|
||||||
|
// value is an array or map, leaves the elements wrapped by corresponding
|
||||||
|
// wrapper objects defined in reflect package.
|
||||||
|
func UnpackReflected(reader io.Reader) (v reflect.Value, n int, err os.Error) {
|
||||||
|
return unpack(reader, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
@@ -1,4 +1,4 @@
|
|||||||
Copyright (c) 2009, Hideyuki Tanaka
|
Copyright (c) 2009-2010, Hideyuki Tanaka
|
||||||
All rights reserved.
|
All rights reserved.
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without
|
Redistribution and use in source and binary forms, with or without
|
||||||
|
@@ -1,137 +0,0 @@
|
|||||||
#include <msgpack.h>
|
|
||||||
|
|
||||||
void msgpack_sbuffer_init_wrap(msgpack_sbuffer* sbuf)
|
|
||||||
{
|
|
||||||
msgpack_sbuffer_init(sbuf);
|
|
||||||
}
|
|
||||||
|
|
||||||
void msgpack_sbuffer_destroy_wrap(msgpack_sbuffer* sbuf)
|
|
||||||
{
|
|
||||||
msgpack_sbuffer_destroy(sbuf);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_sbuffer_write_wrap(void* data, const char* buf, unsigned int len)
|
|
||||||
{
|
|
||||||
return msgpack_sbuffer_write(data, buf, len);
|
|
||||||
}
|
|
||||||
|
|
||||||
msgpack_packer* msgpack_packer_new_wrap(void *data, msgpack_packer_write callback)
|
|
||||||
{
|
|
||||||
return msgpack_packer_new(data, callback);
|
|
||||||
}
|
|
||||||
|
|
||||||
void msgpack_packer_free_wrap(msgpack_packer* pk)
|
|
||||||
{
|
|
||||||
msgpack_packer_free(pk);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_uint8_wrap(msgpack_packer* pk, uint8_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_uint8(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_uint16_wrap(msgpack_packer* pk, uint16_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_uint16(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_uint32_wrap(msgpack_packer* pk, uint32_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_uint32(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_uint64_wrap(msgpack_packer* pk, uint64_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_uint64(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_int8_wrap(msgpack_packer* pk, int8_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_int8(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_int16_wrap(msgpack_packer* pk, int16_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_int16(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_int32_wrap(msgpack_packer* pk, int32_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_int32(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_int64_wrap(msgpack_packer* pk, int64_t d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_int64(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_double_wrap(msgpack_packer* pk, double d)
|
|
||||||
{
|
|
||||||
return msgpack_pack_double(pk, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_nil_wrap(msgpack_packer* pk)
|
|
||||||
{
|
|
||||||
return msgpack_pack_nil(pk);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_true_wrap(msgpack_packer* pk)
|
|
||||||
{
|
|
||||||
return msgpack_pack_true(pk);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_false_wrap(msgpack_packer* pk)
|
|
||||||
{
|
|
||||||
return msgpack_pack_false(pk);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_array_wrap(msgpack_packer* pk, unsigned int n)
|
|
||||||
{
|
|
||||||
return msgpack_pack_array(pk, n);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_map_wrap(msgpack_packer* pk, unsigned int n)
|
|
||||||
{
|
|
||||||
return msgpack_pack_map(pk, n);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_raw_wrap(msgpack_packer* pk, size_t l)
|
|
||||||
{
|
|
||||||
return msgpack_pack_raw(pk, l);
|
|
||||||
}
|
|
||||||
|
|
||||||
int msgpack_pack_raw_body_wrap(msgpack_packer* pk, const void *b, size_t l)
|
|
||||||
{
|
|
||||||
return msgpack_pack_raw_body(pk, b, l);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool msgpack_unpacker_reserve_buffer_wrap(msgpack_unpacker *mpac, size_t size)
|
|
||||||
{
|
|
||||||
return msgpack_unpacker_reserve_buffer(mpac, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
char *msgpack_unpacker_buffer_wrap(msgpack_unpacker *mpac)
|
|
||||||
{
|
|
||||||
return msgpack_unpacker_buffer(mpac);
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t msgpack_unpacker_buffer_capacity_wrap(const msgpack_unpacker *mpac)
|
|
||||||
{
|
|
||||||
return msgpack_unpacker_buffer_capacity(mpac);
|
|
||||||
}
|
|
||||||
|
|
||||||
void msgpack_unpacker_buffer_consumed_wrap(msgpack_unpacker *mpac, size_t size)
|
|
||||||
{
|
|
||||||
msgpack_unpacker_buffer_consumed(mpac, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void msgpack_unpacker_data_wrap(msgpack_unpacker *mpac, msgpack_object *obj)
|
|
||||||
{
|
|
||||||
*obj=msgpack_unpacker_data(mpac);
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t msgpack_unpacker_message_size_wrap(const msgpack_unpacker *mpac)
|
|
||||||
{
|
|
||||||
return msgpack_unpacker_message_size(mpac);
|
|
||||||
}
|
|
||||||
|
|
@@ -1,32 +1,50 @@
|
|||||||
Name: msgpack
|
Name: msgpack
|
||||||
Version: 0.2.0
|
Version: 0.6.1.1
|
||||||
License: BSD3
|
Synopsis: A Haskell implementation of MessagePack
|
||||||
License-File: LICENSE
|
|
||||||
Author: Hideyuki Tanaka
|
|
||||||
Maintainer: Hideyuki Tanaka <tanaka.hideyuki@gmail.com>
|
|
||||||
Category: Data
|
|
||||||
Synopsis: A Haskell binding to MessagePack
|
|
||||||
Description:
|
Description:
|
||||||
A Haskell binding to MessagePack <http://msgpack.sourceforge.jp/>
|
A Haskell implementation of MessagePack <http://msgpack.org/>
|
||||||
Homepage: http://github.com/tanakh/hsmsgpack
|
|
||||||
Stability: Experimental
|
|
||||||
Tested-with: GHC==6.10.4
|
|
||||||
Cabal-Version: >=1.2
|
|
||||||
Build-Type: Simple
|
|
||||||
|
|
||||||
library
|
License: BSD3
|
||||||
build-depends: base>=4 && <5, mtl, bytestring
|
License-File: LICENSE
|
||||||
ghc-options: -O2 -Wall
|
Copyright: Copyright (c) 2009-2011, Hideyuki Tanaka
|
||||||
hs-source-dirs: src
|
Category: Data
|
||||||
extra-libraries: msgpackc
|
Author: Hideyuki Tanaka
|
||||||
|
Maintainer: Hideyuki Tanaka <tanaka.hideyuki@gmail.com>
|
||||||
|
Homepage: http://github.com/msgpack/msgpack
|
||||||
|
Stability: Experimental
|
||||||
|
Cabal-Version: >= 1.6
|
||||||
|
Build-Type: Simple
|
||||||
|
|
||||||
|
Extra-source-files:
|
||||||
|
test/Test.hs
|
||||||
|
test/UserData.hs
|
||||||
|
|
||||||
|
Library
|
||||||
|
Build-depends: base >=4 && <5,
|
||||||
|
transformers >= 0.2 && < 0.3,
|
||||||
|
bytestring >= 0.9 && < 0.10,
|
||||||
|
text >= 0.11 && < 0.12,
|
||||||
|
vector >= 0.7 && < 0.8,
|
||||||
|
attoparsec >= 0.8 && < 0.9,
|
||||||
|
binary >= 0.5.0 && < 0.5.1,
|
||||||
|
data-binary-ieee754 >= 0.4 && < 0.5,
|
||||||
|
deepseq >= 1.1 && <1.2,
|
||||||
|
template-haskell >= 2.4 && < 2.6
|
||||||
|
|
||||||
|
Ghc-options: -Wall
|
||||||
|
Hs-source-dirs: src
|
||||||
|
|
||||||
Exposed-modules:
|
Exposed-modules:
|
||||||
Data.MessagePack
|
Data.MessagePack
|
||||||
Data.MessagePack.Base
|
Data.MessagePack.Assoc
|
||||||
Data.MessagePack.Class
|
Data.MessagePack.Pack
|
||||||
Data.MessagePack.Feed
|
Data.MessagePack.Unpack
|
||||||
Data.MessagePack.Monad
|
Data.MessagePack.Object
|
||||||
Data.MessagePack.Stream
|
Data.MessagePack.Derive
|
||||||
|
|
||||||
C-Sources:
|
Other-modules:
|
||||||
cbits/msgpack.c
|
Data.MessagePack.Internal.Utf8
|
||||||
|
|
||||||
|
Source-repository head
|
||||||
|
Type: git
|
||||||
|
Location: git://github.com/msgpack/msgpack.git
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
--------------------------------------------------------------------
|
--------------------------------------------------------------------
|
||||||
-- |
|
-- |
|
||||||
-- Module : Data.MessagePack
|
-- Module : Data.MessagePack
|
||||||
-- Copyright : (c) Hideyuki Tanaka, 2009
|
-- Copyright : (c) Hideyuki Tanaka, 2009-2010
|
||||||
-- License : BSD3
|
-- License : BSD3
|
||||||
--
|
--
|
||||||
-- Maintainer: tanaka.hideyuki@gmail.com
|
-- Maintainer: tanaka.hideyuki@gmail.com
|
||||||
@@ -13,51 +13,15 @@
|
|||||||
--------------------------------------------------------------------
|
--------------------------------------------------------------------
|
||||||
|
|
||||||
module Data.MessagePack(
|
module Data.MessagePack(
|
||||||
module Data.MessagePack.Base,
|
module Data.MessagePack.Assoc,
|
||||||
module Data.MessagePack.Class,
|
module Data.MessagePack.Pack,
|
||||||
module Data.MessagePack.Feed,
|
module Data.MessagePack.Unpack,
|
||||||
module Data.MessagePack.Monad,
|
module Data.MessagePack.Object,
|
||||||
module Data.MessagePack.Stream,
|
module Data.MessagePack.Derive,
|
||||||
|
|
||||||
-- * Pack and Unpack
|
|
||||||
packb,
|
|
||||||
unpackb,
|
|
||||||
|
|
||||||
-- * Pure version of Pack and Unpack
|
|
||||||
packb',
|
|
||||||
unpackb',
|
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Data.ByteString (ByteString)
|
import Data.MessagePack.Assoc
|
||||||
import System.IO.Unsafe
|
import Data.MessagePack.Pack
|
||||||
|
import Data.MessagePack.Unpack
|
||||||
import Data.MessagePack.Base
|
import Data.MessagePack.Object
|
||||||
import Data.MessagePack.Class
|
import Data.MessagePack.Derive
|
||||||
import Data.MessagePack.Feed
|
|
||||||
import Data.MessagePack.Monad
|
|
||||||
import Data.MessagePack.Stream
|
|
||||||
|
|
||||||
-- | Pack Haskell data to MessagePack string.
|
|
||||||
packb :: OBJECT a => a -> IO ByteString
|
|
||||||
packb dat = do
|
|
||||||
sb <- newSimpleBuffer
|
|
||||||
pc <- newPacker sb
|
|
||||||
pack pc dat
|
|
||||||
simpleBufferData sb
|
|
||||||
|
|
||||||
-- | Unpack MessagePack string to Haskell data.
|
|
||||||
unpackb :: OBJECT a => ByteString -> IO (Result a)
|
|
||||||
unpackb bs = do
|
|
||||||
withZone $ \z -> do
|
|
||||||
r <- unpackObject z bs
|
|
||||||
return $ case r of
|
|
||||||
Left err -> Left (show err)
|
|
||||||
Right (_, dat) -> fromObject dat
|
|
||||||
|
|
||||||
-- | Pure version of 'packb'.
|
|
||||||
packb' :: OBJECT a => a -> ByteString
|
|
||||||
packb' dat = unsafePerformIO $ packb dat
|
|
||||||
|
|
||||||
-- | Pure version of 'unpackb'.
|
|
||||||
unpackb' :: OBJECT a => ByteString -> Result a
|
|
||||||
unpackb' bs = unsafePerformIO $ unpackb bs
|
|
||||||
|
28
haskell/src/Data/MessagePack/Assoc.hs
Normal file
28
haskell/src/Data/MessagePack/Assoc.hs
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
{-# Language DeriveDataTypeable #-}
|
||||||
|
{-# Language GeneralizedNewtypeDeriving #-}
|
||||||
|
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : Data.MessagePack.Assoc
|
||||||
|
-- Copyright : (c) Daiki Handa, 2010
|
||||||
|
-- License : BSD3
|
||||||
|
--
|
||||||
|
-- Maintainer: tanaka.hideyuki@gmail.com
|
||||||
|
-- Stability : experimental
|
||||||
|
-- Portability: portable
|
||||||
|
--
|
||||||
|
-- MessagePack map labeling type
|
||||||
|
--
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
|
||||||
|
module Data.MessagePack.Assoc (
|
||||||
|
Assoc(..)
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Control.DeepSeq
|
||||||
|
import Data.Typeable
|
||||||
|
|
||||||
|
-- not defined for general Functor for performance reason.
|
||||||
|
-- (ie. you would want to write custom instances for each type using specialized mapM-like functions)
|
||||||
|
newtype Assoc a=Assoc{unAssoc :: a} deriving(Show,Eq,Ord,Typeable,NFData)
|
||||||
|
|
@@ -1,581 +0,0 @@
|
|||||||
{-# LANGUAGE CPP #-}
|
|
||||||
{-# LANGUAGE ForeignFunctionInterface #-}
|
|
||||||
|
|
||||||
--------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : Data.MessagePack.Base
|
|
||||||
-- Copyright : (c) Hideyuki Tanaka, 2009
|
|
||||||
-- License : BSD3
|
|
||||||
--
|
|
||||||
-- Maintainer: tanaka.hideyuki@gmail.com
|
|
||||||
-- Stability : experimental
|
|
||||||
-- Portability: portable
|
|
||||||
--
|
|
||||||
-- Low Level Interface to MessagePack C API
|
|
||||||
--
|
|
||||||
--------------------------------------------------------------------
|
|
||||||
|
|
||||||
module Data.MessagePack.Base(
|
|
||||||
-- * Simple Buffer
|
|
||||||
SimpleBuffer,
|
|
||||||
newSimpleBuffer,
|
|
||||||
simpleBufferData,
|
|
||||||
|
|
||||||
-- * Serializer
|
|
||||||
Packer,
|
|
||||||
newPacker,
|
|
||||||
|
|
||||||
packU8,
|
|
||||||
packU16,
|
|
||||||
packU32,
|
|
||||||
packU64,
|
|
||||||
packS8,
|
|
||||||
packS16,
|
|
||||||
packS32,
|
|
||||||
packS64,
|
|
||||||
|
|
||||||
packTrue,
|
|
||||||
packFalse,
|
|
||||||
|
|
||||||
packInt,
|
|
||||||
packDouble,
|
|
||||||
packNil,
|
|
||||||
packBool,
|
|
||||||
|
|
||||||
packArray,
|
|
||||||
packMap,
|
|
||||||
packRAW,
|
|
||||||
packRAWBody,
|
|
||||||
packRAW',
|
|
||||||
|
|
||||||
-- * Stream Deserializer
|
|
||||||
Unpacker,
|
|
||||||
defaultInitialBufferSize,
|
|
||||||
newUnpacker,
|
|
||||||
unpackerReserveBuffer,
|
|
||||||
unpackerBuffer,
|
|
||||||
unpackerBufferCapacity,
|
|
||||||
unpackerBufferConsumed,
|
|
||||||
unpackerFeed,
|
|
||||||
unpackerExecute,
|
|
||||||
unpackerData,
|
|
||||||
unpackerReleaseZone,
|
|
||||||
unpackerResetZone,
|
|
||||||
unpackerReset,
|
|
||||||
unpackerMessageSize,
|
|
||||||
|
|
||||||
-- * MessagePack Object
|
|
||||||
Object(..),
|
|
||||||
packObject,
|
|
||||||
|
|
||||||
UnpackReturn(..),
|
|
||||||
unpackObject,
|
|
||||||
|
|
||||||
-- * Memory Zone
|
|
||||||
Zone,
|
|
||||||
newZone,
|
|
||||||
freeZone,
|
|
||||||
withZone,
|
|
||||||
) where
|
|
||||||
|
|
||||||
import Control.Exception
|
|
||||||
import Control.Monad
|
|
||||||
import Data.ByteString (ByteString)
|
|
||||||
import qualified Data.ByteString as BS hiding (pack, unpack)
|
|
||||||
import Data.Int
|
|
||||||
import Data.Word
|
|
||||||
import Foreign.C
|
|
||||||
import Foreign.Concurrent
|
|
||||||
import Foreign.ForeignPtr hiding (newForeignPtr)
|
|
||||||
import Foreign.Marshal.Alloc
|
|
||||||
import Foreign.Marshal.Array
|
|
||||||
import Foreign.Ptr
|
|
||||||
import Foreign.Storable
|
|
||||||
|
|
||||||
#include <msgpack.h>
|
|
||||||
|
|
||||||
type SimpleBuffer = ForeignPtr ()
|
|
||||||
|
|
||||||
type WriteCallback = Ptr () -> CString -> CUInt -> IO CInt
|
|
||||||
|
|
||||||
-- | Create a new Simple Buffer. It will be deleted automatically.
|
|
||||||
newSimpleBuffer :: IO SimpleBuffer
|
|
||||||
newSimpleBuffer = do
|
|
||||||
ptr <- mallocBytes (#size msgpack_sbuffer)
|
|
||||||
fptr <- newForeignPtr ptr $ do
|
|
||||||
msgpack_sbuffer_destroy ptr
|
|
||||||
free ptr
|
|
||||||
withForeignPtr fptr $ \p ->
|
|
||||||
msgpack_sbuffer_init p
|
|
||||||
return fptr
|
|
||||||
|
|
||||||
-- | Get data of Simple Buffer.
|
|
||||||
simpleBufferData :: SimpleBuffer -> IO ByteString
|
|
||||||
simpleBufferData sb =
|
|
||||||
withForeignPtr sb $ \ptr -> do
|
|
||||||
size <- (#peek msgpack_sbuffer, size) ptr
|
|
||||||
dat <- (#peek msgpack_sbuffer, data) ptr
|
|
||||||
BS.packCStringLen (dat, fromIntegral (size :: CSize))
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_sbuffer_init_wrap" msgpack_sbuffer_init ::
|
|
||||||
Ptr () -> IO ()
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_sbuffer_destroy_wrap" msgpack_sbuffer_destroy ::
|
|
||||||
Ptr () -> IO ()
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_sbuffer_write_wrap" msgpack_sbuffer_write ::
|
|
||||||
WriteCallback
|
|
||||||
|
|
||||||
type Packer = ForeignPtr ()
|
|
||||||
|
|
||||||
-- | Create new Packer. It will be deleted automatically.
|
|
||||||
newPacker :: SimpleBuffer -> IO Packer
|
|
||||||
newPacker sbuf = do
|
|
||||||
cb <- wrap_callback msgpack_sbuffer_write
|
|
||||||
ptr <- withForeignPtr sbuf $ \ptr ->
|
|
||||||
msgpack_packer_new ptr cb
|
|
||||||
fptr <- newForeignPtr ptr $ do
|
|
||||||
msgpack_packer_free ptr
|
|
||||||
return fptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_packer_new_wrap" msgpack_packer_new ::
|
|
||||||
Ptr () -> FunPtr WriteCallback -> IO (Ptr ())
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_packer_free_wrap" msgpack_packer_free ::
|
|
||||||
Ptr () -> IO ()
|
|
||||||
|
|
||||||
foreign import ccall "wrapper" wrap_callback ::
|
|
||||||
WriteCallback -> IO (FunPtr WriteCallback)
|
|
||||||
|
|
||||||
packU8 :: Packer -> Word8 -> IO Int
|
|
||||||
packU8 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_uint8 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_uint8_wrap" msgpack_pack_uint8 ::
|
|
||||||
Ptr () -> Word8 -> IO CInt
|
|
||||||
|
|
||||||
packU16 :: Packer -> Word16 -> IO Int
|
|
||||||
packU16 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_uint16 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_uint16_wrap" msgpack_pack_uint16 ::
|
|
||||||
Ptr () -> Word16 -> IO CInt
|
|
||||||
|
|
||||||
packU32 :: Packer -> Word32 -> IO Int
|
|
||||||
packU32 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_uint32 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_uint32_wrap" msgpack_pack_uint32 ::
|
|
||||||
Ptr () -> Word32 -> IO CInt
|
|
||||||
|
|
||||||
packU64 :: Packer -> Word64 -> IO Int
|
|
||||||
packU64 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_uint64 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_uint64_wrap" msgpack_pack_uint64 ::
|
|
||||||
Ptr () -> Word64 -> IO CInt
|
|
||||||
|
|
||||||
packS8 :: Packer -> Int8 -> IO Int
|
|
||||||
packS8 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_int8 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_int8_wrap" msgpack_pack_int8 ::
|
|
||||||
Ptr () -> Int8 -> IO CInt
|
|
||||||
|
|
||||||
packS16 :: Packer -> Int16 -> IO Int
|
|
||||||
packS16 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_int16 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_int16_wrap" msgpack_pack_int16 ::
|
|
||||||
Ptr () -> Int16 -> IO CInt
|
|
||||||
|
|
||||||
packS32 :: Packer -> Int32 -> IO Int
|
|
||||||
packS32 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_int32 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_int32_wrap" msgpack_pack_int32 ::
|
|
||||||
Ptr () -> Int32 -> IO CInt
|
|
||||||
|
|
||||||
packS64 :: Packer -> Int64 -> IO Int
|
|
||||||
packS64 pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_int64 ptr n
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_int64_wrap" msgpack_pack_int64 ::
|
|
||||||
Ptr () -> Int64 -> IO CInt
|
|
||||||
|
|
||||||
-- | Pack an integral data.
|
|
||||||
packInt :: Integral a => Packer -> a -> IO Int
|
|
||||||
packInt pc n = packS64 pc $ fromIntegral n
|
|
||||||
|
|
||||||
-- | Pack a double data.
|
|
||||||
packDouble :: Packer -> Double -> IO Int
|
|
||||||
packDouble pc d =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_double ptr (realToFrac d)
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_double_wrap" msgpack_pack_double ::
|
|
||||||
Ptr () -> CDouble -> IO CInt
|
|
||||||
|
|
||||||
-- | Pack a nil.
|
|
||||||
packNil :: Packer -> IO Int
|
|
||||||
packNil pc =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_nil ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_nil_wrap" msgpack_pack_nil ::
|
|
||||||
Ptr () -> IO CInt
|
|
||||||
|
|
||||||
packTrue :: Packer -> IO Int
|
|
||||||
packTrue pc =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_true ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_true_wrap" msgpack_pack_true ::
|
|
||||||
Ptr () -> IO CInt
|
|
||||||
|
|
||||||
packFalse :: Packer -> IO Int
|
|
||||||
packFalse pc =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_false ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_false_wrap" msgpack_pack_false ::
|
|
||||||
Ptr () -> IO CInt
|
|
||||||
|
|
||||||
-- | Pack a bool data.
|
|
||||||
packBool :: Packer -> Bool -> IO Int
|
|
||||||
packBool pc True = packTrue pc
|
|
||||||
packBool pc False = packFalse pc
|
|
||||||
|
|
||||||
-- | 'packArray' @p n@ starts packing an array.
|
|
||||||
-- Next @n@ data will consist this array.
|
|
||||||
packArray :: Packer -> Int -> IO Int
|
|
||||||
packArray pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_array ptr (fromIntegral n)
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_array_wrap" msgpack_pack_array ::
|
|
||||||
Ptr () -> CUInt -> IO CInt
|
|
||||||
|
|
||||||
-- | 'packMap' @p n@ starts packing a map.
|
|
||||||
-- Next @n@ pairs of data (2*n data) will consist this map.
|
|
||||||
packMap :: Packer -> Int -> IO Int
|
|
||||||
packMap pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_map ptr (fromIntegral n)
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_map_wrap" msgpack_pack_map ::
|
|
||||||
Ptr () -> CUInt -> IO CInt
|
|
||||||
|
|
||||||
-- | 'packRAW' @p n@ starts packing a byte sequence.
|
|
||||||
-- Next total @n@ bytes of 'packRAWBody' call will consist this sequence.
|
|
||||||
packRAW :: Packer -> Int -> IO Int
|
|
||||||
packRAW pc n =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
msgpack_pack_raw ptr (fromIntegral n)
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_raw_wrap" msgpack_pack_raw ::
|
|
||||||
Ptr () -> CSize -> IO CInt
|
|
||||||
|
|
||||||
-- | Pack a byte sequence.
|
|
||||||
packRAWBody :: Packer -> ByteString -> IO Int
|
|
||||||
packRAWBody pc bs =
|
|
||||||
liftM fromIntegral $ withForeignPtr pc $ \ptr ->
|
|
||||||
BS.useAsCStringLen bs $ \(str, len) ->
|
|
||||||
msgpack_pack_raw_body ptr (castPtr str) (fromIntegral len)
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_pack_raw_body_wrap" msgpack_pack_raw_body ::
|
|
||||||
Ptr () -> Ptr () -> CSize -> IO CInt
|
|
||||||
|
|
||||||
-- | Pack a single byte stream. It calls 'packRAW' and 'packRAWBody'.
|
|
||||||
packRAW' :: Packer -> ByteString -> IO Int
|
|
||||||
packRAW' pc bs = do
|
|
||||||
packRAW pc (BS.length bs)
|
|
||||||
packRAWBody pc bs
|
|
||||||
|
|
||||||
type Unpacker = ForeignPtr ()
|
|
||||||
|
|
||||||
defaultInitialBufferSize :: Int
|
|
||||||
defaultInitialBufferSize = 32 * 1024 -- #const MSGPACK_UNPACKER_DEFAULT_INITIAL_BUFFER_SIZE
|
|
||||||
|
|
||||||
-- | 'newUnpacker' @initialBufferSize@ creates a new Unpacker. It will be deleted automatically.
|
|
||||||
newUnpacker :: Int -> IO Unpacker
|
|
||||||
newUnpacker initialBufferSize = do
|
|
||||||
ptr <- msgpack_unpacker_new (fromIntegral initialBufferSize)
|
|
||||||
fptr <- newForeignPtr ptr $ do
|
|
||||||
msgpack_unpacker_free ptr
|
|
||||||
return fptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_new" msgpack_unpacker_new ::
|
|
||||||
CSize -> IO (Ptr ())
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_free" msgpack_unpacker_free ::
|
|
||||||
Ptr() -> IO ()
|
|
||||||
|
|
||||||
-- | 'unpackerReserveBuffer' @up size@ reserves at least @size@ bytes of buffer.
|
|
||||||
unpackerReserveBuffer :: Unpacker -> Int -> IO Bool
|
|
||||||
unpackerReserveBuffer up size =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
liftM (/=0) $ msgpack_unpacker_reserve_buffer ptr (fromIntegral size)
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_reserve_buffer_wrap" msgpack_unpacker_reserve_buffer ::
|
|
||||||
Ptr () -> CSize -> IO CChar
|
|
||||||
|
|
||||||
-- | Get a pointer of unpacker buffer.
|
|
||||||
unpackerBuffer :: Unpacker -> IO (Ptr CChar)
|
|
||||||
unpackerBuffer up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
msgpack_unpacker_buffer ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_buffer_wrap" msgpack_unpacker_buffer ::
|
|
||||||
Ptr () -> IO (Ptr CChar)
|
|
||||||
|
|
||||||
-- | Get size of allocated buffer.
|
|
||||||
unpackerBufferCapacity :: Unpacker -> IO Int
|
|
||||||
unpackerBufferCapacity up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
liftM fromIntegral $ msgpack_unpacker_buffer_capacity ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_buffer_capacity_wrap" msgpack_unpacker_buffer_capacity ::
|
|
||||||
Ptr () -> IO CSize
|
|
||||||
|
|
||||||
-- | 'unpackerBufferConsumed' @up size@ notices that writed @size@ bytes to buffer.
|
|
||||||
unpackerBufferConsumed :: Unpacker -> Int -> IO ()
|
|
||||||
unpackerBufferConsumed up size =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
msgpack_unpacker_buffer_consumed ptr (fromIntegral size)
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_buffer_consumed_wrap" msgpack_unpacker_buffer_consumed ::
|
|
||||||
Ptr () -> CSize -> IO ()
|
|
||||||
|
|
||||||
-- | Write byte sequence to Unpacker. It is utility funciton, calls 'unpackerReserveBuffer', 'unpackerBuffer' and 'unpackerBufferConsumed'.
|
|
||||||
unpackerFeed :: Unpacker -> ByteString -> IO ()
|
|
||||||
unpackerFeed up bs =
|
|
||||||
BS.useAsCStringLen bs $ \(str, len) -> do
|
|
||||||
True <- unpackerReserveBuffer up len
|
|
||||||
ptr <- unpackerBuffer up
|
|
||||||
copyArray ptr str len
|
|
||||||
unpackerBufferConsumed up len
|
|
||||||
|
|
||||||
-- | Execute deserializing. It returns 0 when buffer contains not enough bytes, returns 1 when succeeded, returns negative value when it failed.
|
|
||||||
unpackerExecute :: Unpacker -> IO Int
|
|
||||||
unpackerExecute up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
liftM fromIntegral $ msgpack_unpacker_execute ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_execute" msgpack_unpacker_execute ::
|
|
||||||
Ptr () -> IO CInt
|
|
||||||
|
|
||||||
-- | Returns a deserialized object when 'unpackerExecute' returned 1.
|
|
||||||
unpackerData :: Unpacker -> IO Object
|
|
||||||
unpackerData up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
allocaBytes (#size msgpack_object) $ \pobj -> do
|
|
||||||
msgpack_unpacker_data ptr pobj
|
|
||||||
peekObject pobj
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_data_wrap" msgpack_unpacker_data ::
|
|
||||||
Ptr () -> Ptr () -> IO ()
|
|
||||||
|
|
||||||
-- | Release memory zone. The returned zone must be freed by calling 'freeZone'.
|
|
||||||
unpackerReleaseZone :: Unpacker -> IO Zone
|
|
||||||
unpackerReleaseZone up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
msgpack_unpacker_release_zone ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_release_zone" msgpack_unpacker_release_zone ::
|
|
||||||
Ptr () -> IO (Ptr ())
|
|
||||||
|
|
||||||
-- | Free memory zone used by Unapcker.
|
|
||||||
unpackerResetZone :: Unpacker -> IO ()
|
|
||||||
unpackerResetZone up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
msgpack_unpacker_reset_zone ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_reset_zone" msgpack_unpacker_reset_zone ::
|
|
||||||
Ptr () -> IO ()
|
|
||||||
|
|
||||||
-- | Reset Unpacker state except memory zone.
|
|
||||||
unpackerReset :: Unpacker -> IO ()
|
|
||||||
unpackerReset up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
msgpack_unpacker_reset ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_reset" msgpack_unpacker_reset ::
|
|
||||||
Ptr () -> IO ()
|
|
||||||
|
|
||||||
-- | Returns number of bytes of sequence of deserializing object.
|
|
||||||
unpackerMessageSize :: Unpacker -> IO Int
|
|
||||||
unpackerMessageSize up =
|
|
||||||
withForeignPtr up $ \ptr ->
|
|
||||||
liftM fromIntegral $ msgpack_unpacker_message_size ptr
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpacker_message_size_wrap" msgpack_unpacker_message_size ::
|
|
||||||
Ptr () -> IO CSize
|
|
||||||
|
|
||||||
type Zone = Ptr ()
|
|
||||||
|
|
||||||
-- | Create a new memory zone. It must be freed manually.
|
|
||||||
newZone :: IO Zone
|
|
||||||
newZone =
|
|
||||||
msgpack_zone_new (#const MSGPACK_ZONE_CHUNK_SIZE)
|
|
||||||
|
|
||||||
-- | Free a memory zone.
|
|
||||||
freeZone :: Zone -> IO ()
|
|
||||||
freeZone z =
|
|
||||||
msgpack_zone_free z
|
|
||||||
|
|
||||||
-- | Create a memory zone, then execute argument, then free memory zone.
|
|
||||||
withZone :: (Zone -> IO a) -> IO a
|
|
||||||
withZone z =
|
|
||||||
bracket newZone freeZone z
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_zone_new" msgpack_zone_new ::
|
|
||||||
CSize -> IO Zone
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_zone_free" msgpack_zone_free ::
|
|
||||||
Zone -> IO ()
|
|
||||||
|
|
||||||
-- | Object Representation of MessagePack data.
|
|
||||||
data Object =
|
|
||||||
ObjectNil
|
|
||||||
| ObjectBool Bool
|
|
||||||
| ObjectInteger Int
|
|
||||||
| ObjectDouble Double
|
|
||||||
| ObjectRAW ByteString
|
|
||||||
| ObjectArray [Object]
|
|
||||||
| ObjectMap [(Object, Object)]
|
|
||||||
deriving (Show)
|
|
||||||
|
|
||||||
peekObject :: Ptr a -> IO Object
|
|
||||||
peekObject ptr = do
|
|
||||||
typ <- (#peek msgpack_object, type) ptr
|
|
||||||
case (typ :: CInt) of
|
|
||||||
(#const MSGPACK_OBJECT_NIL) ->
|
|
||||||
return ObjectNil
|
|
||||||
(#const MSGPACK_OBJECT_BOOLEAN) ->
|
|
||||||
peekObjectBool ptr
|
|
||||||
(#const MSGPACK_OBJECT_POSITIVE_INTEGER) ->
|
|
||||||
peekObjectPositiveInteger ptr
|
|
||||||
(#const MSGPACK_OBJECT_NEGATIVE_INTEGER) ->
|
|
||||||
peekObjectNegativeInteger ptr
|
|
||||||
(#const MSGPACK_OBJECT_DOUBLE) ->
|
|
||||||
peekObjectDouble ptr
|
|
||||||
(#const MSGPACK_OBJECT_RAW) ->
|
|
||||||
peekObjectRAW ptr
|
|
||||||
(#const MSGPACK_OBJECT_ARRAY) ->
|
|
||||||
peekObjectArray ptr
|
|
||||||
(#const MSGPACK_OBJECT_MAP) ->
|
|
||||||
peekObjectMap ptr
|
|
||||||
_ ->
|
|
||||||
fail "peekObject: unknown object type"
|
|
||||||
|
|
||||||
peekObjectBool :: Ptr a -> IO Object
|
|
||||||
peekObjectBool ptr = do
|
|
||||||
b <- (#peek msgpack_object, via.boolean) ptr
|
|
||||||
return $ ObjectBool $ (b :: CUChar) /= 0
|
|
||||||
|
|
||||||
peekObjectPositiveInteger :: Ptr a -> IO Object
|
|
||||||
peekObjectPositiveInteger ptr = do
|
|
||||||
n <- (#peek msgpack_object, via.u64) ptr
|
|
||||||
return $ ObjectInteger $ fromIntegral (n :: Word64)
|
|
||||||
|
|
||||||
peekObjectNegativeInteger :: Ptr a -> IO Object
|
|
||||||
peekObjectNegativeInteger ptr = do
|
|
||||||
n <- (#peek msgpack_object, via.i64) ptr
|
|
||||||
return $ ObjectInteger $ fromIntegral (n :: Int64)
|
|
||||||
|
|
||||||
peekObjectDouble :: Ptr a -> IO Object
|
|
||||||
peekObjectDouble ptr = do
|
|
||||||
d <- (#peek msgpack_object, via.dec) ptr
|
|
||||||
return $ ObjectDouble $ realToFrac (d :: CDouble)
|
|
||||||
|
|
||||||
peekObjectRAW :: Ptr a -> IO Object
|
|
||||||
peekObjectRAW ptr = do
|
|
||||||
size <- (#peek msgpack_object, via.raw.size) ptr
|
|
||||||
p <- (#peek msgpack_object, via.raw.ptr) ptr
|
|
||||||
bs <- BS.packCStringLen (p, fromIntegral (size :: Word32))
|
|
||||||
return $ ObjectRAW bs
|
|
||||||
|
|
||||||
peekObjectArray :: Ptr a -> IO Object
|
|
||||||
peekObjectArray ptr = do
|
|
||||||
size <- (#peek msgpack_object, via.array.size) ptr
|
|
||||||
p <- (#peek msgpack_object, via.array.ptr) ptr
|
|
||||||
objs <- mapM (\i -> peekObject $ p `plusPtr`
|
|
||||||
((#size msgpack_object) * i))
|
|
||||||
[0..size-1]
|
|
||||||
return $ ObjectArray objs
|
|
||||||
|
|
||||||
peekObjectMap :: Ptr a -> IO Object
|
|
||||||
peekObjectMap ptr = do
|
|
||||||
size <- (#peek msgpack_object, via.map.size) ptr
|
|
||||||
p <- (#peek msgpack_object, via.map.ptr) ptr
|
|
||||||
dat <- mapM (\i -> peekObjectKV $ p `plusPtr`
|
|
||||||
((#size msgpack_object_kv) * i))
|
|
||||||
[0..size-1]
|
|
||||||
return $ ObjectMap dat
|
|
||||||
|
|
||||||
peekObjectKV :: Ptr a -> IO (Object, Object)
|
|
||||||
peekObjectKV ptr = do
|
|
||||||
k <- peekObject $ ptr `plusPtr` (#offset msgpack_object_kv, key)
|
|
||||||
v <- peekObject $ ptr `plusPtr` (#offset msgpack_object_kv, val)
|
|
||||||
return (k, v)
|
|
||||||
|
|
||||||
-- | Pack a Object.
|
|
||||||
packObject :: Packer -> Object -> IO ()
|
|
||||||
packObject pc ObjectNil = packNil pc >> return ()
|
|
||||||
|
|
||||||
packObject pc (ObjectBool b) = packBool pc b >> return ()
|
|
||||||
|
|
||||||
packObject pc (ObjectInteger n) = packInt pc n >> return ()
|
|
||||||
|
|
||||||
packObject pc (ObjectDouble d) = packDouble pc d >> return ()
|
|
||||||
|
|
||||||
packObject pc (ObjectRAW bs) = packRAW' pc bs >> return ()
|
|
||||||
|
|
||||||
packObject pc (ObjectArray ls) = do
|
|
||||||
packArray pc (length ls)
|
|
||||||
mapM_ (packObject pc) ls
|
|
||||||
|
|
||||||
packObject pc (ObjectMap ls) = do
|
|
||||||
packMap pc (length ls)
|
|
||||||
mapM_ (\(a, b) -> packObject pc a >> packObject pc b) ls
|
|
||||||
|
|
||||||
data UnpackReturn =
|
|
||||||
UnpackContinue -- ^ not enough bytes to unpack object
|
|
||||||
| UnpackParseError -- ^ got invalid bytes
|
|
||||||
| UnpackError -- ^ other error
|
|
||||||
deriving (Eq, Show)
|
|
||||||
|
|
||||||
-- | Unpack a single MessagePack object from byte sequence.
|
|
||||||
unpackObject :: Zone -> ByteString -> IO (Either UnpackReturn (Int, Object))
|
|
||||||
unpackObject z dat =
|
|
||||||
allocaBytes (#size msgpack_object) $ \ptr ->
|
|
||||||
BS.useAsCStringLen dat $ \(str, len) ->
|
|
||||||
alloca $ \poff -> do
|
|
||||||
ret <- msgpack_unpack str (fromIntegral len) poff z ptr
|
|
||||||
case ret of
|
|
||||||
(#const MSGPACK_UNPACK_SUCCESS) -> do
|
|
||||||
off <- peek poff
|
|
||||||
obj <- peekObject ptr
|
|
||||||
return $ Right (fromIntegral off, obj)
|
|
||||||
(#const MSGPACK_UNPACK_EXTRA_BYTES) -> do
|
|
||||||
off <- peek poff
|
|
||||||
obj <- peekObject ptr
|
|
||||||
return $ Right (fromIntegral off, obj)
|
|
||||||
(#const MSGPACK_UNPACK_CONTINUE) ->
|
|
||||||
return $ Left UnpackContinue
|
|
||||||
(#const MSGPACK_UNPACK_PARSE_ERROR) ->
|
|
||||||
return $ Left UnpackParseError
|
|
||||||
_ ->
|
|
||||||
return $ Left UnpackError
|
|
||||||
|
|
||||||
foreign import ccall "msgpack_unpack" msgpack_unpack ::
|
|
||||||
Ptr CChar -> CSize -> Ptr CSize -> Zone -> Ptr () -> IO CInt
|
|
@@ -1,97 +0,0 @@
|
|||||||
{-# LANGUAGE TypeSynonymInstances #-}
|
|
||||||
{-# LANGUAGE FlexibleInstances #-}
|
|
||||||
{-# LANGUAGE OverlappingInstances #-}
|
|
||||||
{-# LANGUAGE IncoherentInstances #-}
|
|
||||||
|
|
||||||
--------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : Data.MessagePack.Class
|
|
||||||
-- Copyright : (c) Hideyuki Tanaka, 2009
|
|
||||||
-- License : BSD3
|
|
||||||
--
|
|
||||||
-- Maintainer: tanaka.hideyuki@gmail.com
|
|
||||||
-- Stability : experimental
|
|
||||||
-- Portability: portable
|
|
||||||
--
|
|
||||||
-- Serializing Haskell values to and from MessagePack Objects.
|
|
||||||
--
|
|
||||||
--------------------------------------------------------------------
|
|
||||||
|
|
||||||
module Data.MessagePack.Class(
|
|
||||||
-- * Serialization to and from Object
|
|
||||||
OBJECT(..),
|
|
||||||
Result,
|
|
||||||
pack,
|
|
||||||
) where
|
|
||||||
|
|
||||||
import Control.Monad.Error
|
|
||||||
import Data.ByteString.Char8 (ByteString)
|
|
||||||
import qualified Data.ByteString.Char8 as C8
|
|
||||||
import Data.Either
|
|
||||||
|
|
||||||
import Data.MessagePack.Base
|
|
||||||
|
|
||||||
-- | The class of types serializable to and from MessagePack object
|
|
||||||
class OBJECT a where
|
|
||||||
toObject :: a -> Object
|
|
||||||
fromObject :: Object -> Result a
|
|
||||||
|
|
||||||
-- | A type for parser results
|
|
||||||
type Result a = Either String a
|
|
||||||
|
|
||||||
instance OBJECT Object where
|
|
||||||
toObject = id
|
|
||||||
fromObject = Right
|
|
||||||
|
|
||||||
fromObjectError :: String
|
|
||||||
fromObjectError = "fromObject: cannot cast"
|
|
||||||
|
|
||||||
instance OBJECT Int where
|
|
||||||
toObject = ObjectInteger
|
|
||||||
fromObject (ObjectInteger n) = Right n
|
|
||||||
fromObject _ = Left fromObjectError
|
|
||||||
|
|
||||||
instance OBJECT Bool where
|
|
||||||
toObject = ObjectBool
|
|
||||||
fromObject (ObjectBool b) = Right b
|
|
||||||
fromObject _ = Left fromObjectError
|
|
||||||
|
|
||||||
instance OBJECT Double where
|
|
||||||
toObject = ObjectDouble
|
|
||||||
fromObject (ObjectDouble d) = Right d
|
|
||||||
fromObject _ = Left fromObjectError
|
|
||||||
|
|
||||||
instance OBJECT ByteString where
|
|
||||||
toObject = ObjectRAW
|
|
||||||
fromObject (ObjectRAW bs) = Right bs
|
|
||||||
fromObject _ = Left fromObjectError
|
|
||||||
|
|
||||||
instance OBJECT String where
|
|
||||||
toObject = toObject . C8.pack
|
|
||||||
fromObject obj = liftM C8.unpack $ fromObject obj
|
|
||||||
|
|
||||||
instance OBJECT a => OBJECT [a] where
|
|
||||||
toObject = ObjectArray . map toObject
|
|
||||||
fromObject (ObjectArray arr) =
|
|
||||||
mapM fromObject arr
|
|
||||||
fromObject _ =
|
|
||||||
Left fromObjectError
|
|
||||||
|
|
||||||
instance (OBJECT a, OBJECT b) => OBJECT [(a, b)] where
|
|
||||||
toObject =
|
|
||||||
ObjectMap . map (\(a, b) -> (toObject a, toObject b))
|
|
||||||
fromObject (ObjectMap mem) = do
|
|
||||||
mapM (\(a, b) -> liftM2 (,) (fromObject a) (fromObject b)) mem
|
|
||||||
fromObject _ =
|
|
||||||
Left fromObjectError
|
|
||||||
|
|
||||||
instance OBJECT a => OBJECT (Maybe a) where
|
|
||||||
toObject (Just a) = toObject a
|
|
||||||
toObject Nothing = ObjectNil
|
|
||||||
|
|
||||||
fromObject ObjectNil = return Nothing
|
|
||||||
fromObject obj = liftM Just $ fromObject obj
|
|
||||||
|
|
||||||
-- | Pack a serializable Haskell value.
|
|
||||||
pack :: OBJECT a => Packer -> a -> IO ()
|
|
||||||
pack pc = packObject pc . toObject
|
|
106
haskell/src/Data/MessagePack/Derive.hs
Normal file
106
haskell/src/Data/MessagePack/Derive.hs
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
{-# Language TemplateHaskell #-}
|
||||||
|
|
||||||
|
module Data.MessagePack.Derive (
|
||||||
|
derivePack,
|
||||||
|
deriveUnpack,
|
||||||
|
deriveObject,
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Control.Applicative
|
||||||
|
import Control.Monad
|
||||||
|
import Language.Haskell.TH
|
||||||
|
|
||||||
|
import Data.MessagePack.Pack
|
||||||
|
import Data.MessagePack.Unpack
|
||||||
|
import Data.MessagePack.Object
|
||||||
|
|
||||||
|
deriveUnpack :: Name -> Q [Dec]
|
||||||
|
deriveUnpack typName = do
|
||||||
|
TyConI (DataD _ name _ cons _) <- reify typName
|
||||||
|
|
||||||
|
return
|
||||||
|
[ InstanceD [] (AppT (ConT ''Unpackable) (ConT name))
|
||||||
|
[ FunD 'get [Clause [] (NormalB $ ch $ map body cons) []]
|
||||||
|
]]
|
||||||
|
|
||||||
|
where
|
||||||
|
body (NormalC conName elms) =
|
||||||
|
DoE $
|
||||||
|
tupOrListP (map VarP names) (VarE 'get) ++
|
||||||
|
[ NoBindS $ AppE (VarE 'return) $ foldl AppE (ConE conName) $ map VarE names ]
|
||||||
|
where
|
||||||
|
names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms
|
||||||
|
|
||||||
|
body (RecC conName elms) =
|
||||||
|
body (NormalC conName $ map (\(_, b, c) -> (b, c)) elms)
|
||||||
|
|
||||||
|
ch = foldl1 (\e f -> AppE (AppE (VarE '(<|>)) e) f)
|
||||||
|
|
||||||
|
derivePack :: Name -> Q [Dec]
|
||||||
|
derivePack typName = do
|
||||||
|
TyConI (DataD _ name _ cons _) <- reify typName
|
||||||
|
|
||||||
|
return
|
||||||
|
[ InstanceD [] (AppT (ConT ''Packable) (ConT name))
|
||||||
|
[ FunD 'put (map body cons)
|
||||||
|
]]
|
||||||
|
|
||||||
|
where
|
||||||
|
body (NormalC conName elms) =
|
||||||
|
Clause
|
||||||
|
[ ConP conName $ map VarP names ]
|
||||||
|
(NormalB $ AppE (VarE 'put) $ tupOrListE $ map VarE names) []
|
||||||
|
where
|
||||||
|
names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms
|
||||||
|
|
||||||
|
body (RecC conName elms) =
|
||||||
|
body (NormalC conName $ map (\(_, b, c) -> (b, c)) elms)
|
||||||
|
|
||||||
|
deriveObject :: Name -> Q [Dec]
|
||||||
|
deriveObject typName = do
|
||||||
|
g <- derivePack typName
|
||||||
|
p <- deriveUnpack typName
|
||||||
|
|
||||||
|
TyConI (DataD _ name _ cons _) <- reify typName
|
||||||
|
let o = InstanceD [] (AppT (ConT ''OBJECT) (ConT name))
|
||||||
|
[ FunD 'toObject (map toObjectBody cons),
|
||||||
|
FunD 'tryFromObject [Clause [ VarP oname ]
|
||||||
|
(NormalB $ ch $ map tryFromObjectBody cons) []]]
|
||||||
|
|
||||||
|
return $ g ++ p ++ [o]
|
||||||
|
where
|
||||||
|
toObjectBody (NormalC conName elms) =
|
||||||
|
Clause
|
||||||
|
[ ConP conName $ map VarP names ]
|
||||||
|
(NormalB $ AppE (VarE 'toObject) $ tupOrListE $ map VarE names) []
|
||||||
|
where
|
||||||
|
names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms
|
||||||
|
toObjectBody (RecC conName elms) =
|
||||||
|
toObjectBody (NormalC conName $ map (\(_, b, c) -> (b, c)) elms)
|
||||||
|
|
||||||
|
tryFromObjectBody (NormalC conName elms) =
|
||||||
|
DoE $
|
||||||
|
tupOrListP (map VarP names) (AppE (VarE 'tryFromObject) (VarE oname)) ++
|
||||||
|
[ NoBindS $ AppE (VarE 'return) $ foldl AppE (ConE conName) $ map VarE names ]
|
||||||
|
where
|
||||||
|
names = zipWith (\ix _ -> mkName $ "a" ++ show (ix :: Int)) [1..] elms
|
||||||
|
tryFromObjectBody (RecC conName elms) =
|
||||||
|
tryFromObjectBody (NormalC conName $ map (\(_, b, c) -> (b, c)) elms)
|
||||||
|
|
||||||
|
oname = mkName "o"
|
||||||
|
ch = foldl1 (\e f -> AppE (AppE (VarE '(<|>)) e) f)
|
||||||
|
|
||||||
|
tupOrListP :: [Pat] -> Exp -> [Stmt]
|
||||||
|
tupOrListP ls e
|
||||||
|
| length ls == 0 =
|
||||||
|
let lsname = mkName "ls" in
|
||||||
|
[ BindS (VarP lsname) e
|
||||||
|
, NoBindS $ AppE (VarE 'guard) $ AppE (VarE 'null) $ SigE (VarE lsname) (AppT ListT (ConT ''())) ]
|
||||||
|
| length ls == 1 = [ BindS (ListP ls) e ]
|
||||||
|
| otherwise = [ BindS (TupP ls) e ]
|
||||||
|
|
||||||
|
tupOrListE :: [Exp] -> Exp
|
||||||
|
tupOrListE ls
|
||||||
|
| length ls == 0 = SigE (ListE []) (AppT ListT (ConT ''()))
|
||||||
|
| length ls == 1 = ListE ls
|
||||||
|
| otherwise = TupE ls
|
@@ -1,59 +0,0 @@
|
|||||||
--------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : Data.MessagePack.Feed
|
|
||||||
-- Copyright : (c) Hideyuki Tanaka, 2009
|
|
||||||
-- License : BSD3
|
|
||||||
--
|
|
||||||
-- Maintainer: tanaka.hideyuki@gmail.com
|
|
||||||
-- Stability : experimental
|
|
||||||
-- Portability: portable
|
|
||||||
--
|
|
||||||
-- Feeders for Stream Deserializers
|
|
||||||
--
|
|
||||||
--------------------------------------------------------------------
|
|
||||||
|
|
||||||
module Data.MessagePack.Feed(
|
|
||||||
-- * Feeder type
|
|
||||||
Feeder,
|
|
||||||
-- * Feeders
|
|
||||||
feederFromHandle,
|
|
||||||
feederFromFile,
|
|
||||||
feederFromString,
|
|
||||||
) where
|
|
||||||
|
|
||||||
import Control.Monad
|
|
||||||
import Data.ByteString (ByteString)
|
|
||||||
import qualified Data.ByteString as BS
|
|
||||||
import Data.IORef
|
|
||||||
import System.IO
|
|
||||||
|
|
||||||
-- | Feeder returns Just ByteString when bytes remains, otherwise Nothing.
|
|
||||||
type Feeder = IO (Maybe ByteString)
|
|
||||||
|
|
||||||
-- | Feeder from Handle
|
|
||||||
feederFromHandle :: Handle -> IO Feeder
|
|
||||||
feederFromHandle h = return $ do
|
|
||||||
bs <- BS.hGet h bufSize
|
|
||||||
if BS.length bs > 0
|
|
||||||
then return $ Just bs
|
|
||||||
else do
|
|
||||||
hClose h
|
|
||||||
return Nothing
|
|
||||||
where
|
|
||||||
bufSize = 4096
|
|
||||||
|
|
||||||
-- | Feeder from File
|
|
||||||
feederFromFile :: FilePath -> IO Feeder
|
|
||||||
feederFromFile path =
|
|
||||||
openFile path ReadMode >>= feederFromHandle
|
|
||||||
|
|
||||||
-- | Feeder from ByteString
|
|
||||||
feederFromString :: ByteString -> IO Feeder
|
|
||||||
feederFromString bs = do
|
|
||||||
r <- newIORef (Just bs)
|
|
||||||
return $ f r
|
|
||||||
where
|
|
||||||
f r = do
|
|
||||||
mb <- readIORef r
|
|
||||||
writeIORef r Nothing
|
|
||||||
return mb
|
|
28
haskell/src/Data/MessagePack/Internal/Utf8.hs
Normal file
28
haskell/src/Data/MessagePack/Internal/Utf8.hs
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
module Data.MessagePack.Internal.Utf8 (
|
||||||
|
encodeUtf8,
|
||||||
|
decodeUtf8,
|
||||||
|
skipChar,
|
||||||
|
toLBS,
|
||||||
|
fromLBS,
|
||||||
|
) where
|
||||||
|
|
||||||
|
import qualified Data.ByteString as B
|
||||||
|
import qualified Data.ByteString.Lazy as BL
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Text.Encoding as T
|
||||||
|
import qualified Data.Text.Encoding.Error as T
|
||||||
|
|
||||||
|
encodeUtf8 :: String -> B.ByteString
|
||||||
|
encodeUtf8 = T.encodeUtf8 . T.pack
|
||||||
|
|
||||||
|
decodeUtf8 :: B.ByteString -> String
|
||||||
|
decodeUtf8 = T.unpack . T.decodeUtf8With skipChar
|
||||||
|
|
||||||
|
skipChar :: T.OnDecodeError
|
||||||
|
skipChar _ _ = Nothing
|
||||||
|
|
||||||
|
toLBS :: B.ByteString -> BL.ByteString
|
||||||
|
toLBS bs = BL.fromChunks [bs]
|
||||||
|
|
||||||
|
fromLBS :: BL.ByteString -> B.ByteString
|
||||||
|
fromLBS = B.concat . BL.toChunks
|
@@ -1,153 +0,0 @@
|
|||||||
--------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : Data.MessagePack.Monad
|
|
||||||
-- Copyright : (c) Hideyuki Tanaka, 2009
|
|
||||||
-- License : BSD3
|
|
||||||
--
|
|
||||||
-- Maintainer: tanaka.hideyuki@gmail.com
|
|
||||||
-- Stability : experimental
|
|
||||||
-- Portability: portable
|
|
||||||
--
|
|
||||||
-- Monadic Stream Serializers and Deserializers
|
|
||||||
--
|
|
||||||
--------------------------------------------------------------------
|
|
||||||
|
|
||||||
module Data.MessagePack.Monad(
|
|
||||||
-- * Classes
|
|
||||||
MonadPacker(..),
|
|
||||||
MonadUnpacker(..),
|
|
||||||
|
|
||||||
-- * Packer and Unpacker type
|
|
||||||
PackerT(..),
|
|
||||||
UnpackerT(..),
|
|
||||||
|
|
||||||
-- * Packers
|
|
||||||
packToString,
|
|
||||||
packToHandle,
|
|
||||||
packToFile,
|
|
||||||
|
|
||||||
-- * Unpackers
|
|
||||||
unpackFrom,
|
|
||||||
unpackFromString,
|
|
||||||
unpackFromHandle,
|
|
||||||
unpackFromFile,
|
|
||||||
) where
|
|
||||||
|
|
||||||
import Control.Monad
|
|
||||||
import Control.Monad.Trans
|
|
||||||
import Data.ByteString (ByteString)
|
|
||||||
import qualified Data.ByteString as BS
|
|
||||||
import System.IO
|
|
||||||
|
|
||||||
import Data.MessagePack.Base hiding (Unpacker)
|
|
||||||
import qualified Data.MessagePack.Base as Base
|
|
||||||
import Data.MessagePack.Class
|
|
||||||
import Data.MessagePack.Feed
|
|
||||||
|
|
||||||
class Monad m => MonadPacker m where
|
|
||||||
-- | Serialize a object
|
|
||||||
put :: OBJECT a => a -> m ()
|
|
||||||
|
|
||||||
class Monad m => MonadUnpacker m where
|
|
||||||
-- | Deserialize a object
|
|
||||||
get :: OBJECT a => m a
|
|
||||||
|
|
||||||
-- | Serializer Type
|
|
||||||
newtype PackerT m r = PackerT { runPackerT :: Base.Packer -> m r }
|
|
||||||
|
|
||||||
instance Monad m => Monad (PackerT m) where
|
|
||||||
a >>= b =
|
|
||||||
PackerT $ \pc -> do
|
|
||||||
r <- runPackerT a pc
|
|
||||||
runPackerT (b r) pc
|
|
||||||
|
|
||||||
return r =
|
|
||||||
PackerT $ \_ -> return r
|
|
||||||
|
|
||||||
instance MonadTrans PackerT where
|
|
||||||
lift m = PackerT $ \_ -> m
|
|
||||||
|
|
||||||
instance MonadIO m => MonadIO (PackerT m) where
|
|
||||||
liftIO = lift . liftIO
|
|
||||||
|
|
||||||
instance MonadIO m => MonadPacker (PackerT m) where
|
|
||||||
put v = PackerT $ \pc -> liftIO $ do
|
|
||||||
pack pc v
|
|
||||||
|
|
||||||
-- | Execute given serializer and returns byte sequence.
|
|
||||||
packToString :: MonadIO m => PackerT m r -> m ByteString
|
|
||||||
packToString m = do
|
|
||||||
sb <- liftIO $ newSimpleBuffer
|
|
||||||
pc <- liftIO $ newPacker sb
|
|
||||||
runPackerT m pc
|
|
||||||
liftIO $ simpleBufferData sb
|
|
||||||
|
|
||||||
-- | Execcute given serializer and write byte sequence to Handle.
|
|
||||||
packToHandle :: MonadIO m => Handle -> PackerT m r -> m ()
|
|
||||||
packToHandle h m = do
|
|
||||||
sb <- packToString m
|
|
||||||
liftIO $ BS.hPut h sb
|
|
||||||
liftIO $ hFlush h
|
|
||||||
|
|
||||||
-- | Execute given serializer and write byte sequence to file.
|
|
||||||
packToFile :: MonadIO m => FilePath -> PackerT m r -> m ()
|
|
||||||
packToFile p m = do
|
|
||||||
sb <- packToString m
|
|
||||||
liftIO $ BS.writeFile p sb
|
|
||||||
|
|
||||||
-- | Deserializer type
|
|
||||||
newtype UnpackerT m r = UnpackerT { runUnpackerT :: Base.Unpacker -> Feeder -> m r }
|
|
||||||
|
|
||||||
instance Monad m => Monad (UnpackerT m) where
|
|
||||||
a >>= b =
|
|
||||||
UnpackerT $ \up feed -> do
|
|
||||||
r <- runUnpackerT a up feed
|
|
||||||
runUnpackerT (b r) up feed
|
|
||||||
|
|
||||||
return r =
|
|
||||||
UnpackerT $ \_ _ -> return r
|
|
||||||
|
|
||||||
instance MonadTrans UnpackerT where
|
|
||||||
lift m = UnpackerT $ \_ _ -> m
|
|
||||||
|
|
||||||
instance MonadIO m => MonadIO (UnpackerT m) where
|
|
||||||
liftIO = lift . liftIO
|
|
||||||
|
|
||||||
instance MonadIO m => MonadUnpacker (UnpackerT m) where
|
|
||||||
get = UnpackerT $ \up feed -> liftIO $ do
|
|
||||||
resp <- unpackerExecute up
|
|
||||||
guard $ resp>=0
|
|
||||||
when (resp==0) $ do
|
|
||||||
Just bs <- feed
|
|
||||||
unpackerFeed up bs
|
|
||||||
resp2 <- unpackerExecute up
|
|
||||||
guard $ resp2==1
|
|
||||||
obj <- unpackerData up
|
|
||||||
freeZone =<< unpackerReleaseZone up
|
|
||||||
unpackerReset up
|
|
||||||
let Right r = fromObject obj
|
|
||||||
return r
|
|
||||||
|
|
||||||
-- | Execute deserializer using given feeder.
|
|
||||||
unpackFrom :: MonadIO m => Feeder -> UnpackerT m r -> m r
|
|
||||||
unpackFrom f m = do
|
|
||||||
up <- liftIO $ newUnpacker defaultInitialBufferSize
|
|
||||||
runUnpackerT m up f
|
|
||||||
|
|
||||||
-- | Execute deserializer using given handle.
|
|
||||||
unpackFromHandle :: MonadIO m => Handle -> UnpackerT m r -> m r
|
|
||||||
unpackFromHandle h m =
|
|
||||||
flip unpackFrom m =<< liftIO (feederFromHandle h)
|
|
||||||
|
|
||||||
-- | Execute deserializer using given file content.
|
|
||||||
unpackFromFile :: MonadIO m => FilePath -> UnpackerT m r -> m r
|
|
||||||
unpackFromFile p m = do
|
|
||||||
h <- liftIO $ openFile p ReadMode
|
|
||||||
r <- flip unpackFrom m =<< liftIO (feederFromHandle h)
|
|
||||||
liftIO $ hClose h
|
|
||||||
return r
|
|
||||||
|
|
||||||
-- | Execute deserializer from given byte sequence.
|
|
||||||
unpackFromString :: MonadIO m => ByteString -> UnpackerT m r -> m r
|
|
||||||
unpackFromString bs m = do
|
|
||||||
flip unpackFrom m =<< liftIO (feederFromString bs)
|
|
334
haskell/src/Data/MessagePack/Object.hs
Normal file
334
haskell/src/Data/MessagePack/Object.hs
Normal file
@@ -0,0 +1,334 @@
|
|||||||
|
{-# Language TypeSynonymInstances #-}
|
||||||
|
{-# Language FlexibleInstances #-}
|
||||||
|
{-# Language IncoherentInstances #-}
|
||||||
|
{-# Language DeriveDataTypeable #-}
|
||||||
|
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : Data.MessagePack.Object
|
||||||
|
-- Copyright : (c) Hideyuki Tanaka, 2009-2010
|
||||||
|
-- License : BSD3
|
||||||
|
--
|
||||||
|
-- Maintainer: tanaka.hideyuki@gmail.com
|
||||||
|
-- Stability : experimental
|
||||||
|
-- Portability: portable
|
||||||
|
--
|
||||||
|
-- MessagePack object definition
|
||||||
|
--
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
|
||||||
|
module Data.MessagePack.Object(
|
||||||
|
-- * MessagePack Object
|
||||||
|
Object(..),
|
||||||
|
|
||||||
|
-- * Serialization to and from Object
|
||||||
|
OBJECT(..),
|
||||||
|
-- Result,
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Control.DeepSeq
|
||||||
|
import Control.Exception
|
||||||
|
import Control.Monad
|
||||||
|
import Control.Monad.Trans.Error ()
|
||||||
|
import qualified Data.Attoparsec as A
|
||||||
|
import qualified Data.ByteString as B
|
||||||
|
import qualified Data.ByteString.Lazy as BL
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Text.Encoding as T
|
||||||
|
import qualified Data.Text.Lazy as TL
|
||||||
|
import qualified Data.Text.Lazy.Encoding as TL
|
||||||
|
import Data.Typeable
|
||||||
|
|
||||||
|
import Data.MessagePack.Assoc
|
||||||
|
import Data.MessagePack.Pack
|
||||||
|
import Data.MessagePack.Unpack
|
||||||
|
import Data.MessagePack.Internal.Utf8
|
||||||
|
|
||||||
|
-- | Object Representation of MessagePack data.
|
||||||
|
data Object
|
||||||
|
= ObjectNil
|
||||||
|
| ObjectBool Bool
|
||||||
|
| ObjectInteger Int
|
||||||
|
| ObjectFloat Float
|
||||||
|
| ObjectDouble Double
|
||||||
|
| ObjectRAW B.ByteString
|
||||||
|
| ObjectArray [Object]
|
||||||
|
| ObjectMap [(Object, Object)]
|
||||||
|
deriving (Show, Eq, Ord, Typeable)
|
||||||
|
|
||||||
|
|
||||||
|
instance NFData Object where
|
||||||
|
rnf obj =
|
||||||
|
case obj of
|
||||||
|
ObjectNil -> ()
|
||||||
|
ObjectBool b -> rnf b
|
||||||
|
ObjectInteger n -> rnf n
|
||||||
|
ObjectFloat f -> rnf f
|
||||||
|
ObjectDouble d -> rnf d
|
||||||
|
ObjectRAW bs -> bs `seq` ()
|
||||||
|
ObjectArray a -> rnf a
|
||||||
|
ObjectMap m -> rnf m
|
||||||
|
|
||||||
|
|
||||||
|
instance Unpackable Object where
|
||||||
|
get =
|
||||||
|
A.choice
|
||||||
|
[ liftM ObjectInteger get
|
||||||
|
, liftM (\() -> ObjectNil) get
|
||||||
|
, liftM ObjectBool get
|
||||||
|
, liftM ObjectFloat get
|
||||||
|
, liftM ObjectDouble get
|
||||||
|
, liftM ObjectRAW get
|
||||||
|
, liftM ObjectArray get
|
||||||
|
, liftM (ObjectMap . unAssoc) get
|
||||||
|
]
|
||||||
|
|
||||||
|
instance Packable Object where
|
||||||
|
put obj =
|
||||||
|
case obj of
|
||||||
|
ObjectInteger n ->
|
||||||
|
put n
|
||||||
|
ObjectNil ->
|
||||||
|
put ()
|
||||||
|
ObjectBool b ->
|
||||||
|
put b
|
||||||
|
ObjectFloat f ->
|
||||||
|
put f
|
||||||
|
ObjectDouble d ->
|
||||||
|
put d
|
||||||
|
ObjectRAW raw ->
|
||||||
|
put raw
|
||||||
|
ObjectArray arr ->
|
||||||
|
put arr
|
||||||
|
ObjectMap m ->
|
||||||
|
put $ Assoc m
|
||||||
|
|
||||||
|
-- | The class of types serializable to and from MessagePack object
|
||||||
|
class (Unpackable a, Packable a) => OBJECT a where
|
||||||
|
-- | Encode a value to MessagePack object
|
||||||
|
toObject :: a -> Object
|
||||||
|
toObject = unpack . pack
|
||||||
|
|
||||||
|
-- | Decode a value from MessagePack object
|
||||||
|
fromObject :: Object -> a
|
||||||
|
fromObject a =
|
||||||
|
case tryFromObject a of
|
||||||
|
Left err ->
|
||||||
|
throw $ UnpackError err
|
||||||
|
Right ret ->
|
||||||
|
ret
|
||||||
|
|
||||||
|
-- | Decode a value from MessagePack object
|
||||||
|
tryFromObject :: Object -> Either String a
|
||||||
|
tryFromObject = tryUnpack . pack
|
||||||
|
|
||||||
|
instance OBJECT Object where
|
||||||
|
toObject = id
|
||||||
|
tryFromObject = Right
|
||||||
|
|
||||||
|
tryFromObjectError :: Either String a
|
||||||
|
tryFromObjectError = Left "tryFromObject: cannot cast"
|
||||||
|
|
||||||
|
instance OBJECT () where
|
||||||
|
toObject = const ObjectNil
|
||||||
|
tryFromObject ObjectNil = Right ()
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT Int where
|
||||||
|
toObject = ObjectInteger
|
||||||
|
tryFromObject (ObjectInteger n) = Right n
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT Bool where
|
||||||
|
toObject = ObjectBool
|
||||||
|
tryFromObject (ObjectBool b) = Right b
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT Double where
|
||||||
|
toObject = ObjectDouble
|
||||||
|
tryFromObject (ObjectDouble d) = Right d
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT Float where
|
||||||
|
toObject = ObjectFloat
|
||||||
|
tryFromObject (ObjectFloat f) = Right f
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT String where
|
||||||
|
toObject = toObject . encodeUtf8
|
||||||
|
tryFromObject obj = liftM decodeUtf8 $ tryFromObject obj
|
||||||
|
|
||||||
|
instance OBJECT B.ByteString where
|
||||||
|
toObject = ObjectRAW
|
||||||
|
tryFromObject (ObjectRAW bs) = Right bs
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT BL.ByteString where
|
||||||
|
toObject = ObjectRAW . fromLBS
|
||||||
|
tryFromObject (ObjectRAW bs) = Right $ toLBS bs
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT T.Text where
|
||||||
|
toObject = ObjectRAW . T.encodeUtf8
|
||||||
|
tryFromObject (ObjectRAW bs) = Right $ T.decodeUtf8With skipChar bs
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT TL.Text where
|
||||||
|
toObject = ObjectRAW . fromLBS . TL.encodeUtf8
|
||||||
|
tryFromObject (ObjectRAW bs) = Right $ TL.decodeUtf8With skipChar $ toLBS bs
|
||||||
|
tryFromObject _ = tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT a => OBJECT [a] where
|
||||||
|
toObject = ObjectArray . map toObject
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
mapM tryFromObject arr
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2) => OBJECT (a1, a2) where
|
||||||
|
toObject (a1, a2) = ObjectArray [toObject a1, toObject a2]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
return (v1, v2)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2, OBJECT a3) => OBJECT (a1, a2, a3) where
|
||||||
|
toObject (a1, a2, a3) = ObjectArray [toObject a1, toObject a2, toObject a3]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2, o3] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
v3 <- tryFromObject o3
|
||||||
|
return (v1, v2, v3)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4) => OBJECT (a1, a2, a3, a4) where
|
||||||
|
toObject (a1, a2, a3, a4) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2, o3, o4] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
v3 <- tryFromObject o3
|
||||||
|
v4 <- tryFromObject o4
|
||||||
|
return (v1, v2, v3, v4)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5) => OBJECT (a1, a2, a3, a4, a5) where
|
||||||
|
toObject (a1, a2, a3, a4, a5) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2, o3, o4, o5] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
v3 <- tryFromObject o3
|
||||||
|
v4 <- tryFromObject o4
|
||||||
|
v5 <- tryFromObject o5
|
||||||
|
return (v1, v2, v3, v4, v5)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6) => OBJECT (a1, a2, a3, a4, a5, a6) where
|
||||||
|
toObject (a1, a2, a3, a4, a5, a6) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2, o3, o4, o5, o6] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
v3 <- tryFromObject o3
|
||||||
|
v4 <- tryFromObject o4
|
||||||
|
v5 <- tryFromObject o5
|
||||||
|
v6 <- tryFromObject o6
|
||||||
|
return (v1, v2, v3, v4, v5, v6)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6, OBJECT a7) => OBJECT (a1, a2, a3, a4, a5, a6, a7) where
|
||||||
|
toObject (a1, a2, a3, a4, a5, a6, a7) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2, o3, o4, o5, o6, o7] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
v3 <- tryFromObject o3
|
||||||
|
v4 <- tryFromObject o4
|
||||||
|
v5 <- tryFromObject o5
|
||||||
|
v6 <- tryFromObject o6
|
||||||
|
v7 <- tryFromObject o7
|
||||||
|
return (v1, v2, v3, v4, v5, v6, v7)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6, OBJECT a7, OBJECT a8) => OBJECT (a1, a2, a3, a4, a5, a6, a7, a8) where
|
||||||
|
toObject (a1, a2, a3, a4, a5, a6, a7, a8) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7, toObject a8]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2, o3, o4, o5, o6, o7, o8] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
v3 <- tryFromObject o3
|
||||||
|
v4 <- tryFromObject o4
|
||||||
|
v5 <- tryFromObject o5
|
||||||
|
v6 <- tryFromObject o6
|
||||||
|
v7 <- tryFromObject o7
|
||||||
|
v8 <- tryFromObject o8
|
||||||
|
return (v1, v2, v3, v4, v5, v6, v7, v8)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a1, OBJECT a2, OBJECT a3, OBJECT a4, OBJECT a5, OBJECT a6, OBJECT a7, OBJECT a8, OBJECT a9) => OBJECT (a1, a2, a3, a4, a5, a6, a7, a8, a9) where
|
||||||
|
toObject (a1, a2, a3, a4, a5, a6, a7, a8, a9) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7, toObject a8, toObject a9]
|
||||||
|
tryFromObject (ObjectArray arr) =
|
||||||
|
case arr of
|
||||||
|
[o1, o2, o3, o4, o5, o6, o7, o8, o9] -> do
|
||||||
|
v1 <- tryFromObject o1
|
||||||
|
v2 <- tryFromObject o2
|
||||||
|
v3 <- tryFromObject o3
|
||||||
|
v4 <- tryFromObject o4
|
||||||
|
v5 <- tryFromObject o5
|
||||||
|
v6 <- tryFromObject o6
|
||||||
|
v7 <- tryFromObject o7
|
||||||
|
v8 <- tryFromObject o8
|
||||||
|
v9 <- tryFromObject o9
|
||||||
|
return (v1, v2, v3, v4, v5, v6, v7, v8, v9)
|
||||||
|
_ ->
|
||||||
|
tryFromObjectError
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance (OBJECT a, OBJECT b) => OBJECT (Assoc [(a,b)]) where
|
||||||
|
toObject =
|
||||||
|
ObjectMap . map (\(a, b) -> (toObject a, toObject b)) . unAssoc
|
||||||
|
tryFromObject (ObjectMap mem) = do
|
||||||
|
liftM Assoc $ mapM (\(a, b) -> liftM2 (,) (tryFromObject a) (tryFromObject b)) mem
|
||||||
|
tryFromObject _ =
|
||||||
|
tryFromObjectError
|
||||||
|
|
||||||
|
instance OBJECT a => OBJECT (Maybe a) where
|
||||||
|
toObject (Just a) = toObject a
|
||||||
|
toObject Nothing = ObjectNil
|
||||||
|
|
||||||
|
tryFromObject ObjectNil = return Nothing
|
||||||
|
tryFromObject obj = liftM Just $ tryFromObject obj
|
||||||
|
|
204
haskell/src/Data/MessagePack/Pack.hs
Normal file
204
haskell/src/Data/MessagePack/Pack.hs
Normal file
@@ -0,0 +1,204 @@
|
|||||||
|
{-# Language FlexibleInstances #-}
|
||||||
|
{-# Language IncoherentInstances #-}
|
||||||
|
{-# Language TypeSynonymInstances #-}
|
||||||
|
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : Data.MessagePack.Pack
|
||||||
|
-- Copyright : (c) Hideyuki Tanaka, 2009-2010
|
||||||
|
-- License : BSD3
|
||||||
|
--
|
||||||
|
-- Maintainer: tanaka.hideyuki@gmail.com
|
||||||
|
-- Stability : experimental
|
||||||
|
-- Portability: portable
|
||||||
|
--
|
||||||
|
-- MessagePack Serializer using @Data.Binary.Pack@
|
||||||
|
--
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
|
||||||
|
module Data.MessagePack.Pack (
|
||||||
|
-- * Serializable class
|
||||||
|
Packable(..),
|
||||||
|
-- * Simple function to pack a Haskell value
|
||||||
|
pack,
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Data.Binary.Put
|
||||||
|
import Data.Binary.IEEE754
|
||||||
|
import Data.Bits
|
||||||
|
import qualified Data.ByteString as B
|
||||||
|
import qualified Data.ByteString.Lazy as BL
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Text.Encoding as T
|
||||||
|
import qualified Data.Text.Lazy as TL
|
||||||
|
import qualified Data.Text.Lazy.Encoding as TL
|
||||||
|
import qualified Data.Vector as V
|
||||||
|
|
||||||
|
import Data.MessagePack.Assoc
|
||||||
|
import Data.MessagePack.Internal.Utf8
|
||||||
|
|
||||||
|
-- | Serializable class
|
||||||
|
class Packable a where
|
||||||
|
-- | Serialize a value
|
||||||
|
put :: a -> Put
|
||||||
|
|
||||||
|
-- | Pack Haskell data to MessagePack string.
|
||||||
|
pack :: Packable a => a -> BL.ByteString
|
||||||
|
pack = runPut . put
|
||||||
|
|
||||||
|
instance Packable Int where
|
||||||
|
put n =
|
||||||
|
case n of
|
||||||
|
_ | n >= 0 && n <= 127 ->
|
||||||
|
putWord8 $ fromIntegral n
|
||||||
|
_ | n >= -32 && n <= -1 ->
|
||||||
|
putWord8 $ fromIntegral n
|
||||||
|
_ | n >= 0 && n < 0x100 -> do
|
||||||
|
putWord8 0xCC
|
||||||
|
putWord8 $ fromIntegral n
|
||||||
|
_ | n >= 0 && n < 0x10000 -> do
|
||||||
|
putWord8 0xCD
|
||||||
|
putWord16be $ fromIntegral n
|
||||||
|
_ | n >= 0 && n < 0x100000000 -> do
|
||||||
|
putWord8 0xCE
|
||||||
|
putWord32be $ fromIntegral n
|
||||||
|
_ | n >= 0 -> do
|
||||||
|
putWord8 0xCF
|
||||||
|
putWord64be $ fromIntegral n
|
||||||
|
_ | n >= -0x80 -> do
|
||||||
|
putWord8 0xD0
|
||||||
|
putWord8 $ fromIntegral n
|
||||||
|
_ | n >= -0x8000 -> do
|
||||||
|
putWord8 0xD1
|
||||||
|
putWord16be $ fromIntegral n
|
||||||
|
_ | n >= -0x80000000 -> do
|
||||||
|
putWord8 0xD2
|
||||||
|
putWord32be $ fromIntegral n
|
||||||
|
_ -> do
|
||||||
|
putWord8 0xD3
|
||||||
|
putWord64be $ fromIntegral n
|
||||||
|
|
||||||
|
instance Packable () where
|
||||||
|
put _ =
|
||||||
|
putWord8 0xC0
|
||||||
|
|
||||||
|
instance Packable Bool where
|
||||||
|
put True = putWord8 0xC3
|
||||||
|
put False = putWord8 0xC2
|
||||||
|
|
||||||
|
instance Packable Float where
|
||||||
|
put f = do
|
||||||
|
putWord8 0xCA
|
||||||
|
putFloat32be f
|
||||||
|
|
||||||
|
instance Packable Double where
|
||||||
|
put d = do
|
||||||
|
putWord8 0xCB
|
||||||
|
putFloat64be d
|
||||||
|
|
||||||
|
instance Packable String where
|
||||||
|
put = putString encodeUtf8 B.length putByteString
|
||||||
|
|
||||||
|
instance Packable B.ByteString where
|
||||||
|
put = putString id B.length putByteString
|
||||||
|
|
||||||
|
instance Packable BL.ByteString where
|
||||||
|
put = putString id (fromIntegral . BL.length) putLazyByteString
|
||||||
|
|
||||||
|
instance Packable T.Text where
|
||||||
|
put = putString T.encodeUtf8 B.length putByteString
|
||||||
|
|
||||||
|
instance Packable TL.Text where
|
||||||
|
put = putString TL.encodeUtf8 (fromIntegral . BL.length) putLazyByteString
|
||||||
|
|
||||||
|
putString :: (s -> t) -> (t -> Int) -> (t -> Put) -> s -> Put
|
||||||
|
putString cnv lf pf str = do
|
||||||
|
let bs = cnv str
|
||||||
|
case lf bs of
|
||||||
|
len | len <= 31 -> do
|
||||||
|
putWord8 $ 0xA0 .|. fromIntegral len
|
||||||
|
len | len < 0x10000 -> do
|
||||||
|
putWord8 0xDA
|
||||||
|
putWord16be $ fromIntegral len
|
||||||
|
len -> do
|
||||||
|
putWord8 0xDB
|
||||||
|
putWord32be $ fromIntegral len
|
||||||
|
pf bs
|
||||||
|
|
||||||
|
instance Packable a => Packable [a] where
|
||||||
|
put = putArray length (mapM_ put)
|
||||||
|
|
||||||
|
instance Packable a => Packable (V.Vector a) where
|
||||||
|
put = putArray V.length (V.mapM_ put)
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2) => Packable (a1, a2) where
|
||||||
|
put = putArray (const 2) f where
|
||||||
|
f (a1, a2) = put a1 >> put a2
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2, Packable a3) => Packable (a1, a2, a3) where
|
||||||
|
put = putArray (const 3) f where
|
||||||
|
f (a1, a2, a3) = put a1 >> put a2 >> put a3
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2, Packable a3, Packable a4) => Packable (a1, a2, a3, a4) where
|
||||||
|
put = putArray (const 4) f where
|
||||||
|
f (a1, a2, a3, a4) = put a1 >> put a2 >> put a3 >> put a4
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5) => Packable (a1, a2, a3, a4, a5) where
|
||||||
|
put = putArray (const 5) f where
|
||||||
|
f (a1, a2, a3, a4, a5) = put a1 >> put a2 >> put a3 >> put a4 >> put a5
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6) => Packable (a1, a2, a3, a4, a5, a6) where
|
||||||
|
put = putArray (const 6) f where
|
||||||
|
f (a1, a2, a3, a4, a5, a6) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6, Packable a7) => Packable (a1, a2, a3, a4, a5, a6, a7) where
|
||||||
|
put = putArray (const 7) f where
|
||||||
|
f (a1, a2, a3, a4, a5, a6, a7) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6 >> put a7
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6, Packable a7, Packable a8) => Packable (a1, a2, a3, a4, a5, a6, a7, a8) where
|
||||||
|
put = putArray (const 8) f where
|
||||||
|
f (a1, a2, a3, a4, a5, a6, a7, a8) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6 >> put a7 >> put a8
|
||||||
|
|
||||||
|
instance (Packable a1, Packable a2, Packable a3, Packable a4, Packable a5, Packable a6, Packable a7, Packable a8, Packable a9) => Packable (a1, a2, a3, a4, a5, a6, a7, a8, a9) where
|
||||||
|
put = putArray (const 9) f where
|
||||||
|
f (a1, a2, a3, a4, a5, a6, a7, a8, a9) = put a1 >> put a2 >> put a3 >> put a4 >> put a5 >> put a6 >> put a7 >> put a8 >> put a9
|
||||||
|
|
||||||
|
putArray :: (a -> Int) -> (a -> Put) -> a -> Put
|
||||||
|
putArray lf pf arr = do
|
||||||
|
case lf arr of
|
||||||
|
len | len <= 15 ->
|
||||||
|
putWord8 $ 0x90 .|. fromIntegral len
|
||||||
|
len | len < 0x10000 -> do
|
||||||
|
putWord8 0xDC
|
||||||
|
putWord16be $ fromIntegral len
|
||||||
|
len -> do
|
||||||
|
putWord8 0xDD
|
||||||
|
putWord32be $ fromIntegral len
|
||||||
|
pf arr
|
||||||
|
|
||||||
|
instance (Packable k, Packable v) => Packable (Assoc [(k,v)]) where
|
||||||
|
put = putMap length (mapM_ putPair) . unAssoc
|
||||||
|
|
||||||
|
instance (Packable k, Packable v) => Packable (Assoc (V.Vector (k,v))) where
|
||||||
|
put = putMap V.length (V.mapM_ putPair) . unAssoc
|
||||||
|
|
||||||
|
putPair :: (Packable a, Packable b) => (a, b) -> Put
|
||||||
|
putPair (a, b) = put a >> put b
|
||||||
|
|
||||||
|
putMap :: (a -> Int) -> (a -> Put) -> a -> Put
|
||||||
|
putMap lf pf m = do
|
||||||
|
case lf m of
|
||||||
|
len | len <= 15 ->
|
||||||
|
putWord8 $ 0x80 .|. fromIntegral len
|
||||||
|
len | len < 0x10000 -> do
|
||||||
|
putWord8 0xDE
|
||||||
|
putWord16be $ fromIntegral len
|
||||||
|
len -> do
|
||||||
|
putWord8 0xDF
|
||||||
|
putWord32be $ fromIntegral len
|
||||||
|
pf m
|
||||||
|
|
||||||
|
instance Packable a => Packable (Maybe a) where
|
||||||
|
put Nothing = put ()
|
||||||
|
put (Just a) = put a
|
||||||
|
|
@@ -1,84 +0,0 @@
|
|||||||
--------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : Data.MessagePack.Stream
|
|
||||||
-- Copyright : (c) Hideyuki Tanaka, 2009
|
|
||||||
-- License : BSD3
|
|
||||||
--
|
|
||||||
-- Maintainer: tanaka.hideyuki@gmail.com
|
|
||||||
-- Stability : experimental
|
|
||||||
-- Portability: portable
|
|
||||||
--
|
|
||||||
-- Lazy Stream Serializers and Deserializers
|
|
||||||
--
|
|
||||||
--------------------------------------------------------------------
|
|
||||||
|
|
||||||
module Data.MessagePack.Stream(
|
|
||||||
unpackObjects,
|
|
||||||
unpackObjectsFromFile,
|
|
||||||
unpackObjectsFromHandle,
|
|
||||||
unpackObjectsFromString,
|
|
||||||
) where
|
|
||||||
|
|
||||||
import Control.Monad
|
|
||||||
import Data.ByteString (ByteString)
|
|
||||||
import qualified Data.ByteString as BS
|
|
||||||
import System.IO
|
|
||||||
import System.IO.Unsafe
|
|
||||||
|
|
||||||
import Data.MessagePack.Base
|
|
||||||
import Data.MessagePack.Feed
|
|
||||||
|
|
||||||
-- | Unpack objects using given feeder.
|
|
||||||
unpackObjects :: Feeder -> IO [Object]
|
|
||||||
unpackObjects feeder = do
|
|
||||||
up <- newUnpacker defaultInitialBufferSize
|
|
||||||
f up
|
|
||||||
where
|
|
||||||
f up = unsafeInterleaveIO $ do
|
|
||||||
mbo <- unpackOnce up
|
|
||||||
case mbo of
|
|
||||||
Just o -> do
|
|
||||||
os <- f up
|
|
||||||
return $ o:os
|
|
||||||
Nothing ->
|
|
||||||
return []
|
|
||||||
|
|
||||||
unpackOnce up = do
|
|
||||||
resp <- unpackerExecute up
|
|
||||||
case resp of
|
|
||||||
0 -> do
|
|
||||||
r <- feedOnce up
|
|
||||||
if r
|
|
||||||
then unpackOnce up
|
|
||||||
else return Nothing
|
|
||||||
1 -> do
|
|
||||||
obj <- unpackerData up
|
|
||||||
freeZone =<< unpackerReleaseZone up
|
|
||||||
unpackerReset up
|
|
||||||
return $ Just obj
|
|
||||||
_ ->
|
|
||||||
error $ "unpackerExecute fails: " ++ show resp
|
|
||||||
|
|
||||||
feedOnce up = do
|
|
||||||
dat <- feeder
|
|
||||||
case dat of
|
|
||||||
Nothing ->
|
|
||||||
return False
|
|
||||||
Just bs -> do
|
|
||||||
unpackerFeed up bs
|
|
||||||
return True
|
|
||||||
|
|
||||||
-- | Unpack objects from file.
|
|
||||||
unpackObjectsFromFile :: FilePath -> IO [Object]
|
|
||||||
unpackObjectsFromFile fname =
|
|
||||||
unpackObjects =<< feederFromFile fname
|
|
||||||
|
|
||||||
-- | Unpack objects from handle.
|
|
||||||
unpackObjectsFromHandle :: Handle -> IO [Object]
|
|
||||||
unpackObjectsFromHandle h =
|
|
||||||
unpackObjects =<< feederFromHandle h
|
|
||||||
|
|
||||||
-- | Unpack oobjects from given byte sequence.
|
|
||||||
unpackObjectsFromString :: ByteString -> IO [Object]
|
|
||||||
unpackObjectsFromString bs =
|
|
||||||
unpackObjects =<< feederFromString bs
|
|
323
haskell/src/Data/MessagePack/Unpack.hs
Normal file
323
haskell/src/Data/MessagePack/Unpack.hs
Normal file
@@ -0,0 +1,323 @@
|
|||||||
|
{-# Language FlexibleInstances #-}
|
||||||
|
{-# Language IncoherentInstances #-}
|
||||||
|
{-# Language TypeSynonymInstances #-}
|
||||||
|
{-# Language DeriveDataTypeable #-}
|
||||||
|
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
-- |
|
||||||
|
-- Module : Data.MessagePack.Unpack
|
||||||
|
-- Copyright : (c) Hideyuki Tanaka, 2009-2010
|
||||||
|
-- License : BSD3
|
||||||
|
--
|
||||||
|
-- Maintainer: tanaka.hideyuki@gmail.com
|
||||||
|
-- Stability : experimental
|
||||||
|
-- Portability: portable
|
||||||
|
--
|
||||||
|
-- MessagePack Deserializer using @Data.Attoparsec@
|
||||||
|
--
|
||||||
|
--------------------------------------------------------------------
|
||||||
|
|
||||||
|
module Data.MessagePack.Unpack(
|
||||||
|
-- * MessagePack deserializer
|
||||||
|
Unpackable(..),
|
||||||
|
-- * Simple function to unpack a Haskell value
|
||||||
|
unpack,
|
||||||
|
tryUnpack,
|
||||||
|
-- * Unpack exception
|
||||||
|
UnpackError(..),
|
||||||
|
-- * ByteString utils
|
||||||
|
IsByteString(..),
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Control.Exception
|
||||||
|
import Control.Monad
|
||||||
|
import qualified Data.Attoparsec as A
|
||||||
|
import Data.Binary.Get
|
||||||
|
import Data.Binary.IEEE754
|
||||||
|
import Data.Bits
|
||||||
|
import qualified Data.ByteString as B
|
||||||
|
import qualified Data.ByteString.Lazy as BL
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Text.Encoding as T
|
||||||
|
import qualified Data.Text.Lazy as TL
|
||||||
|
import qualified Data.Text.Lazy.Encoding as TL
|
||||||
|
import Data.Int
|
||||||
|
import Data.Typeable
|
||||||
|
import qualified Data.Vector as V
|
||||||
|
import Data.Word
|
||||||
|
import Text.Printf
|
||||||
|
|
||||||
|
import Data.MessagePack.Assoc
|
||||||
|
import Data.MessagePack.Internal.Utf8
|
||||||
|
|
||||||
|
-- | Deserializable class
|
||||||
|
class Unpackable a where
|
||||||
|
-- | Deserialize a value
|
||||||
|
get :: A.Parser a
|
||||||
|
|
||||||
|
class IsByteString s where
|
||||||
|
toBS :: s -> B.ByteString
|
||||||
|
|
||||||
|
instance IsByteString B.ByteString where
|
||||||
|
toBS = id
|
||||||
|
|
||||||
|
instance IsByteString BL.ByteString where
|
||||||
|
toBS = B.concat . BL.toChunks
|
||||||
|
|
||||||
|
-- | The exception of unpack
|
||||||
|
data UnpackError =
|
||||||
|
UnpackError String
|
||||||
|
deriving (Show, Typeable)
|
||||||
|
|
||||||
|
instance Exception UnpackError
|
||||||
|
|
||||||
|
-- | Unpack MessagePack string to Haskell data.
|
||||||
|
unpack :: (Unpackable a, IsByteString s) => s -> a
|
||||||
|
unpack bs =
|
||||||
|
case tryUnpack bs of
|
||||||
|
Left err ->
|
||||||
|
throw $ UnpackError err
|
||||||
|
Right ret ->
|
||||||
|
ret
|
||||||
|
|
||||||
|
-- | Unpack MessagePack string to Haskell data.
|
||||||
|
tryUnpack :: (Unpackable a, IsByteString s) => s -> Either String a
|
||||||
|
tryUnpack bs =
|
||||||
|
case A.parse get (toBS bs) of
|
||||||
|
A.Fail _ _ err ->
|
||||||
|
Left err
|
||||||
|
A.Partial _ ->
|
||||||
|
Left "not enough input"
|
||||||
|
A.Done _ ret ->
|
||||||
|
Right ret
|
||||||
|
|
||||||
|
instance Unpackable Int where
|
||||||
|
get = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
_ | c .&. 0x80 == 0x00 ->
|
||||||
|
return $ fromIntegral c
|
||||||
|
_ | c .&. 0xE0 == 0xE0 ->
|
||||||
|
return $ fromIntegral (fromIntegral c :: Int8)
|
||||||
|
0xCC ->
|
||||||
|
return . fromIntegral =<< A.anyWord8
|
||||||
|
0xCD ->
|
||||||
|
return . fromIntegral =<< parseUint16
|
||||||
|
0xCE ->
|
||||||
|
return . fromIntegral =<< parseUint32
|
||||||
|
0xCF ->
|
||||||
|
return . fromIntegral =<< parseUint64
|
||||||
|
0xD0 ->
|
||||||
|
return . fromIntegral =<< parseInt8
|
||||||
|
0xD1 ->
|
||||||
|
return . fromIntegral =<< parseInt16
|
||||||
|
0xD2 ->
|
||||||
|
return . fromIntegral =<< parseInt32
|
||||||
|
0xD3 ->
|
||||||
|
return . fromIntegral =<< parseInt64
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid integer tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance Unpackable () where
|
||||||
|
get = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
0xC0 ->
|
||||||
|
return ()
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid nil tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance Unpackable Bool where
|
||||||
|
get = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
0xC3 ->
|
||||||
|
return True
|
||||||
|
0xC2 ->
|
||||||
|
return False
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid bool tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance Unpackable Float where
|
||||||
|
get = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
0xCA ->
|
||||||
|
return . runGet getFloat32be . toLBS =<< A.take 4
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid float tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance Unpackable Double where
|
||||||
|
get = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
0xCB ->
|
||||||
|
return . runGet getFloat64be . toLBS =<< A.take 8
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid double tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance Unpackable String where
|
||||||
|
get = parseString (\n -> return . decodeUtf8 =<< A.take n)
|
||||||
|
|
||||||
|
instance Unpackable B.ByteString where
|
||||||
|
get = parseString A.take
|
||||||
|
|
||||||
|
instance Unpackable BL.ByteString where
|
||||||
|
get = parseString (\n -> return . toLBS =<< A.take n)
|
||||||
|
|
||||||
|
instance Unpackable T.Text where
|
||||||
|
get = parseString (\n -> return . T.decodeUtf8With skipChar =<< A.take n)
|
||||||
|
|
||||||
|
instance Unpackable TL.Text where
|
||||||
|
get = parseString (\n -> return . TL.decodeUtf8With skipChar . toLBS =<< A.take n)
|
||||||
|
|
||||||
|
parseString :: (Int -> A.Parser a) -> A.Parser a
|
||||||
|
parseString aget = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
_ | c .&. 0xE0 == 0xA0 ->
|
||||||
|
aget . fromIntegral $ c .&. 0x1F
|
||||||
|
0xDA ->
|
||||||
|
aget . fromIntegral =<< parseUint16
|
||||||
|
0xDB ->
|
||||||
|
aget . fromIntegral =<< parseUint32
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid raw tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance Unpackable a => Unpackable [a] where
|
||||||
|
get = parseArray (flip replicateM get)
|
||||||
|
|
||||||
|
instance Unpackable a => Unpackable (V.Vector a) where
|
||||||
|
get = parseArray (flip V.replicateM get)
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2) => Unpackable (a1, a2) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 2 = get >>= \a1 -> get >>= \a2 -> return (a1, a2)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 2 but got " n
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2, Unpackable a3) => Unpackable (a1, a2, a3) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 3 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> return (a1, a2, a3)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 3 but got " n
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4) => Unpackable (a1, a2, a3, a4) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 4 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> return (a1, a2, a3, a4)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 4 but got " n
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5) => Unpackable (a1, a2, a3, a4, a5) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 5 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> return (a1, a2, a3, a4, a5)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 5 but got " n
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6) => Unpackable (a1, a2, a3, a4, a5, a6) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 6 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> return (a1, a2, a3, a4, a5, a6)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 6 but got " n
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6, Unpackable a7) => Unpackable (a1, a2, a3, a4, a5, a6, a7) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 7 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> get >>= \a7 -> return (a1, a2, a3, a4, a5, a6, a7)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 7 but got " n
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6, Unpackable a7, Unpackable a8) => Unpackable (a1, a2, a3, a4, a5, a6, a7, a8) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 8 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> get >>= \a7 -> get >>= \a8 -> return (a1, a2, a3, a4, a5, a6, a7, a8)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 8 but got " n
|
||||||
|
|
||||||
|
instance (Unpackable a1, Unpackable a2, Unpackable a3, Unpackable a4, Unpackable a5, Unpackable a6, Unpackable a7, Unpackable a8, Unpackable a9) => Unpackable (a1, a2, a3, a4, a5, a6, a7, a8, a9) where
|
||||||
|
get = parseArray f where
|
||||||
|
f 9 = get >>= \a1 -> get >>= \a2 -> get >>= \a3 -> get >>= \a4 -> get >>= \a5 -> get >>= \a6 -> get >>= \a7 -> get >>= \a8 -> get >>= \a9 -> return (a1, a2, a3, a4, a5, a6, a7, a8, a9)
|
||||||
|
f n = fail $ printf "wrong tupple size: expected 9 but got " n
|
||||||
|
|
||||||
|
parseArray :: (Int -> A.Parser a) -> A.Parser a
|
||||||
|
parseArray aget = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
_ | c .&. 0xF0 == 0x90 ->
|
||||||
|
aget . fromIntegral $ c .&. 0x0F
|
||||||
|
0xDC ->
|
||||||
|
aget . fromIntegral =<< parseUint16
|
||||||
|
0xDD ->
|
||||||
|
aget . fromIntegral =<< parseUint32
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid array tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance (Unpackable k, Unpackable v) => Unpackable (Assoc [(k,v)]) where
|
||||||
|
get = liftM Assoc $ parseMap (flip replicateM parsePair)
|
||||||
|
|
||||||
|
instance (Unpackable k, Unpackable v) => Unpackable (Assoc (V.Vector (k, v))) where
|
||||||
|
get = liftM Assoc $ parseMap (flip V.replicateM parsePair)
|
||||||
|
|
||||||
|
parsePair :: (Unpackable k, Unpackable v) => A.Parser (k, v)
|
||||||
|
parsePair = do
|
||||||
|
a <- get
|
||||||
|
b <- get
|
||||||
|
return (a, b)
|
||||||
|
|
||||||
|
parseMap :: (Int -> A.Parser a) -> A.Parser a
|
||||||
|
parseMap aget = do
|
||||||
|
c <- A.anyWord8
|
||||||
|
case c of
|
||||||
|
_ | c .&. 0xF0 == 0x80 ->
|
||||||
|
aget . fromIntegral $ c .&. 0x0F
|
||||||
|
0xDE ->
|
||||||
|
aget . fromIntegral =<< parseUint16
|
||||||
|
0xDF ->
|
||||||
|
aget . fromIntegral =<< parseUint32
|
||||||
|
_ ->
|
||||||
|
fail $ printf "invlid map tag: 0x%02X" c
|
||||||
|
|
||||||
|
instance Unpackable a => Unpackable (Maybe a) where
|
||||||
|
get =
|
||||||
|
A.choice
|
||||||
|
[ liftM Just get
|
||||||
|
, liftM (\() -> Nothing) get ]
|
||||||
|
|
||||||
|
parseUint16 :: A.Parser Word16
|
||||||
|
parseUint16 = do
|
||||||
|
b0 <- A.anyWord8
|
||||||
|
b1 <- A.anyWord8
|
||||||
|
return $ (fromIntegral b0 `shiftL` 8) .|. fromIntegral b1
|
||||||
|
|
||||||
|
parseUint32 :: A.Parser Word32
|
||||||
|
parseUint32 = do
|
||||||
|
b0 <- A.anyWord8
|
||||||
|
b1 <- A.anyWord8
|
||||||
|
b2 <- A.anyWord8
|
||||||
|
b3 <- A.anyWord8
|
||||||
|
return $ (fromIntegral b0 `shiftL` 24) .|.
|
||||||
|
(fromIntegral b1 `shiftL` 16) .|.
|
||||||
|
(fromIntegral b2 `shiftL` 8) .|.
|
||||||
|
fromIntegral b3
|
||||||
|
|
||||||
|
parseUint64 :: A.Parser Word64
|
||||||
|
parseUint64 = do
|
||||||
|
b0 <- A.anyWord8
|
||||||
|
b1 <- A.anyWord8
|
||||||
|
b2 <- A.anyWord8
|
||||||
|
b3 <- A.anyWord8
|
||||||
|
b4 <- A.anyWord8
|
||||||
|
b5 <- A.anyWord8
|
||||||
|
b6 <- A.anyWord8
|
||||||
|
b7 <- A.anyWord8
|
||||||
|
return $ (fromIntegral b0 `shiftL` 56) .|.
|
||||||
|
(fromIntegral b1 `shiftL` 48) .|.
|
||||||
|
(fromIntegral b2 `shiftL` 40) .|.
|
||||||
|
(fromIntegral b3 `shiftL` 32) .|.
|
||||||
|
(fromIntegral b4 `shiftL` 24) .|.
|
||||||
|
(fromIntegral b5 `shiftL` 16) .|.
|
||||||
|
(fromIntegral b6 `shiftL` 8) .|.
|
||||||
|
fromIntegral b7
|
||||||
|
|
||||||
|
parseInt8 :: A.Parser Int8
|
||||||
|
parseInt8 = return . fromIntegral =<< A.anyWord8
|
||||||
|
|
||||||
|
parseInt16 :: A.Parser Int16
|
||||||
|
parseInt16 = return . fromIntegral =<< parseUint16
|
||||||
|
|
||||||
|
parseInt32 :: A.Parser Int32
|
||||||
|
parseInt32 = return . fromIntegral =<< parseUint32
|
||||||
|
|
||||||
|
parseInt64 :: A.Parser Int64
|
||||||
|
parseInt64 = return . fromIntegral =<< parseUint64
|
@@ -1,16 +1,21 @@
|
|||||||
import Control.Monad.Trans
|
{-# Language OverloadedStrings #-}
|
||||||
|
|
||||||
|
import Control.Monad.IO.Class
|
||||||
|
import qualified Data.ByteString as B
|
||||||
import Data.MessagePack
|
import Data.MessagePack
|
||||||
|
|
||||||
main = do
|
main = do
|
||||||
sb <- packToString $ do
|
sb <- return $ packToString $ do
|
||||||
put [1,2,3::Int]
|
put [1,2,3::Int]
|
||||||
put (3.14 :: Double)
|
put (3.14 :: Double)
|
||||||
put "Hoge"
|
put ("Hoge" :: B.ByteString)
|
||||||
|
|
||||||
print sb
|
print sb
|
||||||
|
|
||||||
unpackFromString sb $ do
|
r <- unpackFromString sb $ do
|
||||||
arr <- get
|
arr <- get
|
||||||
dbl <- get
|
dbl <- get
|
||||||
str <- get
|
str <- get
|
||||||
liftIO $ print (arr :: [Int], dbl :: Double, str :: String)
|
return (arr :: [Int], dbl :: Double, str :: B.ByteString)
|
||||||
|
|
||||||
|
print r
|
||||||
|
@@ -1,14 +0,0 @@
|
|||||||
import Control.Applicative
|
|
||||||
import qualified Data.ByteString as BS
|
|
||||||
import Data.MessagePack
|
|
||||||
|
|
||||||
main = do
|
|
||||||
sb <- newSimpleBuffer
|
|
||||||
pc <- newPacker sb
|
|
||||||
pack pc [1,2,3::Int]
|
|
||||||
pack pc True
|
|
||||||
pack pc "hoge"
|
|
||||||
bs <- simpleBufferData sb
|
|
||||||
|
|
||||||
os <- unpackObjectsFromString bs
|
|
||||||
mapM_ print os
|
|
@@ -1,36 +1,70 @@
|
|||||||
|
import Test.Framework
|
||||||
|
import Test.Framework.Providers.QuickCheck2
|
||||||
|
import Test.QuickCheck
|
||||||
|
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
|
import qualified Data.ByteString.Char8 as B
|
||||||
|
import qualified Data.ByteString.Lazy.Char8 as L
|
||||||
import Data.MessagePack
|
import Data.MessagePack
|
||||||
|
|
||||||
{-
|
instance Arbitrary a => Arbitrary (Assoc a) where
|
||||||
main = do
|
arbitrary = liftM Assoc arbitrary
|
||||||
sb <- newSimpleBuffer
|
|
||||||
pc <- newPacker sb
|
|
||||||
|
|
||||||
pack pc [(1,2),(2,3),(3::Int,4::Int)]
|
|
||||||
pack pc [4,5,6::Int]
|
|
||||||
pack pc "hoge"
|
|
||||||
|
|
||||||
bs <- simpleBufferData sb
|
|
||||||
print bs
|
|
||||||
|
|
||||||
up <- newUnpacker defaultInitialBufferSize
|
|
||||||
|
|
||||||
unpackerFeed up bs
|
|
||||||
|
|
||||||
let f = do
|
mid :: (Packable a, Unpackable a) => a -> a
|
||||||
res <- unpackerExecute up
|
mid = unpack . pack
|
||||||
when (res==1) $ do
|
|
||||||
obj <- unpackerData up
|
|
||||||
print obj
|
|
||||||
f
|
|
||||||
|
|
||||||
f
|
|
||||||
|
|
||||||
return ()
|
prop_mid_int a = a == mid a
|
||||||
-}
|
where types = a :: Int
|
||||||
|
prop_mid_nil a = a == mid a
|
||||||
|
where types = a :: ()
|
||||||
|
prop_mid_bool a = a == mid a
|
||||||
|
where types = a :: Bool
|
||||||
|
prop_mid_double a = a == mid a
|
||||||
|
where types = a :: Double
|
||||||
|
prop_mid_string a = a == mid a
|
||||||
|
where types = a :: String
|
||||||
|
prop_mid_bytestring a = B.pack a == mid (B.pack a)
|
||||||
|
where types = a :: String
|
||||||
|
prop_mid_lazy_bytestring a = (L.pack a) == mid (L.pack a)
|
||||||
|
where types = a :: String
|
||||||
|
prop_mid_array_int a = a == mid a
|
||||||
|
where types = a :: [Int]
|
||||||
|
prop_mid_array_string a = a == mid a
|
||||||
|
where types = a :: [String]
|
||||||
|
prop_mid_pair2 a = a == mid a
|
||||||
|
where types = a :: (Int, Int)
|
||||||
|
prop_mid_pair3 a = a == mid a
|
||||||
|
where types = a :: (Int, Int, Int)
|
||||||
|
prop_mid_pair4 a = a == mid a
|
||||||
|
where types = a :: (Int, Int, Int, Int)
|
||||||
|
prop_mid_pair5 a = a == mid a
|
||||||
|
where types = a :: (Int, Int, Int, Int, Int)
|
||||||
|
prop_mid_list_int_double a = a == mid a
|
||||||
|
where types = a :: [(Int, Double)]
|
||||||
|
prop_mid_list_string_string a = a == mid a
|
||||||
|
where types = a :: [(String, String)]
|
||||||
|
prop_mid_map_string_int a = a == mid a
|
||||||
|
where types = a :: Assoc [(String,Int)]
|
||||||
|
|
||||||
main = do
|
tests =
|
||||||
bs <- packb [(1,2),(2,3),(3::Int,4::Int)]
|
[ testGroup "simple"
|
||||||
print bs
|
[ testProperty "int" prop_mid_int
|
||||||
dat <- unpackb bs
|
, testProperty "nil" prop_mid_nil
|
||||||
print (dat :: Result [(Int, Int)])
|
, testProperty "bool" prop_mid_bool
|
||||||
|
, testProperty "double" prop_mid_double
|
||||||
|
, testProperty "string" prop_mid_string
|
||||||
|
, testProperty "bytestring" prop_mid_bytestring
|
||||||
|
, testProperty "lazy-bytestring" prop_mid_lazy_bytestring
|
||||||
|
, testProperty "[int]" prop_mid_array_int
|
||||||
|
, testProperty "[string]" prop_mid_array_string
|
||||||
|
, testProperty "(int, int)" prop_mid_pair2
|
||||||
|
, testProperty "(int, int, int)" prop_mid_pair3
|
||||||
|
, testProperty "(int, int, int, int)" prop_mid_pair4
|
||||||
|
, testProperty "(int, int, int, int, int)" prop_mid_pair5
|
||||||
|
, testProperty "[(int, double)]" prop_mid_list_int_double
|
||||||
|
, testProperty "[(string, string)]" prop_mid_list_string_string
|
||||||
|
, testProperty "Assoc [(string, int)]" prop_mid_map_string_int
|
||||||
|
]
|
||||||
|
]
|
||||||
|
|
||||||
|
main = defaultMain tests
|
||||||
|
43
haskell/test/UserData.hs
Normal file
43
haskell/test/UserData.hs
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
{-# Language TemplateHaskell #-}
|
||||||
|
|
||||||
|
import Data.MessagePack
|
||||||
|
import Data.MessagePack.Derive
|
||||||
|
|
||||||
|
data T
|
||||||
|
= A Int String
|
||||||
|
| B Double
|
||||||
|
deriving (Show, Eq)
|
||||||
|
|
||||||
|
$(deriveObject ''T)
|
||||||
|
|
||||||
|
data U
|
||||||
|
= C { c1 :: Int, c2 :: String }
|
||||||
|
| D { d1 :: Double }
|
||||||
|
deriving (Show, Eq)
|
||||||
|
|
||||||
|
$(deriveObject ''U)
|
||||||
|
|
||||||
|
data V
|
||||||
|
= E String | F
|
||||||
|
deriving (Show, Eq)
|
||||||
|
|
||||||
|
$(deriveObject ''V)
|
||||||
|
|
||||||
|
test :: (OBJECT a, Show a, Eq a) => a -> IO ()
|
||||||
|
test v = do
|
||||||
|
let bs = pack v
|
||||||
|
print bs
|
||||||
|
print (unpack bs == v)
|
||||||
|
|
||||||
|
let oa = toObject v
|
||||||
|
print oa
|
||||||
|
print (fromObject oa == v)
|
||||||
|
|
||||||
|
main = do
|
||||||
|
test $ A 123 "hoge"
|
||||||
|
test $ B 3.14
|
||||||
|
test $ C 123 "hoge"
|
||||||
|
test $ D 3.14
|
||||||
|
test $ E "hello"
|
||||||
|
test $ F
|
||||||
|
return ()
|
@@ -1,5 +1,13 @@
|
|||||||
#Mon Apr 19 22:18:48 JST 2010
|
#Tue Apr 12 19:14:17 JST 2011
|
||||||
org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6
|
eclipse.preferences.version=1
|
||||||
eclipse.preferences.version=1
|
org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled
|
||||||
org.eclipse.jdt.core.compiler.source=1.6
|
org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6
|
||||||
org.eclipse.jdt.core.compiler.compliance=1.6
|
org.eclipse.jdt.core.compiler.codegen.unusedLocal=preserve
|
||||||
|
org.eclipse.jdt.core.compiler.compliance=1.6
|
||||||
|
org.eclipse.jdt.core.compiler.debug.lineNumber=generate
|
||||||
|
org.eclipse.jdt.core.compiler.debug.localVariable=generate
|
||||||
|
org.eclipse.jdt.core.compiler.debug.sourceFile=generate
|
||||||
|
org.eclipse.jdt.core.compiler.problem.assertIdentifier=error
|
||||||
|
org.eclipse.jdt.core.compiler.problem.enumIdentifier=error
|
||||||
|
org.eclipse.jdt.core.compiler.problem.forbiddenReference=warning
|
||||||
|
org.eclipse.jdt.core.compiler.source=1.6
|
||||||
|
9
java/.settings/org.maven.ide.eclipse.prefs
Normal file
9
java/.settings/org.maven.ide.eclipse.prefs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
#Mon Sep 27 07:43:27 JST 2010
|
||||||
|
activeProfiles=
|
||||||
|
eclipse.preferences.version=1
|
||||||
|
fullBuildGoals=process-test-resources
|
||||||
|
includeModules=false
|
||||||
|
resolveWorkspaceProjects=true
|
||||||
|
resourceFilterGoals=process-resources resources\:testResources
|
||||||
|
skipCompilerPlugin=true
|
||||||
|
version=1
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user