1 module msgpack.packer;
2 
3 import msgpack.common;
4 import msgpack.attribute;
5 import msgpack.exception;
6 
7 import std.array;
8 import std.exception;
9 import std.range;
10 import std.stdio;
11 import std.traits;
12 import std.typecons;
13 import std.typetuple;
14 import std.container;
15 
16 
17 /**
18  * $(D Packer) is a $(D MessagePack) serializer
19  *
20  * Example:
21  * -----
22  * auto packer = packer(Appender!(ubyte[])());
23  *
24  * packer.packArray(false, 100, 1e-10, null);
25  *
26  * stdout.rawWrite(packer.stream.data);
27  * -----
28  *
29  * NOTE:
30  *  Current implementation can't deal with a circular reference.
31  *  If you try to serialize a object that has circular reference, runtime raises 'Stack Overflow'.
32  */
33 struct PackerImpl(Stream) if (isOutputRange!(Stream, ubyte) && isOutputRange!(Stream, ubyte[]))
34 {
35   private:
36     static @system
37     {
38         alias void delegate(ref PackerImpl, void*) PackHandler;
39         PackHandler[TypeInfo] packHandlers;
40 
41         public void registerHandler(T, alias Handler)()
42         {
43             packHandlers[typeid(T)] = delegate(ref PackerImpl packer, void* obj) {
44                 Handler(packer, *cast(T*)obj);
45             };
46         }
47 
48         public void register(T)()
49         {
50             packHandlers[typeid(T)] = delegate(ref PackerImpl packer, void* obj) {
51                 packer.packObject(*cast(T*)obj);
52             };
53         }
54     }
55 
56     enum size_t Offset = 1;  // type-information offset
57 
58     Stream                   stream_;  // the stream to write
59     ubyte[Offset + RealSize] store_;   // stores serialized value
60     bool                     withFieldName_;
61 
62 
63   public:
64     /**
65      * Constructs a packer with $(D_PARAM stream).
66      *
67      * Params:
68      *  stream        = the stream to write.
69      *  withFieldName = serialize class / struct with field name
70      */
71     this(Stream stream, bool withFieldName = false)
72     {
73         stream_        = stream;
74         withFieldName_ = withFieldName;
75     }
76 
77 
78     /**
79      * Constructs a packer with $(D_PARAM withFieldName).
80      *
81      * Params:
82      *  withFieldName = serialize class / struct with field name
83      */
84     this(bool withFieldName)
85     {
86         withFieldName_ = withFieldName;
87     }
88 
89 
90     /**
91      * Forwards to stream.
92      *
93      * Returns:
94      *  the stream.
95      */
96     @property @safe
97     nothrow ref Stream stream()
98     {
99         return stream_;
100     }
101 
102 
103     /**
104      * Serializes argument and writes to stream.
105      *
106      * If the argument is the pointer type, dereferences the pointer and serializes pointed value.
107      * -----
108      * int  a = 10;
109      * int* b = &b;
110      *
111      * packer.pack(b);  // serializes 10, not address of a
112      * -----
113      * Serializes nil if the argument of nullable type is null.
114      *
115      * NOTE:
116      *  MessagePack doesn't define $(D_KEYWORD real) type format.
117      *  Don't serialize $(D_KEYWORD real) if you communicate with other languages.
118      *  Transfer $(D_KEYWORD double) serialization if $(D_KEYWORD real) on your environment equals $(D_KEYWORD double).
119      *
120      * Params:
121      *  value = the content to serialize.
122      *
123      * Returns:
124      *  self, i.e. for method chaining.
125      */
126     ref PackerImpl pack(T)(in T value) if (is(Unqual!T == bool))
127     {
128         if (value)
129             stream_.put(Format.TRUE);
130         else
131             stream_.put(Format.FALSE);
132 
133         return this;
134     }
135 
136 
137     /// ditto
138     ref PackerImpl pack(T)(in T value) if (isUnsigned!T && !is(Unqual!T == enum))
139     {
140         // ulong < ulong is slower than uint < uint
141         static if (!is(Unqual!T  == ulong)) {
142             enum Bits = T.sizeof * 8;
143 
144             if (value < (1 << 8)) {
145                 if (value < (1 << 7)) {
146                     // fixnum
147                     stream_.put(take8from!Bits(value));
148                 } else {
149                     // uint 8
150                     store_[0] = Format.UINT8;
151                     store_[1] = take8from!Bits(value);
152                     stream_.put(store_[0..Offset + ubyte.sizeof]);
153                 }
154             } else {
155                 if (value < (1 << 16)) {
156                     // uint 16
157                     const temp = convertEndianTo!16(value);
158 
159                     store_[0] = Format.UINT16;
160                     *cast(ushort*)&store_[Offset] = temp;
161                     stream_.put(store_[0..Offset + ushort.sizeof]);
162                 } else {
163                     // uint 32
164                     const temp = convertEndianTo!32(value);
165 
166                     store_[0] = Format.UINT32;
167                     *cast(uint*)&store_[Offset] = temp;
168                     stream_.put(store_[0..Offset + uint.sizeof]);
169                 }
170             }
171         } else {
172             if (value < (1UL << 8)) {
173                 if (value < (1UL << 7)) {
174                     // fixnum
175                     stream_.put(take8from!64(value));
176                 } else {
177                     // uint 8
178                     store_[0] = Format.UINT8;
179                     store_[1] = take8from!64(value);
180                     stream_.put(store_[0..Offset + ubyte.sizeof]);
181                 }
182             } else {
183                 if (value < (1UL << 16)) {
184                     // uint 16
185                     const temp = convertEndianTo!16(value);
186 
187                     store_[0] = Format.UINT16;
188                     *cast(ushort*)&store_[Offset] = temp;
189                     stream_.put(store_[0..Offset + ushort.sizeof]);
190                 } else if (value < (1UL << 32)){
191                     // uint 32
192                     const temp = convertEndianTo!32(value);
193 
194                     store_[0] = Format.UINT32;
195                     *cast(uint*)&store_[Offset] = temp;
196                     stream_.put(store_[0..Offset + uint.sizeof]);
197                 } else {
198                     // uint 64
199                     const temp = convertEndianTo!64(value);
200 
201                     store_[0] = Format.UINT64;
202                     *cast(ulong*)&store_[Offset] = temp;
203                     stream_.put(store_[0..Offset + ulong.sizeof]);
204                 }
205             }
206         }
207 
208         return this;
209     }
210 
211 
212     /// ditto
213     ref PackerImpl pack(T)(in T value) if (isSigned!T && isIntegral!T && !is(Unqual!T == enum))
214     {
215         // long < long is slower than int < int
216         static if (!is(Unqual!T == long)) {
217             enum Bits = T.sizeof * 8;
218 
219             if (value < -(1 << 5)) {
220                 if (value < -(1 << 15)) {
221                     // int 32
222                     const temp = convertEndianTo!32(value);
223 
224                     store_[0] = Format.INT32;
225                     *cast(int*)&store_[Offset] = temp;
226                     stream_.put(store_[0..Offset + int.sizeof]);
227                 } else if (value < -(1 << 7)) {
228                     // int 16
229                     const temp = convertEndianTo!16(value);
230 
231                     store_[0] = Format.INT16;
232                     *cast(short*)&store_[Offset] = temp;
233                     stream_.put(store_[0..Offset + short.sizeof]);
234                 } else {
235                     // int 8
236                     store_[0] = Format.INT8;
237                     store_[1] = take8from!Bits(value);
238                     stream_.put(store_[0..Offset + byte.sizeof]);
239                 }
240             } else if (value < (1 << 7)) {
241                 // fixnum
242                 stream_.put(take8from!Bits(value));
243             } else {
244                 if (value < (1 << 8)) {
245                     // uint 8
246                     store_[0] = Format.UINT8;
247                     store_[1] = take8from!Bits(value);
248                     stream_.put(store_[0..Offset + ubyte.sizeof]);
249                 } else if (value < (1 << 16)) {
250                     // uint 16
251                     const temp = convertEndianTo!16(value);
252 
253                     store_[0] = Format.UINT16;
254                     *cast(ushort*)&store_[Offset] = temp;
255                     stream_.put(store_[0..Offset + ushort.sizeof]);
256                 } else {
257                     // uint 32
258                     const temp = convertEndianTo!32(value);
259 
260                     store_[0] = Format.UINT32;
261                     *cast(uint*)&store_[Offset] = temp;
262                     stream_.put(store_[0..Offset + uint.sizeof]);
263                 }
264             }
265         } else {
266             if (value < -(1L << 5)) {
267                 if (value < -(1L << 15)) {
268                     if (value < -(1L << 31)) {
269                         // int 64
270                         const temp = convertEndianTo!64(value);
271 
272                         store_[0] = Format.INT64;
273                         *cast(long*)&store_[Offset] = temp;
274                         stream_.put(store_[0..Offset + long.sizeof]);
275                     } else {
276                         // int 32
277                         const temp = convertEndianTo!32(value);
278 
279                         store_[0] = Format.INT32;
280                         *cast(int*)&store_[Offset] = temp;
281                         stream_.put(store_[0..Offset + int.sizeof]);
282                     }
283                 } else {
284                     if (value < -(1L << 7)) {
285                         // int 16
286                         const temp = convertEndianTo!16(value);
287 
288                         store_[0] = Format.INT16;
289                         *cast(short*)&store_[Offset] = temp;
290                         stream_.put(store_[0..Offset + short.sizeof]);
291                     } else {
292                         // int 8
293                         store_[0] = Format.INT8;
294                         store_[1] = take8from!64(value);
295                         stream_.put(store_[0..Offset + byte.sizeof]);
296                     }
297                 }
298             } else if (value < (1L << 7)) {
299                 // fixnum
300                 stream_.put(take8from!64(value));
301             } else {
302                 if (value < (1L << 16)) {
303                     if (value < (1L << 8)) {
304                         // uint 8
305                         store_[0] = Format.UINT8;
306                         store_[1] = take8from!64(value);
307                         stream_.put(store_[0..Offset + ubyte.sizeof]);
308                     } else {
309                         // uint 16
310                         const temp = convertEndianTo!16(value);
311 
312                         store_[0] = Format.UINT16;
313                         *cast(ushort*)&store_[Offset] = temp;
314                         stream_.put(store_[0..Offset + ushort.sizeof]);
315                     }
316                 } else {
317                     if (value < (1L << 32)) {
318                         // uint 32
319                         const temp = convertEndianTo!32(value);
320 
321                         store_[0] = Format.UINT32;
322                         *cast(uint*)&store_[Offset] = temp;
323                         stream_.put(store_[0..Offset + uint.sizeof]);
324                     } else {
325                         // uint 64
326                         const temp = convertEndianTo!64(value);
327 
328                         store_[0] = Format.UINT64;
329                         *cast(ulong*)&store_[Offset] = temp;
330                         stream_.put(store_[0..Offset + ulong.sizeof]);
331                     }
332                 }
333             }
334         }
335 
336         return this;
337     }
338 
339 
340     /// ditto
341     ref PackerImpl pack(T)(in T value) if (isSomeChar!T && !is(Unqual!T == enum))
342     {
343         static if (is(Unqual!T == char)) {
344             return pack(cast(ubyte)(value));
345         } else static if (is(Unqual!T == wchar)) {
346             return pack(cast(ushort)(value));
347         } else static if (is(Unqual!T == dchar)) {
348             return pack(cast(uint)(value));
349         }
350     }
351 
352 
353     /// ditto
354     ref PackerImpl pack(T)(in T value) if (isFloatingPoint!T && !is(Unqual!T == enum))
355     {
356         static if (is(Unqual!T == float)) {
357             const temp = convertEndianTo!32(_f(value).i);
358 
359             store_[0] = Format.FLOAT;
360             *cast(uint*)&store_[Offset] = temp;
361             stream_.put(store_[0..Offset + uint.sizeof]);
362         } else static if (is(Unqual!T == double)) {
363             const temp = convertEndianTo!64(_d(value).i);
364 
365             store_[0] = Format.DOUBLE;
366             *cast(ulong*)&store_[Offset] = temp;
367             stream_.put(store_[0..Offset + ulong.sizeof]);
368         } else {
369             static if ((real.sizeof > double.sizeof) && EnableReal) {
370                 store_[0]      = Format.REAL;
371                 const temp     = _r(value);
372                 const fraction = convertEndianTo!64(temp.fraction);
373                 const exponent = convertEndianTo!16(temp.exponent);
374 
375                 *cast(Unqual!(typeof(fraction))*)&store_[Offset]                   = fraction;
376                 *cast(Unqual!(typeof(exponent))*)&store_[Offset + fraction.sizeof] = exponent;
377                 stream_.put(store_[0..$]);
378             } else {  // Non-x86 CPUs, real type equals double type.
379                 pack(cast(double)value);
380             }
381         }
382 
383         return this;
384     }
385 
386 
387     /// ditto
388     ref PackerImpl pack(T)(in T value) if (is(Unqual!T == enum))
389     {
390         pack(cast(OriginalType!T)value);
391 
392         return this;
393     }
394 
395 
396     /// Overload for pack(null) for 2.057 or later
397     static if (!is(typeof(null) == void*))
398     {
399         ref PackerImpl pack(T)(in T value) if (is(Unqual!T == typeof(null)))
400         {
401             return packNil();
402         }
403     }
404 
405 
406     /// ditto
407     ref PackerImpl pack(T)(in T value) if (isPointer!T)
408     {
409         static if (is(Unqual!T == void*)) {  // for pack(null) for 2.056 or earlier
410             enforce(value is null, "Can't serialize void type");
411             stream_.put(Format.NIL);
412         } else {
413             if (value is null)
414                 stream_.put(Format.NIL);
415             else
416                 pack(mixin(AsteriskOf!T ~ "value"));
417         }
418 
419         return this;
420     }
421 
422 
423     /// ditto
424     ref PackerImpl pack(T)(in T array) if ((isArray!T || isInstanceOf!(Array, T)) && !is(Unqual!T == enum))
425     {
426         alias typeof(T.init[0]) U;
427 
428         if (array.empty)
429             return packNil();
430 
431         // Raw bytes
432         static if (isByte2!(U) || isSomeChar!(U)) {
433             ubyte[] raw = cast(ubyte[])array;
434 
435             beginRaw(raw.length);
436             stream_.put(raw);
437         } else {
438             beginArray(array.length);
439             foreach (elem; array)
440                 pack(elem);
441         }
442 
443         return this;
444     }
445 
446 
447     /// ditto
448     ref PackerImpl pack(T)(in T array) if (isAssociativeArray!T)
449     {
450         if (array is null)
451             return packNil();
452 
453         beginMap(array.length);
454         foreach (key, value; array) {
455             pack(key);
456             pack(value);
457         }
458 
459         return this;
460     }
461 
462 
463     /// ditto
464     ref PackerImpl pack(Types...)(auto ref const Types objects) if (Types.length > 1)
465     {
466         foreach (i, T; Types)
467             pack(objects[i]);
468 
469         return this;
470     }
471 
472 
473     /**
474      * Serializes $(D_PARAM object) and writes to stream.
475      *
476      * Calling $(D toMsgpack) if $(D_KEYWORD class) and $(D_KEYWORD struct) implement $(D toMsgpack) method. $(D toMsgpack) signature is:
477      * -----
478      * void toMsgpack(Packer)(ref Packer packer) const
479      * -----
480      * This method serializes all members of T object if $(D_KEYWORD class) and $(D_KEYWORD struct) don't implement $(D toMsgpack).
481      *
482      * An object that doesn't implement $(D toMsgpack) is serialized to Array type.
483      * -----
484      * packer.pack(tuple(true, 1, "Hi!"))  // -> '[true, 1, "Hi!"]', not 'ture, 1, "Hi!"'
485      *
486      * struct Foo
487      * {
488      *     int num    = 10;
489      *     string msg = "D!";
490      * }
491      * packer.pack(Foo());  // -> '[10, "D!"]'
492      *
493      * class Base
494      * {
495      *     bool flag = true;
496      * }
497      * class Derived : Base
498      * {
499      *     double = 0.5f;
500      * }
501      * packer.pack(new Derived());  // -> '[true, 0.5f]'
502      * -----
503      *
504      * Params:
505      *  object = the content to serialize.
506      *
507      * Returns:
508      *  self, i.e. for method chaining.
509      */
510     ref PackerImpl pack(T)(in T object) if (is(Unqual!T == class))
511     {
512         if (object is null)
513             return packNil();
514 
515         static if (hasMember!(T, "toMsgpack"))
516         {
517             static if (__traits(compiles, { object.toMsgpack(this, withFieldName_); })) {
518                 object.toMsgpack(this, withFieldName_);
519             } else static if (__traits(compiles, { object.toMsgpack(this); })) { // backward compatible
520                 object.toMsgpack(this);
521             } else {
522                 static assert(0, "Failed to invoke 'toMsgpack' on type '" ~ Unqual!T.stringof ~ "'");
523             }
524         } else {
525             if (auto handler = object.classinfo in packHandlers) {
526                 (*handler)(this, cast(void*)&object);
527                 return this;
528             }
529             if (T.classinfo !is object.classinfo) {
530                 throw new MessagePackException("Can't pack derived class through reference to base class.");
531             }
532 
533             packObject!(T)(object);
534         }
535 
536         return this;
537     }
538 
539 
540     /// ditto
541     @trusted
542     ref PackerImpl pack(T)(auto ref T object) if (is(Unqual!T == struct) &&
543                                                   !isInstanceOf!(Array, T) &&
544                                                   !is(Unqual!T == ExtValue))
545     {
546         static if (hasMember!(T, "toMsgpack"))
547         {
548             static if (__traits(compiles, { object.toMsgpack(this, withFieldName_); })) {
549                 object.toMsgpack(this, withFieldName_);
550             } else static if (__traits(compiles, { object.toMsgpack(this); })) { // backward compatible
551                 object.toMsgpack(this);
552             } else {
553                 static assert(0, "Failed to invoke 'toMsgpack' on type '" ~ Unqual!T.stringof ~ "'");
554             }
555         } else static if (isTuple!T) {
556             beginArray(object.field.length);
557             foreach (f; object.field)
558                 pack(f);
559         } else {  // simple struct
560             if (auto handler = typeid(Unqual!T) in packHandlers) {
561                 (*handler)(this, cast(void*)&object);
562                 return this;
563             }
564 
565             immutable memberNum = SerializingMemberNumbers!(T);
566             if (withFieldName_)
567                 beginMap(memberNum);
568             else
569                 beginArray(memberNum);
570 
571             if (withFieldName_) {
572                 foreach (i, f; object.tupleof) {
573                     static if (isPackedField!(T.tupleof[i]) && __traits(compiles, { pack(f); }))
574                     {
575                         pack(getFieldName!(T, i));
576                         pack(f);
577                     }
578                 }
579             } else {
580                 foreach (i, f; object.tupleof) {
581                     static if (isPackedField!(T.tupleof[i]) && __traits(compiles, { pack(f); }))
582                         pack(f);
583                 }
584             }
585         }
586 
587         return this;
588     }
589 
590 
591     void packObject(T)(in T object) if (is(Unqual!T == class))
592     {
593         alias SerializingClasses!(T) Classes;
594 
595         immutable memberNum = SerializingMemberNumbers!(Classes);
596         if (withFieldName_)
597             beginMap(memberNum);
598         else
599             beginArray(memberNum);
600 
601         foreach (Class; Classes) {
602             Class obj = cast(Class)object;
603             if (withFieldName_) {
604                 foreach (i, f ; obj.tupleof) {
605                     static if (isPackedField!(Class.tupleof[i])) {
606                         pack(getFieldName!(Class, i));
607                         pack(f);
608                     }
609                 }
610             } else {
611                 foreach (i, f ; obj.tupleof) {
612                     static if (isPackedField!(Class.tupleof[i]))
613                         pack(f);
614                 }
615             }
616         }
617     }
618 
619 
620     /**
621      * Serializes the arguments as container to stream.
622      *
623      * -----
624      * packer.packArray(true, 1);  // -> [true, 1]
625      * packer.packMap("Hi", 100);  // -> ["Hi":100]
626      * -----
627      *
628      * In packMap, the number of arguments must be even.
629      *
630      * Params:
631      *  objects = the contents to serialize.
632      *
633      * Returns:
634      *  self, i.e. for method chaining.
635      */
636     ref PackerImpl packArray(Types...)(auto ref const Types objects)
637     {
638         beginArray(Types.length);
639         foreach (i, T; Types)
640             pack(objects[i]);
641         //pack(objects);  // slow :(
642 
643         return this;
644     }
645 
646 
647     /// ditto
648     ref PackerImpl packMap(Types...)(auto ref const Types objects)
649     {
650         static assert(Types.length % 2 == 0, "The number of arguments must be even");
651 
652         beginMap(Types.length / 2);
653         foreach (i, T; Types)
654             pack(objects[i]);
655 
656         return this;
657     }
658 
659     /**
660      * Packs $(D data) as an extended value of $(D type).
661      *
662      * ----
663      * packer.packExt(3, bytes);
664      * ----
665      *
666      * $(D type) must be a signed byte 0-127.
667      *
668      * Params:
669      *  type = the application-defined type for the data
670      *  data = an array of bytes
671      *
672      * Returns:
673      *  seld, i.e. for method chaining.
674      */
675     ref PackerImpl pack(T)(auto ref const T data) if (is(Unqual!T == ExtValue))
676     {
677         packExt(data.type, data.data);
678         return this;
679     }
680 
681     /**
682      * Packs $(D data) as an extended value of $(D type).
683      *
684      * ----
685      * packer.packExt(3, bytes);
686      * ----
687      *
688      * $(D type) must be a signed byte 0-127.
689      *
690      * Params:
691      *  type = the application-defined type for the data
692      *  data = an array of bytes
693      *
694      * Returns:
695      *  seld, i.e. for method chaining.
696      */
697     ref PackerImpl packExt(in byte type, const ubyte[] data)
698     {
699         ref PackerImpl packExtFixed(int fmt)
700         {
701             store_[0] = cast(ubyte)fmt;
702             store_[1] = type;
703             stream_.put(store_[0 .. 2]);
704             stream_.put(data);
705             return this;
706         }
707 
708         // Try packing to a fixed-length type
709         if (data.length == 1)
710             return packExtFixed(Format.EXT + 0);
711         else if (data.length == 2)
712             return packExtFixed(Format.EXT + 1);
713         else if (data.length == 4)
714             return packExtFixed(Format.EXT + 2);
715         else if (data.length == 8)
716             return packExtFixed(Format.EXT + 3);
717         else if (data.length == 16)
718             return packExtFixed(Format.EXT + 4);
719 
720         int typeByte = void;
721         if (data.length <= (2^^8)-1)
722         {
723             store_[0] = Format.EXT8;
724             store_[1] = cast(ubyte)data.length;
725             typeByte = 2;
726 
727         } else if (data.length <= (2^^16)-1) {
728             store_[0] = Format.EXT16;
729             const temp = convertEndianTo!16(data.length);
730             *cast(ushort*)&store_[Offset] = temp;
731             typeByte = 3;
732         } else if (data.length <= (2^^32)-1) {
733             store_[0] = Format.EXT32;
734             const temp = convertEndianTo!32(data.length);
735             *cast(uint*)&store_[Offset] = temp;
736             typeByte = 5;
737         } else
738             throw new MessagePackException("Data too large to pack as EXT");
739 
740         store_[typeByte] = type;
741         stream_.put(store_[0..typeByte+1]);
742         stream_.put(data);
743 
744         return this;
745     }
746 
747     /*
748      * Serializes raw type-information to stream for binary type.
749      */
750     void beginRaw(in size_t length)
751     {
752         import std.conv : text;
753 
754         if (length < 32) {
755             const ubyte temp = Format.RAW | cast(ubyte)length;
756             stream_.put(take8from(temp));
757         } else if (length < 65536) {
758             const temp = convertEndianTo!16(length);
759 
760             store_[0] = Format.RAW16;
761             *cast(ushort*)&store_[Offset] = temp;
762             stream_.put(store_[0..Offset + ushort.sizeof]);
763         } else {
764             if (length > 0xffffffff)
765                 throw new MessagePackException(text("size of raw is too long to pack: ", length,  " bytes should be <= ", 0xffffffff));
766 
767             const temp = convertEndianTo!32(length);
768 
769             store_[0] = Format.RAW32;
770             *cast(uint*)&store_[Offset] = temp;
771             stream_.put(store_[0..Offset + uint.sizeof]);
772         }
773     }
774 
775     /**
776      * Serializes the type-information to stream.
777      *
778      * These methods don't serialize contents.
779      * You need to call pack method to serialize contents at your own risk.
780      * -----
781      * packer.beginArray(3).pack(true, 1);  // -> [true, 1,
782      *
783      * // other operation
784      *
785      * packer.pack("Hi!");                  // -> [true, 1, "Hi!"]
786      * -----
787      *
788      * Params:
789      *  length = the length of container.
790      *
791      * Returns:
792      *  self, i.e. for method chaining.
793      */
794     ref PackerImpl beginArray(in size_t length)
795     {
796         if (length < 16) {
797             const ubyte temp = Format.ARRAY | cast(ubyte)length;
798             stream_.put(take8from(temp));
799         } else if (length < 65536) {
800             const temp = convertEndianTo!16(length);
801 
802             store_[0] = Format.ARRAY16;
803             *cast(ushort*)&store_[Offset] = temp;
804             stream_.put(store_[0..Offset + ushort.sizeof]);
805         } else {
806             const temp = convertEndianTo!32(length);
807 
808             store_[0] = Format.ARRAY32;
809             *cast(uint*)&store_[Offset] = temp;
810             stream_.put(store_[0..Offset + uint.sizeof]);
811         }
812 
813         return this;
814     }
815 
816 
817     /// ditto
818     ref PackerImpl beginMap(in size_t length)
819     {
820         if (length < 16) {
821             const ubyte temp = Format.MAP | cast(ubyte)length;
822             stream_.put(take8from(temp));
823         } else if (length < 65536) {
824             const temp = convertEndianTo!16(length);
825 
826             store_[0] = Format.MAP16;
827             *cast(ushort*)&store_[Offset] = temp;
828             stream_.put(store_[0..Offset + ushort.sizeof]);
829         } else {
830             const temp = convertEndianTo!32(length);
831 
832             store_[0] = Format.MAP32;
833             *cast(uint*)&store_[Offset] = temp;
834             stream_.put(store_[0..Offset + uint.sizeof]);
835         }
836 
837         return this;
838     }
839 
840 
841   private:
842     /*
843      * Serializes the nil value.
844      */
845     ref PackerImpl packNil()
846     {
847         stream_.put(Format.NIL);
848         return this;
849     }
850 }
851 
852 
853 /// Default serializer
854 alias PackerImpl!(Appender!(ubyte[])) Packer;  // should be pure struct?
855 
856 
857 /**
858  * Helper for $(D Packer) construction.
859  *
860  * Params:
861  *  stream = the stream to write.
862  *  withFieldName = serialize class / struct with field name
863  *
864  * Returns:
865  *  a $(D Packer) object instantiated and initialized according to the arguments.
866  */
867 PackerImpl!(Stream) packer(Stream)(Stream stream, bool withFieldName = false)
868 {
869     return typeof(return)(stream, withFieldName);
870 }
871 
872 
873 version(unittest)
874 {
875     package import std.file, core.stdc..string;
876 
877     package mixin template DefinePacker()
878     {
879         Packer packer;
880     }
881 
882     package mixin template DefineDictionalPacker()
883     {
884         Packer packer = Packer(false);
885     }
886 }
887 
888 
889 unittest
890 {
891     { // unique value
892         mixin DefinePacker;
893 
894         ubyte[] result = [Format.NIL, Format.TRUE, Format.FALSE];
895 
896         packer.pack(null, true, false);
897         foreach (i, value; packer.stream.data)
898             assert(value == result[i]);
899     }
900     { // uint *
901         static struct UTest { ubyte format; ulong value; }
902 
903         enum : ulong { A = ubyte.max, B = ushort.max, C = uint.max, D = ulong.max }
904 
905         static UTest[][] utests = [
906             [{Format.UINT8, A}],
907             [{Format.UINT8, A}, {Format.UINT16, B}],
908             [{Format.UINT8, A}, {Format.UINT16, B}, {Format.UINT32, C}],
909             [{Format.UINT8, A}, {Format.UINT16, B}, {Format.UINT32, C}, {Format.UINT64, D}],
910         ];
911 
912         foreach (I, T; TypeTuple!(ubyte, ushort, uint, ulong)) {
913             foreach (i, test; utests[I]) {
914                 mixin DefinePacker;
915 
916                 packer.pack(cast(T)test.value);
917                 assert(packer.stream.data[0] == test.format);
918 
919                 switch (i) {
920                 case 0:
921                     auto answer = take8from!(T.sizeof * 8)(test.value);
922                     assert(memcmp(&packer.stream.data[1], &answer, ubyte.sizeof) == 0);
923                     break;
924                 case 1:
925                     auto answer = convertEndianTo!16(test.value);
926                     assert(memcmp(&packer.stream.data[1], &answer, ushort.sizeof) == 0);
927                     break;
928                 case 2:
929                     auto answer = convertEndianTo!32(test.value);
930                     assert(memcmp(&packer.stream.data[1], &answer, uint.sizeof) == 0);
931                     break;
932                 default:
933                     auto answer = convertEndianTo!64(test.value);
934                     assert(memcmp(&packer.stream.data[1], &answer, ulong.sizeof) == 0);
935                 }
936             }
937         }
938     }
939     { // int *
940         static struct STest { ubyte format; long value; }
941 
942         enum : long { A = byte.min, B = short.min, C = int.min, D = long.min }
943 
944         static STest[][] stests = [
945             [{Format.INT8, A}],
946             [{Format.INT8, A}, {Format.INT16, B}],
947             [{Format.INT8, A}, {Format.INT16, B}, {Format.INT32, C}],
948             [{Format.INT8, A}, {Format.INT16, B}, {Format.INT32, C}, {Format.INT64, D}],
949         ];
950 
951         foreach (I, T; TypeTuple!(byte, short, int, long)) {
952             foreach (i, test; stests[I]) {
953                 mixin DefinePacker;
954 
955                 packer.pack(cast(T)test.value);
956                 assert(packer.stream.data[0] == test.format);
957 
958                 switch (i) {
959                 case 0:
960                     auto answer = take8from!(T.sizeof * 8)(test.value);
961                     assert(memcmp(&packer.stream.data[1], &answer, byte.sizeof) == 0);
962                     break;
963                 case 1:
964                     auto answer = convertEndianTo!16(test.value);
965                     assert(memcmp(&packer.stream.data[1], &answer, short.sizeof) == 0);
966                     break;
967                 case 2:
968                     auto answer = convertEndianTo!32(test.value);
969                     assert(memcmp(&packer.stream.data[1], &answer, int.sizeof) == 0);
970                     break;
971                 default:
972                     auto answer = convertEndianTo!64(test.value);
973                     assert(memcmp(&packer.stream.data[1], &answer, long.sizeof) == 0);
974                 }
975             }
976         }
977     }
978     { // fload, double
979         static if ((real.sizeof == double.sizeof) || !EnableReal)
980         {
981             alias TypeTuple!(float, double, double) FloatingTypes;
982             static struct FTest { ubyte format; double value; }
983 
984             static FTest[] ftests = [
985                 {Format.FLOAT,  float.min_normal},
986                 {Format.DOUBLE, double.max},
987                 {Format.DOUBLE, double.max},
988             ];
989         }
990         else
991         {
992             alias TypeTuple!(float, double, real) FloatingTypes;
993             static struct FTest { ubyte format; real value; }
994 
995             static FTest[] ftests = [
996                 {Format.FLOAT,  float.min_normal},
997                 {Format.DOUBLE, double.max},
998                 {Format.REAL,   real.max},
999             ];
1000         }
1001 
1002         foreach (I, T; FloatingTypes) {
1003             mixin DefinePacker;
1004 
1005             packer.pack(cast(T)ftests[I].value);
1006             assert(packer.stream.data[0] == ftests[I].format);
1007 
1008             switch (I) {
1009             case 0:
1010                 const answer = convertEndianTo!32(_f(cast(T)ftests[I].value).i);
1011                 assert(memcmp(&packer.stream.data[1], &answer, float.sizeof) == 0);
1012                 break;
1013             case 1:
1014                 const answer = convertEndianTo!64(_d(cast(T)ftests[I].value).i);
1015                 assert(memcmp(&packer.stream.data[1], &answer, double.sizeof) == 0);
1016                 break;
1017             default:
1018                 static if (EnableReal)
1019                 {
1020                     const t = _r(cast(T)ftests[I].value);
1021                     const f = convertEndianTo!64(t.fraction);
1022                     const e = convertEndianTo!16(t.exponent);
1023                     assert(memcmp(&packer.stream.data[1],            &f, f.sizeof) == 0);
1024                     assert(memcmp(&packer.stream.data[1 + f.sizeof], &e, e.sizeof) == 0);
1025                 }
1026                 else
1027                 {
1028                     const answer = convertEndianTo!64(_d(cast(T)ftests[I].value).i);
1029                     assert(memcmp(&packer.stream.data[1], &answer, double.sizeof) == 0);
1030                 }
1031             }
1032         }
1033     }
1034     { // pointer
1035         static struct PTest
1036         {
1037             ubyte format;
1038 
1039             union
1040             {
1041                 ulong*  p0;
1042                 long*   p1;
1043                 double* p2;
1044             }
1045         }
1046 
1047         PTest[] ptests = [PTest(Format.UINT64), PTest(Format.INT64), PTest(Format.DOUBLE)];
1048 
1049         ulong  v0 = ulong.max;
1050         long   v1 = long.min;
1051         double v2 = double.max;
1052 
1053         foreach (I, Index; TypeTuple!("0", "1", "2")) {
1054             mixin DefinePacker;
1055 
1056             mixin("ptests[I].p" ~ Index ~ " = &v" ~ Index ~ ";");
1057 
1058             packer.pack(mixin("ptests[I].p" ~ Index));
1059             assert(packer.stream.data[0] == ptests[I].format);
1060 
1061             switch (I) {
1062             case 0:
1063                 auto answer = convertEndianTo!64(*ptests[I].p0);
1064                 assert(memcmp(&packer.stream.data[1], &answer, ulong.sizeof) == 0);
1065                 break;
1066             case 1:
1067                 auto answer = convertEndianTo!64(*ptests[I].p1);
1068                 assert(memcmp(&packer.stream.data[1], &answer, long.sizeof) == 0);
1069                 break;
1070             default:
1071                 const answer = convertEndianTo!64(_d(*ptests[I].p2).i);
1072                 assert(memcmp(&packer.stream.data[1], &answer, double.sizeof) == 0);
1073             }
1074         }
1075     }
1076     { // enum
1077         enum E : ubyte { A = ubyte.max }
1078 
1079         mixin DefinePacker; E e = E.A;
1080 
1081         packer.pack(e);
1082         assert(packer.stream.data[0] == Format.UINT8);
1083 
1084         auto answer = E.A;
1085         assert(memcmp(&packer.stream.data[1], &answer, (OriginalType!E).sizeof) == 0);
1086     }
1087     { // enum with string
1088         enum E2 : string { A = "test" }
1089 
1090         mixin DefinePacker; E2 e = E2.A;
1091 
1092         packer.pack(e);
1093         assert(packer.stream.data[0] == (Format.RAW | 0x04));
1094     }
1095     { // container
1096         static struct CTest { ubyte format; size_t value; }
1097 
1098         enum : ulong { A = 16 / 2, B = ushort.max, C = uint.max }
1099 
1100         static CTest[][] ctests = [
1101             [{Format.ARRAY | A, Format.ARRAY | A}, {Format.ARRAY16, B}, {Format.ARRAY32, C}],
1102             [{Format.MAP   | A, Format.MAP   | A}, {Format.MAP16,   B}, {Format.MAP32,   C}],
1103             [{Format.RAW   | A, Format.RAW   | A}, {Format.RAW16,   B}, {Format.RAW32,   C}],
1104         ];
1105 
1106         foreach (I, Name; TypeTuple!("Array", "Map", "Raw")) {
1107             auto test = ctests[I];
1108 
1109             foreach (i, T; TypeTuple!(ubyte, ushort, uint)) {
1110                 mixin DefinePacker;
1111                 mixin("packer.begin" ~ Name ~ "(i ? test[i].value : A);");
1112 
1113                 assert(packer.stream.data[0] == test[i].format);
1114 
1115                 switch (i) {
1116                 case 0:
1117                     auto answer = take8from(test[i].value);
1118                     assert(memcmp(&packer.stream.data[0], &answer, ubyte.sizeof) == 0);
1119                     break;
1120                 case 1:
1121                     auto answer = convertEndianTo!16(test[i].value);
1122                     assert(memcmp(&packer.stream.data[1], &answer, ushort.sizeof) == 0);
1123                     break;
1124                 default:
1125                     auto answer = convertEndianTo!32(test[i].value);
1126                     assert(memcmp(&packer.stream.data[1], &answer, uint.sizeof) == 0);
1127                 }
1128             }
1129         }
1130     }
1131     { // larger spec size for string / binary
1132         mixin DefinePacker;
1133 
1134         try {
1135             byte[] bins = new byte[0xffffffffUL + 1];
1136             packer.pack(bins);
1137             assert(false);
1138         } catch (MessagePackException e) {
1139         }
1140     }
1141     { // user defined
1142         {
1143             static struct S
1144             {
1145                 uint num = uint.max;
1146 
1147                 void toMsgpack(P)(ref P p) const { p.packArray(num); }
1148             }
1149 
1150             mixin DefinePacker; S test;
1151 
1152             packer.pack(test);
1153 
1154             assert(packer.stream.data[0] == (Format.ARRAY | 1));
1155             assert(packer.stream.data[1] ==  Format.UINT32);
1156             assert(memcmp(&packer.stream.data[2], &test.num, uint.sizeof) == 0);
1157         }
1158         {
1159             mixin DefinePacker; auto test = tuple(true, false, uint.max);
1160 
1161             packer.pack(test);
1162 
1163             assert(packer.stream.data[0] == (Format.ARRAY | 3));
1164             assert(packer.stream.data[1] ==  Format.TRUE);
1165             assert(packer.stream.data[2] ==  Format.FALSE);
1166             assert(packer.stream.data[3] ==  Format.UINT32);
1167             assert(memcmp(&packer.stream.data[4], &test.field[2], uint.sizeof) == 0);
1168         }
1169         {
1170             static class C
1171             {
1172                 uint num;
1173 
1174                 this(uint n) { num = n; }
1175 
1176                 void toMsgpack(P)(ref P p) const { p.packArray(num); }
1177             }
1178 
1179             mixin DefinePacker; C test = new C(ushort.max);
1180 
1181             packer.pack(test);
1182 
1183             assert(packer.stream.data[0] == (Format.ARRAY | 1));
1184             assert(packer.stream.data[1] ==  Format.UINT16);
1185             assert(memcmp(&packer.stream.data[2], &test.num, ushort.sizeof) == 0);
1186         }
1187     }
1188     { // simple struct and class
1189         {
1190             static struct Simple
1191             {
1192                 uint num = uint.max;
1193             }
1194 
1195             static struct SimpleWithNonPacked1
1196             {
1197                 uint num = uint.max;
1198                 @nonPacked string str = "ignored";
1199             }
1200 
1201             static struct SimpleWithNonPacked2
1202             {
1203                 @nonPacked string str = "ignored";
1204                 uint num = uint.max;
1205             }
1206 
1207             static struct SimpleWithSkippedTypes
1208             {
1209                 int function(int) fn;
1210                 int delegate(int) dg;
1211                 uint num = uint.max;
1212             }
1213 
1214             foreach (Type; TypeTuple!(Simple, SimpleWithNonPacked1, SimpleWithNonPacked2, SimpleWithSkippedTypes)) {
1215                 mixin DefinePacker;
1216 
1217                 Type test;
1218                 packer.pack(test);
1219 
1220                 assert(packer.stream.data[0] == (Format.ARRAY | 1));
1221                 assert(packer.stream.data[1] ==  Format.UINT32);
1222                 assert(memcmp(&packer.stream.data[2], &test.num, uint.sizeof) == 0);
1223             }
1224         }
1225 
1226         static class SimpleA
1227         {
1228             bool flag = true;
1229         }
1230 
1231         static class SimpleB : SimpleA
1232         {
1233             ubyte type = 100;
1234         }
1235 
1236         static class SimpleC : SimpleB
1237         {
1238             uint num = uint.max;
1239         }
1240 
1241         static class SimpleCWithNonPacked1 : SimpleB
1242         {
1243             uint num = uint.max;
1244             @nonPacked string str = "ignored";
1245         }
1246 
1247         static class SimpleCWithNonPacked2 : SimpleB
1248         {
1249             @nonPacked string str = "ignored";
1250             uint num = uint.max;
1251         }
1252 
1253         static class SimpleCWithSkippedTypes : SimpleB
1254         {
1255             uint num = uint.max;
1256             int function(int) fn;
1257             int delegate(int) dg;
1258         }
1259 
1260         {  // from derived class
1261             foreach (Type; TypeTuple!(SimpleC, SimpleCWithNonPacked1, SimpleCWithNonPacked2, SimpleCWithSkippedTypes)) {
1262                 mixin DefinePacker;
1263 
1264                 Type test = new Type();
1265                 packer.pack(test);
1266 
1267                 assert(packer.stream.data[0] == (Format.ARRAY | 3));
1268                 assert(packer.stream.data[1] ==  Format.TRUE);
1269                 assert(packer.stream.data[2] ==  100);
1270                 assert(packer.stream.data[3] ==  Format.UINT32);
1271                 assert(memcmp(&packer.stream.data[4], &test.num, uint.sizeof) == 0);
1272             }
1273         }
1274         {  // from base class
1275             mixin DefinePacker; SimpleB test = new SimpleC();
1276 
1277             try {
1278                 packer.pack(test);
1279                 assert(false);
1280             } catch (Exception e) { }
1281         }
1282     }
1283 
1284     // ext types
1285     {
1286         byte type = 7; // an arbitrary type value
1287 
1288         // fixexts
1289         {
1290             ubyte[16] data;
1291             data[] = 1;
1292             foreach (L; TypeTuple!(1, 2, 4, 8, 16))
1293             {
1294                 mixin DefinePacker;
1295                 packer.pack(ExtValue(type, data[0 .. L]));
1296 
1297                 // format, type, data
1298                 assert(packer.stream.data.length == 2 + L);
1299                 const l = 2 ^^ (packer.stream.data[0] - Format.EXT);
1300                 assert(l == L);
1301                 assert(packer.stream.data[1] == type);
1302                 assert(packer.stream.data[2 .. 2+l] == data[0 .. L]);
1303             }
1304         }
1305 
1306         // ext8
1307         {
1308             foreach (L; TypeTuple!(3, 7, 255))
1309             {
1310                 ubyte[] data = new ubyte[](L);
1311                 data[] = 1;
1312 
1313                 mixin DefinePacker;
1314                 packer.pack(ExtValue(type, data[0 .. L]));
1315 
1316                 // format, length, type, data
1317                 assert(packer.stream.data.length == 3 + L);
1318                 assert(packer.stream.data[0] == Format.EXT8);
1319                 assert(packer.stream.data[1] == L);
1320                 assert(packer.stream.data[2] == type);
1321                 assert(packer.stream.data[3 .. 3 + L] == data);
1322             }
1323         }
1324 
1325         // ext16
1326         {
1327             foreach (L; TypeTuple!(256, (2^^16)-1))
1328             {
1329                 ubyte[] data = new ubyte[](L);
1330                 data[] = 1;
1331 
1332                 mixin DefinePacker;
1333                 packer.pack(ExtValue(type, data[0 .. L]));
1334 
1335                 // format, length, type, data
1336                 import std.conv : text;
1337                 assert(packer.stream.data.length == 4 + L, text(packer.stream.data.length));
1338                 assert(packer.stream.data[0] == Format.EXT16);
1339 
1340                 ushort l = convertEndianTo!16(L);
1341                 assert(memcmp(&packer.stream.data[1], &l, ushort.sizeof) == 0);
1342                 assert(packer.stream.data[3] == type);
1343                 assert(packer.stream.data[4 .. 4 + L] == data);
1344             }
1345         }
1346 
1347         // ext32
1348         {
1349             foreach (L; TypeTuple!(2^^16, 2^^17))
1350             {
1351                 ubyte[] data = new ubyte[](L);
1352                 data[] = 1;
1353 
1354                 mixin DefinePacker;
1355                 packer.pack(ExtValue(type, data[0 .. L]));
1356 
1357                 // format, length, type, data
1358                 import std.conv : text;
1359                 assert(packer.stream.data.length == 6 + L, text(packer.stream.data.length));
1360                 assert(packer.stream.data[0] == Format.EXT32);
1361 
1362                 uint l = convertEndianTo!32(L);
1363                 assert(memcmp(&packer.stream.data[1], &l, uint.sizeof) == 0);
1364                 assert(packer.stream.data[5] == type);
1365                 assert(packer.stream.data[6 .. 6 + L] == data);
1366             }
1367         }
1368     }
1369 }