| // Written in the D programming language. |
| |
| /** |
| This module implements a |
| $(HTTP erdani.org/publications/cuj-04-2002.php.html,discriminated union) |
| type (a.k.a. |
| $(HTTP en.wikipedia.org/wiki/Tagged_union,tagged union), |
| $(HTTP en.wikipedia.org/wiki/Algebraic_data_type,algebraic type)). |
| Such types are useful |
| for type-uniform binary interfaces, interfacing with scripting |
| languages, and comfortable exploratory programming. |
| |
| A $(LREF Variant) object can hold a value of any type, with very few |
| restrictions (such as `shared` types and noncopyable types). Setting the value |
| is as immediate as assigning to the `Variant` object. To read back the value of |
| the appropriate type `T`, use the $(LREF get) method. To query whether a |
| `Variant` currently holds a value of type `T`, use $(LREF peek). To fetch the |
| exact type currently held, call $(LREF type), which returns the `TypeInfo` of |
| the current value. |
| |
| In addition to $(LREF Variant), this module also defines the $(LREF Algebraic) |
| type constructor. Unlike `Variant`, `Algebraic` only allows a finite set of |
| types, which are specified in the instantiation (e.g. $(D Algebraic!(int, |
| string)) may only hold an `int` or a `string`). |
| |
| $(RED Warning: $(LREF Algebraic) is outdated and not recommended for use in new |
| code. Instead, use $(REF SumType, std,sumtype).) |
| |
| Credits: Reviewed by Brad Roberts. Daniel Keep provided a detailed code review |
| prompting the following improvements: (1) better support for arrays; (2) support |
| for associative arrays; (3) friendlier behavior towards the garbage collector. |
| Copyright: Copyright Andrei Alexandrescu 2007 - 2015. |
| License: $(HTTP www.boost.org/LICENSE_1_0.txt, Boost License 1.0). |
| Authors: $(HTTP erdani.org, Andrei Alexandrescu) |
| Source: $(PHOBOSSRC std/variant.d) |
| */ |
| module std.variant; |
| |
| import std.meta, std.traits, std.typecons; |
| |
| /// |
| @system unittest |
| { |
| Variant a; // Must assign before use, otherwise exception ensues |
| // Initialize with an integer; make the type int |
| Variant b = 42; |
| assert(b.type == typeid(int)); |
| // Peek at the value |
| assert(b.peek!(int) !is null && *b.peek!(int) == 42); |
| // Automatically convert per language rules |
| auto x = b.get!(real); |
| |
| // Assign any other type, including other variants |
| a = b; |
| a = 3.14; |
| assert(a.type == typeid(double)); |
| // Implicit conversions work just as with built-in types |
| assert(a < b); |
| // Check for convertibility |
| assert(!a.convertsTo!(int)); // double not convertible to int |
| // Strings and all other arrays are supported |
| a = "now I'm a string"; |
| assert(a == "now I'm a string"); |
| |
| // can also assign arrays |
| a = new int[42]; |
| assert(a.length == 42); |
| a[5] = 7; |
| assert(a[5] == 7); |
| |
| // Can also assign class values |
| class Foo {} |
| auto foo = new Foo; |
| a = foo; |
| assert(*a.peek!(Foo) == foo); // and full type information is preserved |
| } |
| |
| /++ |
| Gives the `sizeof` the largest type given. |
| |
| See_Also: $(LINK https://forum.dlang.org/thread/wbpnncxepehgcswhuazl@forum.dlang.org?page=1) |
| +/ |
| template maxSize(Ts...) |
| { |
| align(1) union Impl |
| { |
| static foreach (i, T; Ts) |
| { |
| static if (!is(T == void)) |
| mixin("T _field_", i, ";"); |
| } |
| } |
| enum maxSize = Impl.sizeof; |
| } |
| |
| /// |
| @safe unittest |
| { |
| struct Cat { int a, b, c; } |
| |
| align(1) struct S |
| { |
| long l; |
| ubyte b; |
| } |
| |
| align(1) struct T |
| { |
| ubyte b; |
| long l; |
| } |
| |
| static assert(maxSize!(int, long) == 8); |
| static assert(maxSize!(bool, byte) == 1); |
| static assert(maxSize!(bool, Cat) == 12); |
| static assert(maxSize!(char) == 1); |
| static assert(maxSize!(char, short, ubyte) == 2); |
| static assert(maxSize!(char, long, ubyte) == 8); |
| import std.algorithm.comparison : max; |
| static assert(maxSize!(long, S) == max(long.sizeof, S.sizeof)); |
| static assert(maxSize!(S, T) == max(S.sizeof, T.sizeof)); |
| static assert(maxSize!(int, ubyte[7]) == 7); |
| static assert(maxSize!(int, ubyte[3]) == 4); |
| static assert(maxSize!(int, int, ubyte[3]) == 4); |
| static assert(maxSize!(void, int, ubyte[3]) == 4); |
| static assert(maxSize!(void) == 1); |
| } |
| |
| struct This; |
| |
| private alias This2Variant(V, T...) = AliasSeq!(ReplaceTypeUnless!(isAlgebraic, This, V, T)); |
| |
| // We can't just use maxAlignment because no types might be specified |
| // to VariantN, so handle that here and then pass along the rest. |
| private template maxVariantAlignment(U...) |
| if (isTypeTuple!U) |
| { |
| static if (U.length == 0) |
| { |
| import std.algorithm.comparison : max; |
| enum maxVariantAlignment = max(real.alignof, size_t.alignof); |
| } |
| else |
| enum maxVariantAlignment = maxAlignment!(U); |
| } |
| |
| /** |
| * Back-end type seldom used directly by user |
| * code. Two commonly-used types using `VariantN` are: |
| * |
| * $(OL $(LI $(LREF Algebraic): A closed discriminated union with a |
| * limited type universe (e.g., $(D Algebraic!(int, double, |
| * string)) only accepts these three types and rejects anything |
| * else).) $(LI $(LREF Variant): An open discriminated union allowing an |
| * unbounded set of types. If any of the types in the `Variant` |
| * are larger than the largest built-in type, they will automatically |
| * be boxed. This means that even large types will only be the size |
| * of a pointer within the `Variant`, but this also implies some |
| * overhead. `Variant` can accommodate all primitive types and |
| * all user-defined types.)) |
| * |
| * Both `Algebraic` and `Variant` share $(D |
| * VariantN)'s interface. (See their respective documentations below.) |
| * |
| * `VariantN` is a discriminated union type parameterized |
| * with the largest size of the types stored (`maxDataSize`) |
| * and with the list of allowed types (`AllowedTypes`). If |
| * the list is empty, then any type up of size up to $(D |
| * maxDataSize) (rounded up for alignment) can be stored in a |
| * `VariantN` object without being boxed (types larger |
| * than this will be boxed). |
| * |
| */ |
| struct VariantN(size_t maxDataSize, AllowedTypesParam...) |
| { |
| /** |
| The list of allowed types. If empty, any type is allowed. |
| */ |
| alias AllowedTypes = This2Variant!(VariantN, AllowedTypesParam); |
| |
| private: |
| // Compute the largest practical size from maxDataSize |
| struct SizeChecker |
| { |
| int function() fptr; |
| ubyte[maxDataSize] data; |
| } |
| enum size = SizeChecker.sizeof - (int function()).sizeof; |
| |
| /** Tells whether a type `T` is statically _allowed for |
| * storage inside a `VariantN` object by looking |
| * `T` up in `AllowedTypes`. |
| */ |
| public template allowed(T) |
| { |
| enum bool allowed |
| = is(T == VariantN) |
| || |
| //T.sizeof <= size && |
| (AllowedTypes.length == 0 || staticIndexOf!(T, AllowedTypes) >= 0); |
| } |
| |
| // Each internal operation is encoded with an identifier. See |
| // the "handler" function below. |
| enum OpID { getTypeInfo, get, compare, equals, testConversion, toString, |
| index, indexAssign, catAssign, copyOut, length, |
| apply, postblit, destruct } |
| |
| // state |
| union |
| { |
| align(maxVariantAlignment!(AllowedTypes)) ubyte[size] store; |
| // conservatively mark the region as pointers |
| static if (size >= (void*).sizeof) |
| void*[size / (void*).sizeof] p; |
| } |
| ptrdiff_t function(OpID selector, ubyte[size]* store, void* data) fptr |
| = &handler!(void); |
| |
| // internals |
| // Handler for an uninitialized value |
| static ptrdiff_t handler(A : void)(OpID selector, ubyte[size]*, void* parm) |
| { |
| switch (selector) |
| { |
| case OpID.getTypeInfo: |
| *cast(TypeInfo *) parm = typeid(A); |
| break; |
| case OpID.copyOut: |
| auto target = cast(VariantN *) parm; |
| target.fptr = &handler!(A); |
| // no need to copy the data (it's garbage) |
| break; |
| case OpID.compare: |
| case OpID.equals: |
| auto rhs = cast(const VariantN *) parm; |
| return rhs.peek!(A) |
| ? 0 // all uninitialized are equal |
| : ptrdiff_t.min; // uninitialized variant is not comparable otherwise |
| case OpID.toString: |
| string * target = cast(string*) parm; |
| *target = "<Uninitialized VariantN>"; |
| break; |
| case OpID.postblit: |
| case OpID.destruct: |
| break; |
| case OpID.get: |
| case OpID.testConversion: |
| case OpID.index: |
| case OpID.indexAssign: |
| case OpID.catAssign: |
| case OpID.length: |
| throw new VariantException( |
| "Attempt to use an uninitialized VariantN"); |
| default: assert(false, "Invalid OpID"); |
| } |
| return 0; |
| } |
| |
| // Handler for all of a type's operations |
| static ptrdiff_t handler(A)(OpID selector, ubyte[size]* pStore, void* parm) |
| { |
| import std.conv : to; |
| static A* getPtr(void* untyped) |
| { |
| if (untyped) |
| { |
| static if (A.sizeof <= size) |
| return cast(A*) untyped; |
| else |
| return *cast(A**) untyped; |
| } |
| return null; |
| } |
| |
| static ptrdiff_t compare(A* rhsPA, A* zis, OpID selector) |
| { |
| static if (is(typeof(*rhsPA == *zis))) |
| { |
| enum isEmptyStructWithoutOpEquals = is(A == struct) && A.tupleof.length == 0 && |
| !__traits(hasMember, A, "opEquals"); |
| static if (isEmptyStructWithoutOpEquals) |
| { |
| // The check above will always succeed if A is an empty struct. |
| // Don't generate unreachable code as seen in |
| // https://issues.dlang.org/show_bug.cgi?id=21231 |
| return 0; |
| } |
| else |
| { |
| if (*rhsPA == *zis) |
| return 0; |
| static if (is(typeof(*zis < *rhsPA))) |
| { |
| // Many types (such as any using the default Object opCmp) |
| // will throw on an invalid opCmp, so do it only |
| // if the caller requests it. |
| if (selector == OpID.compare) |
| return *zis < *rhsPA ? -1 : 1; |
| else |
| return ptrdiff_t.min; |
| } |
| else |
| { |
| // Not equal, and type does not support ordering |
| // comparisons. |
| return ptrdiff_t.min; |
| } |
| } |
| } |
| else |
| { |
| // Type does not support comparisons at all. |
| return ptrdiff_t.min; |
| } |
| } |
| |
| auto zis = getPtr(pStore); |
| // Input: TypeInfo object |
| // Output: target points to a copy of *me, if me was not null |
| // Returns: true iff the A can be converted to the type represented |
| // by the incoming TypeInfo |
| static bool tryPutting(A* src, TypeInfo targetType, void* target) |
| { |
| alias UA = Unqual!A; |
| static if (isStaticArray!A && is(typeof(UA.init[0]))) |
| { |
| alias MutaTypes = AliasSeq!(UA, typeof(UA.init[0])[], AllImplicitConversionTargets!UA); |
| } |
| else |
| { |
| alias MutaTypes = AliasSeq!(UA, AllImplicitConversionTargets!UA); |
| } |
| alias ConstTypes = staticMap!(ConstOf, MutaTypes); |
| alias SharedTypes = staticMap!(SharedOf, MutaTypes); |
| alias SharedConstTypes = staticMap!(SharedConstOf, MutaTypes); |
| alias ImmuTypes = staticMap!(ImmutableOf, MutaTypes); |
| |
| static if (is(A == immutable)) |
| alias AllTypes = AliasSeq!(ImmuTypes, ConstTypes, SharedConstTypes); |
| else static if (is(A == shared)) |
| { |
| static if (is(A == const)) |
| alias AllTypes = SharedConstTypes; |
| else |
| alias AllTypes = AliasSeq!(SharedTypes, SharedConstTypes); |
| } |
| else |
| { |
| static if (is(A == const)) |
| alias AllTypes = ConstTypes; |
| else |
| alias AllTypes = AliasSeq!(MutaTypes, ConstTypes); |
| } |
| |
| foreach (T ; AllTypes) |
| { |
| if (targetType != typeid(T)) |
| continue; |
| |
| // SPECIAL NOTE: variant only will ever create a new value with |
| // tryPutting (effectively), and T is ALWAYS the same type of |
| // A, but with different modifiers (and a limited set of |
| // implicit targets). So this checks to see if we can construct |
| // a T from A, knowing that prerequisite. This handles issues |
| // where the type contains some constant data aside from the |
| // modifiers on the type itself. |
| static if (is(typeof(delegate T() {return *src;})) || |
| is(T == const(U), U) || |
| is(T == shared(U), U) || |
| is(T == shared const(U), U) || |
| is(T == immutable(U), U)) |
| { |
| import core.internal.lifetime : emplaceRef; |
| |
| auto zat = cast(T*) target; |
| if (src) |
| { |
| static if (T.sizeof > 0) |
| assert(target, "target must be non-null"); |
| |
| static if (isStaticArray!A && isDynamicArray!T) |
| { |
| auto this_ = (*src)[]; |
| emplaceRef(*cast(Unqual!T*) zat, cast(Unqual!T) this_); |
| } |
| else |
| { |
| emplaceRef(*cast(Unqual!T*) zat, *cast(UA*) src); |
| } |
| } |
| } |
| else |
| { |
| // type T is not constructible from A |
| if (src) |
| assert(false, A.stringof); |
| } |
| return true; |
| } |
| return false; |
| } |
| |
| switch (selector) |
| { |
| case OpID.getTypeInfo: |
| *cast(TypeInfo *) parm = typeid(A); |
| break; |
| case OpID.copyOut: |
| auto target = cast(VariantN *) parm; |
| assert(target); |
| |
| static if (target.size < A.sizeof) |
| { |
| if (target.type.tsize < A.sizeof) |
| { |
| static if (is(A == U[n], U, size_t n)) |
| { |
| A* p = cast(A*)(new U[n]).ptr; |
| } |
| else |
| { |
| A* p = new A; |
| } |
| *cast(A**)&target.store = p; |
| } |
| } |
| tryPutting(zis, typeid(A), cast(void*) getPtr(&target.store)) |
| || assert(false); |
| target.fptr = &handler!(A); |
| break; |
| case OpID.get: |
| auto t = * cast(Tuple!(TypeInfo, void*)*) parm; |
| return !tryPutting(zis, t[0], t[1]); |
| case OpID.testConversion: |
| return !tryPutting(null, *cast(TypeInfo*) parm, null); |
| case OpID.compare: |
| case OpID.equals: |
| auto rhsP = cast(VariantN *) parm; |
| auto rhsType = rhsP.type; |
| // Are we the same? |
| if (rhsType == typeid(A)) |
| { |
| // cool! Same type! |
| auto rhsPA = getPtr(&rhsP.store); |
| return compare(rhsPA, zis, selector); |
| } |
| else if (rhsType == typeid(void)) |
| { |
| // No support for ordering comparisons with |
| // uninitialized vars |
| return ptrdiff_t.min; |
| } |
| VariantN temp; |
| // Do I convert to rhs? |
| if (tryPutting(zis, rhsType, &temp.store)) |
| { |
| // cool, I do; temp's store contains my data in rhs's type! |
| // also fix up its fptr |
| temp.fptr = rhsP.fptr; |
| // now lhsWithRhsType is a full-blown VariantN of rhs's type |
| if (selector == OpID.compare) |
| return temp.opCmp(*rhsP); |
| else |
| return temp.opEquals(*rhsP) ? 0 : 1; |
| } |
| // Does rhs convert to zis? |
| auto t = tuple(typeid(A), &temp.store); |
| if (rhsP.fptr(OpID.get, &rhsP.store, &t) == 0) |
| { |
| // cool! Now temp has rhs in my type! |
| auto rhsPA = getPtr(&temp.store); |
| return compare(rhsPA, zis, selector); |
| } |
| // Generate the function below only if the Variant's type is |
| // comparable with 'null' |
| static if (__traits(compiles, () => A.init == null)) |
| { |
| if (rhsType == typeid(null)) |
| { |
| // if rhsType is typeof(null), then we're comparing with 'null' |
| // this takes into account 'opEquals' and 'opCmp' |
| // all types that can compare with null have to following properties: |
| // if it's 'null' then it's equal to null, otherwise it's always greater |
| // than 'null' |
| return *zis == null ? 0 : 1; |
| } |
| } |
| return ptrdiff_t.min; // dunno |
| case OpID.toString: |
| auto target = cast(string*) parm; |
| static if (is(typeof(to!(string)(*zis)))) |
| { |
| *target = to!(string)(*zis); |
| break; |
| } |
| // TODO: The following test evaluates to true for shared objects. |
| // Use __traits for now until this is sorted out. |
| // else static if (is(typeof((*zis).toString))) |
| else static if (__traits(compiles, {(*zis).toString();})) |
| { |
| *target = (*zis).toString(); |
| break; |
| } |
| else |
| { |
| throw new VariantException(typeid(A), typeid(string)); |
| } |
| |
| case OpID.index: |
| auto result = cast(Variant*) parm; |
| static if (isArray!(A) && !is(immutable typeof(A.init[0]) == immutable void)) |
| { |
| // array type; input and output are the same VariantN |
| size_t index = result.convertsTo!(int) |
| ? result.get!(int) : result.get!(size_t); |
| *result = (*zis)[index]; |
| break; |
| } |
| else static if (isAssociativeArray!(A)) |
| { |
| *result = (*zis)[result.get!(typeof(A.init.keys[0]))]; |
| break; |
| } |
| else |
| { |
| throw new VariantException(typeid(A), result[0].type); |
| } |
| |
| case OpID.indexAssign: |
| // array type; result comes first, index comes second |
| auto args = cast(Variant*) parm; |
| static if (isArray!(A) && is(typeof((*zis)[0] = (*zis)[0]))) |
| { |
| size_t index = args[1].convertsTo!(int) |
| ? args[1].get!(int) : args[1].get!(size_t); |
| (*zis)[index] = args[0].get!(typeof((*zis)[0])); |
| break; |
| } |
| else static if (isAssociativeArray!(A) && is(typeof((*zis)[A.init.keys[0]] = A.init.values[0]))) |
| { |
| (*zis)[args[1].get!(typeof(A.init.keys[0]))] |
| = args[0].get!(typeof(A.init.values[0])); |
| break; |
| } |
| else |
| { |
| throw new VariantException(typeid(A), args[0].type); |
| } |
| |
| case OpID.catAssign: |
| static if (!is(immutable typeof((*zis)[0]) == immutable void) && |
| is(typeof((*zis)[0])) && is(typeof(*zis ~= *zis))) |
| { |
| // array type; parm is the element to append |
| auto arg = cast(Variant*) parm; |
| alias E = typeof((*zis)[0]); |
| if (arg[0].convertsTo!(E)) |
| { |
| // append one element to the array |
| (*zis) ~= [ arg[0].get!(E) ]; |
| } |
| else |
| { |
| // append a whole array to the array |
| (*zis) ~= arg[0].get!(A); |
| } |
| break; |
| } |
| else |
| { |
| throw new VariantException(typeid(A), typeid(void[])); |
| } |
| |
| case OpID.length: |
| static if (isArray!(A) || isAssociativeArray!(A)) |
| { |
| return zis.length; |
| } |
| else |
| { |
| throw new VariantException(typeid(A), typeid(void[])); |
| } |
| |
| case OpID.apply: |
| static if (!isFunctionPointer!A && !isDelegate!A) |
| { |
| import std.conv : text; |
| import std.exception : enforce; |
| enforce(0, text("Cannot apply `()' to a value of type `", |
| A.stringof, "'.")); |
| } |
| else |
| { |
| import std.conv : text; |
| import std.exception : enforce; |
| alias ParamTypes = Parameters!A; |
| auto p = cast(Variant*) parm; |
| auto argCount = p.get!size_t; |
| // To assign the tuple we need to use the unqualified version, |
| // otherwise we run into issues such as with const values. |
| // We still get the actual type from the Variant though |
| // to ensure that we retain const correctness. |
| Tuple!(staticMap!(Unqual, ParamTypes)) t; |
| enforce(t.length == argCount, |
| text("Argument count mismatch: ", |
| A.stringof, " expects ", t.length, |
| " argument(s), not ", argCount, ".")); |
| auto variantArgs = p[1 .. argCount + 1]; |
| foreach (i, T; ParamTypes) |
| { |
| t[i] = cast() variantArgs[i].get!T; |
| } |
| |
| auto args = cast(Tuple!(ParamTypes))t; |
| static if (is(ReturnType!A == void)) |
| { |
| (*zis)(args.expand); |
| *p = Variant.init; // void returns uninitialized Variant. |
| } |
| else |
| { |
| *p = (*zis)(args.expand); |
| } |
| } |
| break; |
| |
| case OpID.postblit: |
| static if (hasElaborateCopyConstructor!A) |
| { |
| zis.__xpostblit(); |
| } |
| break; |
| |
| case OpID.destruct: |
| static if (hasElaborateDestructor!A) |
| { |
| zis.__xdtor(); |
| } |
| break; |
| |
| default: assert(false); |
| } |
| return 0; |
| } |
| |
| public: |
| /** Constructs a `VariantN` value given an argument of a |
| * generic type. Statically rejects disallowed types. |
| */ |
| |
| this(T)(T value) |
| { |
| static assert(allowed!(T), "Cannot store a " ~ T.stringof |
| ~ " in a " ~ VariantN.stringof); |
| opAssign(value); |
| } |
| |
| /// Allows assignment from a subset algebraic type |
| this(T : VariantN!(tsize, Types), size_t tsize, Types...)(T value) |
| if (!is(T : VariantN) && Types.length > 0 && allSatisfy!(allowed, Types)) |
| { |
| opAssign(value); |
| } |
| |
| static if (!AllowedTypes.length || anySatisfy!(hasElaborateCopyConstructor, AllowedTypes)) |
| { |
| this(this) |
| { |
| fptr(OpID.postblit, &store, null); |
| } |
| } |
| |
| static if (!AllowedTypes.length || anySatisfy!(hasElaborateDestructor, AllowedTypes)) |
| { |
| ~this() |
| { |
| // Infer the safety of the provided types |
| static if (AllowedTypes.length) |
| { |
| if (0) |
| { |
| AllowedTypes var; |
| } |
| } |
| (() @trusted => fptr(OpID.destruct, &store, null))(); |
| } |
| } |
| |
| /** Assigns a `VariantN` from a generic |
| * argument. Statically rejects disallowed types. */ |
| |
| VariantN opAssign(T)(T rhs) |
| { |
| static assert(allowed!(T), "Cannot store a " ~ T.stringof |
| ~ " in a " ~ VariantN.stringof ~ ". Valid types are " |
| ~ AllowedTypes.stringof); |
| |
| static if (is(T : VariantN)) |
| { |
| rhs.fptr(OpID.copyOut, &rhs.store, &this); |
| } |
| else static if (is(T : const(VariantN))) |
| { |
| static assert(false, |
| "Assigning Variant objects from const Variant"~ |
| " objects is currently not supported."); |
| } |
| else |
| { |
| import core.lifetime : copyEmplace; |
| |
| static if (!AllowedTypes.length || anySatisfy!(hasElaborateDestructor, AllowedTypes)) |
| { |
| // Assignment should destruct previous value |
| fptr(OpID.destruct, &store, null); |
| } |
| |
| static if (T.sizeof <= size) |
| copyEmplace(rhs, *cast(T*) &store); |
| else |
| { |
| static if (is(T == U[n], U, size_t n)) |
| auto p = cast(T*) (new U[n]).ptr; |
| else |
| auto p = new T; |
| copyEmplace(rhs, *p); |
| *(cast(T**) &store) = p; |
| } |
| |
| fptr = &handler!(T); |
| } |
| return this; |
| } |
| |
| // Allow assignment from another variant which is a subset of this one |
| VariantN opAssign(T : VariantN!(tsize, Types), size_t tsize, Types...)(T rhs) |
| if (!is(T : VariantN) && Types.length > 0 && allSatisfy!(allowed, Types)) |
| { |
| // discover which type rhs is actually storing |
| foreach (V; T.AllowedTypes) |
| if (rhs.type == typeid(V)) |
| return this = rhs.get!V; |
| assert(0, T.AllowedTypes.stringof); |
| } |
| |
| |
| Variant opCall(P...)(auto ref P params) |
| { |
| Variant[P.length + 1] pack; |
| pack[0] = P.length; |
| foreach (i, _; params) |
| { |
| pack[i + 1] = params[i]; |
| } |
| fptr(OpID.apply, &store, &pack); |
| return pack[0]; |
| } |
| |
| /** Returns true if and only if the `VariantN` object |
| * holds a valid value (has been initialized with, or assigned |
| * from, a valid value). |
| */ |
| @property bool hasValue() const pure nothrow |
| { |
| // @@@BUG@@@ in compiler, the cast shouldn't be needed |
| return cast(typeof(&handler!(void))) fptr != &handler!(void); |
| } |
| |
| /// |
| version (StdDdoc) |
| @system unittest |
| { |
| Variant a; |
| assert(!a.hasValue); |
| Variant b; |
| a = b; |
| assert(!a.hasValue); // still no value |
| a = 5; |
| assert(a.hasValue); |
| } |
| |
| /** |
| * If the `VariantN` object holds a value of the |
| * $(I exact) type `T`, returns a pointer to that |
| * value. Otherwise, returns `null`. In cases |
| * where `T` is statically disallowed, $(D |
| * peek) will not compile. |
| */ |
| @property inout(T)* peek(T)() inout |
| { |
| static if (!is(T == void)) |
| static assert(allowed!(T), "Cannot store a " ~ T.stringof |
| ~ " in a " ~ VariantN.stringof); |
| if (type != typeid(T)) |
| return null; |
| static if (T.sizeof <= size) |
| return cast(inout T*)&store; |
| else |
| return *cast(inout T**)&store; |
| } |
| |
| /// |
| version (StdDdoc) |
| @system unittest |
| { |
| Variant a = 5; |
| auto b = a.peek!(int); |
| assert(b !is null); |
| *b = 6; |
| assert(a == 6); |
| } |
| |
| /** |
| * Returns the `typeid` of the currently held value. |
| */ |
| |
| @property TypeInfo type() const nothrow @trusted |
| { |
| scope(failure) assert(0); |
| |
| TypeInfo result; |
| fptr(OpID.getTypeInfo, null, &result); |
| return result; |
| } |
| |
| /** |
| * Returns `true` if and only if the `VariantN` |
| * object holds an object implicitly convertible to type `T`. |
| * Implicit convertibility is defined as per |
| * $(REF_ALTTEXT AllImplicitConversionTargets, AllImplicitConversionTargets, std,traits). |
| */ |
| |
| @property bool convertsTo(T)() const |
| { |
| TypeInfo info = typeid(T); |
| return fptr(OpID.testConversion, null, &info) == 0; |
| } |
| |
| /** |
| Returns the value stored in the `VariantN` object, either by specifying the |
| needed type or the index in the list of allowed types. The latter overload |
| only applies to bounded variants (e.g. $(LREF Algebraic)). |
| |
| Params: |
| T = The requested type. The currently stored value must implicitly convert |
| to the requested type, in fact `DecayStaticToDynamicArray!T`. If an |
| implicit conversion is not possible, throws a `VariantException`. |
| index = The index of the type among `AllowedTypesParam`, zero-based. |
| */ |
| @property inout(T) get(T)() inout |
| { |
| inout(T) result = void; |
| static if (is(T == shared)) |
| alias R = shared Unqual!T; |
| else |
| alias R = Unqual!T; |
| auto buf = tuple(typeid(T), cast(R*)&result); |
| |
| if (fptr(OpID.get, cast(ubyte[size]*) &store, &buf)) |
| { |
| throw new VariantException(type, typeid(T)); |
| } |
| return result; |
| } |
| |
| /// Ditto |
| @property auto get(uint index)() inout |
| if (index < AllowedTypes.length) |
| { |
| foreach (i, T; AllowedTypes) |
| { |
| static if (index == i) return get!T; |
| } |
| assert(0); |
| } |
| |
| /** |
| * Returns the value stored in the `VariantN` object, |
| * explicitly converted (coerced) to the requested type $(D |
| * T). If `T` is a string type, the value is formatted as |
| * a string. If the `VariantN` object is a string, a |
| * parse of the string to type `T` is attempted. If a |
| * conversion is not possible, throws a $(D |
| * VariantException). |
| */ |
| |
| @property T coerce(T)() |
| { |
| import std.conv : to, text; |
| static if (isNumeric!T || isBoolean!T) |
| { |
| if (convertsTo!real) |
| { |
| // maybe optimize this fella; handle ints separately |
| return to!T(get!real); |
| } |
| else if (convertsTo!(const(char)[])) |
| { |
| return to!T(get!(const(char)[])); |
| } |
| // I'm not sure why this doesn't convert to const(char), |
| // but apparently it doesn't (probably a deeper bug). |
| // |
| // Until that is fixed, this quick addition keeps a common |
| // function working. "10".coerce!int ought to work. |
| else if (convertsTo!(immutable(char)[])) |
| { |
| return to!T(get!(immutable(char)[])); |
| } |
| else |
| { |
| import std.exception : enforce; |
| enforce(false, text("Type ", type, " does not convert to ", |
| typeid(T))); |
| assert(0); |
| } |
| } |
| else static if (is(T : Object)) |
| { |
| return to!(T)(get!(Object)); |
| } |
| else static if (isSomeString!(T)) |
| { |
| return to!(T)(toString()); |
| } |
| else |
| { |
| // Fix for bug 1649 |
| static assert(false, "unsupported type for coercion"); |
| } |
| } |
| |
| /** |
| * Formats the stored value as a string. |
| */ |
| |
| string toString() |
| { |
| string result; |
| fptr(OpID.toString, &store, &result) == 0 || assert(false); |
| return result; |
| } |
| |
| /** |
| * Comparison for equality used by the "==" and "!=" operators. |
| */ |
| |
| // returns 1 if the two are equal |
| bool opEquals(T)(auto ref T rhs) const |
| if (allowed!T || is(immutable T == immutable VariantN)) |
| { |
| static if (is(immutable T == immutable VariantN)) |
| alias temp = rhs; |
| else |
| auto temp = VariantN(rhs); |
| return !fptr(OpID.equals, cast(ubyte[size]*) &store, |
| cast(void*) &temp); |
| } |
| |
| // workaround for bug 10567 fix |
| int opCmp(ref const VariantN rhs) const |
| { |
| return (cast() this).opCmp!(VariantN)(cast() rhs); |
| } |
| |
| /** |
| * Ordering comparison used by the "<", "<=", ">", and ">=" |
| * operators. In case comparison is not sensible between the held |
| * value and `rhs`, an exception is thrown. |
| */ |
| |
| int opCmp(T)(T rhs) |
| if (allowed!T) // includes T == VariantN |
| { |
| static if (is(T == VariantN)) |
| alias temp = rhs; |
| else |
| auto temp = VariantN(rhs); |
| auto result = fptr(OpID.compare, &store, &temp); |
| if (result == ptrdiff_t.min) |
| { |
| throw new VariantException(type, temp.type); |
| } |
| |
| assert(result >= -1 && result <= 1); // Should be true for opCmp. |
| return cast(int) result; |
| } |
| |
| /** |
| * Computes the hash of the held value. |
| */ |
| |
| size_t toHash() const nothrow @safe |
| { |
| return type.getHash(&store); |
| } |
| |
| private VariantN opArithmetic(T, string op)(T other) |
| { |
| static if (isInstanceOf!(.VariantN, T)) |
| { |
| string tryUseType(string tp) |
| { |
| import std.format : format; |
| return q{ |
| static if (allowed!%1$s && T.allowed!%1$s) |
| if (convertsTo!%1$s && other.convertsTo!%1$s) |
| return VariantN(get!%1$s %2$s other.get!%1$s); |
| }.format(tp, op); |
| } |
| |
| mixin(tryUseType("uint")); |
| mixin(tryUseType("int")); |
| mixin(tryUseType("ulong")); |
| mixin(tryUseType("long")); |
| mixin(tryUseType("float")); |
| mixin(tryUseType("double")); |
| mixin(tryUseType("real")); |
| } |
| else |
| { |
| static if (allowed!T) |
| if (auto pv = peek!T) return VariantN(mixin("*pv " ~ op ~ " other")); |
| static if (allowed!uint && is(typeof(T.max) : uint) && isUnsigned!T) |
| if (convertsTo!uint) return VariantN(mixin("get!(uint) " ~ op ~ " other")); |
| static if (allowed!int && is(typeof(T.max) : int) && !isUnsigned!T) |
| if (convertsTo!int) return VariantN(mixin("get!(int) " ~ op ~ " other")); |
| static if (allowed!ulong && is(typeof(T.max) : ulong) && isUnsigned!T) |
| if (convertsTo!ulong) return VariantN(mixin("get!(ulong) " ~ op ~ " other")); |
| static if (allowed!long && is(typeof(T.max) : long) && !isUnsigned!T) |
| if (convertsTo!long) return VariantN(mixin("get!(long) " ~ op ~ " other")); |
| static if (allowed!float && is(T : float)) |
| if (convertsTo!float) return VariantN(mixin("get!(float) " ~ op ~ " other")); |
| static if (allowed!double && is(T : double)) |
| if (convertsTo!double) return VariantN(mixin("get!(double) " ~ op ~ " other")); |
| static if (allowed!real && is (T : real)) |
| if (convertsTo!real) return VariantN(mixin("get!(real) " ~ op ~ " other")); |
| } |
| |
| throw new VariantException("No possible match found for VariantN "~op~" "~T.stringof); |
| } |
| |
| private VariantN opLogic(T, string op)(T other) |
| { |
| VariantN result; |
| static if (is(T == VariantN)) |
| { |
| if (convertsTo!(uint) && other.convertsTo!(uint)) |
| result = mixin("get!(uint) " ~ op ~ " other.get!(uint)"); |
| else if (convertsTo!(int) && other.convertsTo!(int)) |
| result = mixin("get!(int) " ~ op ~ " other.get!(int)"); |
| else if (convertsTo!(ulong) && other.convertsTo!(ulong)) |
| result = mixin("get!(ulong) " ~ op ~ " other.get!(ulong)"); |
| else |
| result = mixin("get!(long) " ~ op ~ " other.get!(long)"); |
| } |
| else |
| { |
| if (is(typeof(T.max) : uint) && T.min == 0 && convertsTo!(uint)) |
| result = mixin("get!(uint) " ~ op ~ " other"); |
| else if (is(typeof(T.max) : int) && T.min < 0 && convertsTo!(int)) |
| result = mixin("get!(int) " ~ op ~ " other"); |
| else if (is(typeof(T.max) : ulong) && T.min == 0 |
| && convertsTo!(ulong)) |
| result = mixin("get!(ulong) " ~ op ~ " other"); |
| else |
| result = mixin("get!(long) " ~ op ~ " other"); |
| } |
| return result; |
| } |
| |
| /** |
| * Arithmetic between `VariantN` objects and numeric |
| * values. All arithmetic operations return a `VariantN` |
| * object typed depending on the types of both values |
| * involved. The conversion rules mimic D's built-in rules for |
| * arithmetic conversions. |
| */ |
| VariantN opBinary(string op, T)(T rhs) |
| if ((op == "+" || op == "-" || op == "*" || op == "/" || op == "^^" || op == "%") && |
| is(typeof(opArithmetic!(T, op)(rhs)))) |
| { return opArithmetic!(T, op)(rhs); } |
| ///ditto |
| VariantN opBinary(string op, T)(T rhs) |
| if ((op == "&" || op == "|" || op == "^" || op == ">>" || op == "<<" || op == ">>>") && |
| is(typeof(opLogic!(T, op)(rhs)))) |
| { return opLogic!(T, op)(rhs); } |
| ///ditto |
| VariantN opBinaryRight(string op, T)(T lhs) |
| if ((op == "+" || op == "*") && |
| is(typeof(opArithmetic!(T, op)(lhs)))) |
| { return opArithmetic!(T, op)(lhs); } |
| ///ditto |
| VariantN opBinaryRight(string op, T)(T lhs) |
| if ((op == "&" || op == "|" || op == "^") && |
| is(typeof(opLogic!(T, op)(lhs)))) |
| { return opLogic!(T, op)(lhs); } |
| ///ditto |
| VariantN opBinary(string op, T)(T rhs) |
| if (op == "~") |
| { |
| auto temp = this; |
| temp ~= rhs; |
| return temp; |
| } |
| // ///ditto |
| // VariantN opBinaryRight(string op, T)(T rhs) |
| // if (op == "~") |
| // { |
| // VariantN temp = rhs; |
| // temp ~= this; |
| // return temp; |
| // } |
| |
| ///ditto |
| VariantN opOpAssign(string op, T)(T rhs) |
| { |
| static if (op != "~") |
| { |
| mixin("return this = this" ~ op ~ "rhs;"); |
| } |
| else |
| { |
| auto toAppend = Variant(rhs); |
| fptr(OpID.catAssign, &store, &toAppend) == 0 || assert(false); |
| return this; |
| } |
| } |
| |
| /** |
| * Array and associative array operations. If a $(D |
| * VariantN) contains an (associative) array, it can be indexed |
| * into. Otherwise, an exception is thrown. |
| */ |
| inout(Variant) opIndex(K)(K i) inout |
| { |
| auto result = Variant(i); |
| fptr(OpID.index, cast(ubyte[size]*) &store, &result) == 0 || assert(false); |
| return result; |
| } |
| |
| /// |
| version (StdDdoc) |
| @system unittest |
| { |
| Variant a = new int[10]; |
| a[5] = 42; |
| assert(a[5] == 42); |
| a[5] += 8; |
| assert(a[5] == 50); |
| |
| int[int] hash = [ 42:24 ]; |
| a = hash; |
| assert(a[42] == 24); |
| a[42] /= 2; |
| assert(a[42] == 12); |
| } |
| |
| /// ditto |
| Variant opIndexAssign(T, N)(T value, N i) |
| { |
| static if (AllowedTypes.length && !isInstanceOf!(.VariantN, T)) |
| { |
| enum canAssign(U) = __traits(compiles, (U u){ u[i] = value; }); |
| static assert(anySatisfy!(canAssign, AllowedTypes), |
| "Cannot assign " ~ T.stringof ~ " to " ~ VariantN.stringof ~ |
| " indexed with " ~ N.stringof); |
| } |
| Variant[2] args = [ Variant(value), Variant(i) ]; |
| fptr(OpID.indexAssign, &store, &args) == 0 || assert(false); |
| return args[0]; |
| } |
| |
| /// ditto |
| Variant opIndexOpAssign(string op, T, N)(T value, N i) |
| { |
| return opIndexAssign(mixin(`opIndex(i)` ~ op ~ `value`), i); |
| } |
| |
| /** If the `VariantN` contains an (associative) array, |
| * returns the _length of that array. Otherwise, throws an |
| * exception. |
| */ |
| @property size_t length() |
| { |
| return cast(size_t) fptr(OpID.length, &store, null); |
| } |
| |
| /** |
| If the `VariantN` contains an array, applies `dg` to each |
| element of the array in turn. Otherwise, throws an exception. |
| */ |
| int opApply(Delegate)(scope Delegate dg) if (is(Delegate == delegate)) |
| { |
| alias A = Parameters!(Delegate)[0]; |
| if (type == typeid(A[])) |
| { |
| auto arr = get!(A[]); |
| foreach (ref e; arr) |
| { |
| if (dg(e)) return 1; |
| } |
| } |
| else static if (is(A == VariantN)) |
| { |
| foreach (i; 0 .. length) |
| { |
| // @@@TODO@@@: find a better way to not confuse |
| // clients who think they change values stored in the |
| // Variant when in fact they are only changing tmp. |
| auto tmp = this[i]; |
| debug scope(exit) assert(tmp == this[i]); |
| if (dg(tmp)) return 1; |
| } |
| } |
| else |
| { |
| import std.conv : text; |
| import std.exception : enforce; |
| enforce(false, text("Variant type ", type, |
| " not iterable with values of type ", |
| A.stringof)); |
| } |
| return 0; |
| } |
| } |
| |
| /// |
| @system unittest |
| { |
| alias Var = VariantN!(maxSize!(int, double, string)); |
| |
| Var a; // Must assign before use, otherwise exception ensues |
| // Initialize with an integer; make the type int |
| Var b = 42; |
| assert(b.type == typeid(int)); |
| // Peek at the value |
| assert(b.peek!(int) !is null && *b.peek!(int) == 42); |
| // Automatically convert per language rules |
| auto x = b.get!(real); |
| |
| // Assign any other type, including other variants |
| a = b; |
| a = 3.14; |
| assert(a.type == typeid(double)); |
| // Implicit conversions work just as with built-in types |
| assert(a < b); |
| // Check for convertibility |
| assert(!a.convertsTo!(int)); // double not convertible to int |
| // Strings and all other arrays are supported |
| a = "now I'm a string"; |
| assert(a == "now I'm a string"); |
| } |
| |
| /// can also assign arrays |
| @system unittest |
| { |
| alias Var = VariantN!(maxSize!(int[])); |
| |
| Var a = new int[42]; |
| assert(a.length == 42); |
| a[5] = 7; |
| assert(a[5] == 7); |
| } |
| |
| @safe unittest |
| { |
| alias V = VariantN!24; |
| const alignMask = V.alignof - 1; |
| assert(V.sizeof == ((24 + (void*).sizeof + alignMask) & ~alignMask)); |
| } |
| |
| /// Can also assign class values |
| @system unittest |
| { |
| alias Var = VariantN!(maxSize!(int*)); // classes are pointers |
| Var a; |
| |
| class Foo {} |
| auto foo = new Foo; |
| a = foo; |
| assert(*a.peek!(Foo) == foo); // and full type information is preserved |
| } |
| |
| @system unittest |
| { |
| import std.conv : to; |
| Variant v; |
| int foo() { return 42; } |
| v = &foo; |
| assert(v() == 42); |
| |
| static int bar(string s) { return to!int(s); } |
| v = &bar; |
| assert(v("43") == 43); |
| } |
| |
| @system unittest |
| { |
| int[int] hash = [ 42:24 ]; |
| Variant v = hash; |
| assert(v[42] == 24); |
| v[42] = 5; |
| assert(v[42] == 5); |
| } |
| |
| // opIndex with static arrays, https://issues.dlang.org/show_bug.cgi?id=12771 |
| @system unittest |
| { |
| int[4] elements = [0, 1, 2, 3]; |
| Variant v = elements; |
| assert(v == elements); |
| assert(v[2] == 2); |
| assert(v[3] == 3); |
| v[2] = 6; |
| assert(v[2] == 6); |
| assert(v != elements); |
| } |
| |
| @system unittest |
| { |
| import std.exception : assertThrown; |
| Algebraic!(int[]) v = [2, 2]; |
| |
| assert(v == [2, 2]); |
| v[0] = 1; |
| assert(v[0] == 1); |
| assert(v != [2, 2]); |
| |
| // opIndexAssign from Variant |
| v[1] = v[0]; |
| assert(v[1] == 1); |
| |
| static assert(!__traits(compiles, (v[1] = null))); |
| assertThrown!VariantException(v[1] = Variant(null)); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=10879 |
| @system unittest |
| { |
| int[10] arr = [1,2,3,4,5,6,7,8,9,10]; |
| Variant v1 = arr; |
| Variant v2; |
| v2 = arr; |
| assert(v1 == arr); |
| assert(v2 == arr); |
| foreach (i, e; arr) |
| { |
| assert(v1[i] == e); |
| assert(v2[i] == e); |
| } |
| static struct LargeStruct |
| { |
| int[100] data; |
| } |
| LargeStruct ls; |
| ls.data[] = 4; |
| v1 = ls; |
| Variant v3 = ls; |
| assert(v1 == ls); |
| assert(v3 == ls); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=8195 |
| @system unittest |
| { |
| struct S |
| { |
| int a; |
| long b; |
| string c; |
| real d = 0.0; |
| bool e; |
| } |
| |
| static assert(S.sizeof >= Variant.sizeof); |
| alias Types = AliasSeq!(string, int, S); |
| alias MyVariant = VariantN!(maxSize!Types, Types); |
| |
| auto v = MyVariant(S.init); |
| assert(v == S.init); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=10961 |
| @system unittest |
| { |
| // Primarily test that we can assign a void[] to a Variant. |
| void[] elements = cast(void[])[1, 2, 3]; |
| Variant v = elements; |
| void[] returned = v.get!(void[]); |
| assert(returned == elements); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=13352 |
| @system unittest |
| { |
| alias TP = Algebraic!(long); |
| auto a = TP(1L); |
| auto b = TP(2L); |
| assert(!TP.allowed!ulong); |
| assert(a + b == 3L); |
| assert(a + 2 == 3L); |
| assert(1 + b == 3L); |
| |
| alias TP2 = Algebraic!(long, string); |
| auto c = TP2(3L); |
| assert(a + c == 4L); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=13354 |
| @system unittest |
| { |
| alias A = Algebraic!(string[]); |
| A a = ["a", "b"]; |
| assert(a[0] == "a"); |
| assert(a[1] == "b"); |
| a[1] = "c"; |
| assert(a[1] == "c"); |
| |
| alias AA = Algebraic!(int[string]); |
| AA aa = ["a": 1, "b": 2]; |
| assert(aa["a"] == 1); |
| assert(aa["b"] == 2); |
| aa["b"] = 3; |
| assert(aa["b"] == 3); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=14198 |
| @system unittest |
| { |
| Variant a = true; |
| assert(a.type == typeid(bool)); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=14233 |
| @system unittest |
| { |
| alias Atom = Algebraic!(string, This[]); |
| |
| Atom[] values = []; |
| auto a = Atom(values); |
| } |
| |
| pure nothrow @nogc |
| @system unittest |
| { |
| Algebraic!(int, double) a; |
| a = 100; |
| a = 1.0; |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=14457 |
| @system unittest |
| { |
| alias A = Algebraic!(int, float, double); |
| alias B = Algebraic!(int, float); |
| |
| A a = 1; |
| B b = 6f; |
| a = b; |
| |
| assert(a.type == typeid(float)); |
| assert(a.get!float == 6f); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=14585 |
| @system unittest |
| { |
| static struct S |
| { |
| int x = 42; |
| ~this() {assert(x == 42);} |
| } |
| Variant(S()).get!S; |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=14586 |
| @system unittest |
| { |
| const Variant v = new immutable Object; |
| v.get!(immutable Object); |
| } |
| |
| @system unittest |
| { |
| static struct S |
| { |
| T opCast(T)() {assert(false);} |
| } |
| Variant v = S(); |
| v.get!S; |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=13262 |
| @system unittest |
| { |
| static void fun(T)(Variant v){ |
| T x; |
| v = x; |
| auto r = v.get!(T); |
| } |
| Variant v; |
| fun!(shared(int))(v); |
| fun!(shared(int)[])(v); |
| |
| static struct S1 |
| { |
| int c; |
| string a; |
| } |
| |
| static struct S2 |
| { |
| string a; |
| shared int[] b; |
| } |
| |
| static struct S3 |
| { |
| string a; |
| shared int[] b; |
| int c; |
| } |
| |
| fun!(S1)(v); |
| fun!(shared(S1))(v); |
| fun!(S2)(v); |
| fun!(shared(S2))(v); |
| fun!(S3)(v); |
| fun!(shared(S3))(v); |
| |
| // ensure structs that are shared, but don't have shared postblits |
| // can't be used. |
| static struct S4 |
| { |
| int x; |
| this(this) {x = 0;} |
| } |
| |
| fun!(S4)(v); |
| static assert(!is(typeof(fun!(shared(S4))(v)))); |
| } |
| |
| @safe unittest |
| { |
| Algebraic!(int) x; |
| |
| static struct SafeS |
| { |
| @safe ~this() {} |
| } |
| |
| Algebraic!(SafeS) y; |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=19986 |
| @system unittest |
| { |
| VariantN!32 v; |
| v = const(ubyte[33]).init; |
| |
| struct S |
| { |
| ubyte[33] s; |
| } |
| |
| VariantN!32 v2; |
| v2 = const(S).init; |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=21021 |
| @system unittest |
| { |
| static struct S |
| { |
| int h; |
| int[5] array; |
| alias h this; |
| } |
| |
| S msg; |
| msg.array[] = 3; |
| Variant a = msg; |
| auto other = a.get!S; |
| assert(msg.array[0] == 3); |
| assert(other.array[0] == 3); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=21231 |
| // Compatibility with -preview=fieldwise |
| @system unittest |
| { |
| static struct Empty |
| { |
| bool opCmp(const scope ref Empty) const |
| { return false; } |
| } |
| |
| Empty a, b; |
| assert(a == b); |
| assert(!(a < b)); |
| |
| VariantN!(4, Empty) v = a; |
| assert(v == b); |
| assert(!(v < b)); |
| } |
| |
| // Compatibility with -preview=fieldwise |
| @system unittest |
| { |
| static struct Empty |
| { |
| bool opEquals(const scope ref Empty) const |
| { return false; } |
| } |
| |
| Empty a, b; |
| assert(a != b); |
| |
| VariantN!(4, Empty) v = a; |
| assert(v != b); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=22647 |
| // Can compare with 'null' |
| @system unittest |
| { |
| static struct Bar |
| { |
| int* ptr; |
| alias ptr this; |
| } |
| |
| static class Foo {} |
| int* iptr; |
| int[] arr; |
| |
| Variant v = Foo.init; // 'null' |
| assert(v != null); // can only compare objects with 'null' by using 'is' |
| |
| v = iptr; |
| assert(v == null); // pointers can be compared with 'null' |
| |
| v = arr; |
| assert(v == null); // arrays can be compared with 'null' |
| |
| v = ""; |
| assert(v == null); // strings are arrays, an empty string is considered 'null' |
| |
| v = Bar.init; |
| assert(v == null); // works with alias this |
| |
| v = [3]; |
| assert(v != null); |
| assert(v > null); |
| assert(v >= null); |
| assert(!(v < null)); |
| } |
| |
| /** |
| _Algebraic data type restricted to a closed set of possible |
| types. It's an alias for $(LREF VariantN) with an |
| appropriately-constructed maximum size. `Algebraic` is |
| useful when it is desirable to restrict what a discriminated type |
| could hold to the end of defining simpler and more efficient |
| manipulation. |
| |
| $(RED Warning: $(LREF Algebraic) is outdated and not recommended for use in new |
| code. Instead, use $(REF SumType, std,sumtype).) |
| */ |
| template Algebraic(T...) |
| { |
| alias Algebraic = VariantN!(maxSize!T, T); |
| } |
| |
| /// |
| @system unittest |
| { |
| auto v = Algebraic!(int, double, string)(5); |
| assert(v.peek!(int)); |
| v = 3.14; |
| assert(v.peek!(double)); |
| // auto x = v.peek!(long); // won't compile, type long not allowed |
| // v = '1'; // won't compile, type char not allowed |
| } |
| |
| /** |
| $(H4 Self-Referential Types) |
| |
| A useful and popular use of algebraic data structures is for defining $(LUCKY |
| self-referential data structures), i.e. structures that embed references to |
| values of their own type within. |
| |
| This is achieved with `Algebraic` by using `This` as a placeholder whenever a |
| reference to the type being defined is needed. The `Algebraic` instantiation |
| will perform $(LINK2 https://en.wikipedia.org/wiki/Name_resolution_(programming_languages)#Alpha_renaming_to_make_name_resolution_trivial, |
| alpha renaming) on its constituent types, replacing `This` |
| with the self-referenced type. The structure of the type involving `This` may |
| be arbitrarily complex. |
| */ |
| @system unittest |
| { |
| import std.typecons : Tuple, tuple; |
| |
| // A tree is either a leaf or a branch of two other trees |
| alias Tree(Leaf) = Algebraic!(Leaf, Tuple!(This*, This*)); |
| Tree!int tree = tuple(new Tree!int(42), new Tree!int(43)); |
| Tree!int* right = tree.get!1[1]; |
| assert(*right == 43); |
| |
| // An object is a double, a string, or a hash of objects |
| alias Obj = Algebraic!(double, string, This[string]); |
| Obj obj = "hello"; |
| assert(obj.get!1 == "hello"); |
| obj = 42.0; |
| assert(obj.get!0 == 42); |
| obj = ["customer": Obj("John"), "paid": Obj(23.95)]; |
| assert(obj.get!2["customer"] == "John"); |
| } |
| |
| private struct FakeComplexReal |
| { |
| real re, im; |
| } |
| |
| /** |
| Alias for $(LREF VariantN) instantiated with the largest size of `creal`, |
| `char[]`, and `void delegate()`. This ensures that `Variant` is large enough |
| to hold all of D's predefined types unboxed, including all numeric types, |
| pointers, delegates, and class references. You may want to use |
| `VariantN` directly with a different maximum size either for |
| storing larger types unboxed, or for saving memory. |
| */ |
| alias Variant = VariantN!(maxSize!(FakeComplexReal, char[], void delegate())); |
| |
| /// |
| @system unittest |
| { |
| Variant a; // Must assign before use, otherwise exception ensues |
| // Initialize with an integer; make the type int |
| Variant b = 42; |
| assert(b.type == typeid(int)); |
| // Peek at the value |
| assert(b.peek!(int) !is null && *b.peek!(int) == 42); |
| // Automatically convert per language rules |
| auto x = b.get!(real); |
| |
| // Assign any other type, including other variants |
| a = b; |
| a = 3.14; |
| assert(a.type == typeid(double)); |
| // Implicit conversions work just as with built-in types |
| assert(a < b); |
| // Check for convertibility |
| assert(!a.convertsTo!(int)); // double not convertible to int |
| // Strings and all other arrays are supported |
| a = "now I'm a string"; |
| assert(a == "now I'm a string"); |
| } |
| |
| /// can also assign arrays |
| @system unittest |
| { |
| Variant a = new int[42]; |
| assert(a.length == 42); |
| a[5] = 7; |
| assert(a[5] == 7); |
| } |
| |
| /// Can also assign class values |
| @system unittest |
| { |
| Variant a; |
| |
| class Foo {} |
| auto foo = new Foo; |
| a = foo; |
| assert(*a.peek!(Foo) == foo); // and full type information is preserved |
| } |
| |
| /** |
| * Returns an array of variants constructed from `args`. |
| * |
| * This is by design. During construction the `Variant` needs |
| * static type information about the type being held, so as to store a |
| * pointer to function for fast retrieval. |
| */ |
| Variant[] variantArray(T...)(T args) |
| { |
| Variant[] result; |
| foreach (arg; args) |
| { |
| result ~= Variant(arg); |
| } |
| return result; |
| } |
| |
| /// |
| @system unittest |
| { |
| auto a = variantArray(1, 3.14, "Hi!"); |
| assert(a[1] == 3.14); |
| auto b = Variant(a); // variant array as variant |
| assert(b[1] == 3.14); |
| } |
| |
| /** |
| * Thrown in three cases: |
| * |
| * $(OL $(LI An uninitialized `Variant` is used in any way except |
| * assignment and `hasValue`;) $(LI A `get` or |
| * `coerce` is attempted with an incompatible target type;) |
| * $(LI A comparison between `Variant` objects of |
| * incompatible types is attempted.)) |
| * |
| */ |
| |
| // @@@ BUG IN COMPILER. THE 'STATIC' BELOW SHOULD NOT COMPILE |
| static class VariantException : Exception |
| { |
| /// The source type in the conversion or comparison |
| TypeInfo source; |
| /// The target type in the conversion or comparison |
| TypeInfo target; |
| this(string s) |
| { |
| super(s); |
| } |
| this(TypeInfo source, TypeInfo target) |
| { |
| super("Variant: attempting to use incompatible types " |
| ~ source.toString() |
| ~ " and " ~ target.toString()); |
| this.source = source; |
| this.target = target; |
| } |
| } |
| |
| /// |
| @system unittest |
| { |
| import std.exception : assertThrown; |
| |
| Variant v; |
| |
| // uninitialized use |
| assertThrown!VariantException(v + 1); |
| assertThrown!VariantException(v.length); |
| |
| // .get with an incompatible target type |
| assertThrown!VariantException(Variant("a").get!int); |
| |
| // comparison between incompatible types |
| assertThrown!VariantException(Variant(3) < Variant("a")); |
| } |
| |
| @system unittest |
| { |
| alias W1 = This2Variant!(char, int, This[int]); |
| alias W2 = AliasSeq!(int, char[int]); |
| static assert(is(W1 == W2)); |
| |
| alias var_t = Algebraic!(void, string); |
| var_t foo = "quux"; |
| } |
| |
| @system unittest |
| { |
| alias A = Algebraic!(real, This[], This[int], This[This]); |
| A v1, v2, v3; |
| v2 = 5.0L; |
| v3 = 42.0L; |
| //v1 = [ v2 ][]; |
| auto v = v1.peek!(A[]); |
| //writeln(v[0]); |
| v1 = [ 9 : v3 ]; |
| //writeln(v1); |
| v1 = [ v3 : v3 ]; |
| //writeln(v1); |
| } |
| |
| @system unittest |
| { |
| import std.conv : ConvException; |
| import std.exception : assertThrown, collectException; |
| // try it with an oddly small size |
| VariantN!(1) test; |
| assert(test.size > 1); |
| |
| // variantArray tests |
| auto heterogeneous = variantArray(1, 4.5, "hi"); |
| assert(heterogeneous.length == 3); |
| auto variantArrayAsVariant = Variant(heterogeneous); |
| assert(variantArrayAsVariant[0] == 1); |
| assert(variantArrayAsVariant.length == 3); |
| |
| // array tests |
| auto arr = Variant([1.2].dup); |
| auto e = arr[0]; |
| assert(e == 1.2); |
| arr[0] = 2.0; |
| assert(arr[0] == 2); |
| arr ~= 4.5; |
| assert(arr[1] == 4.5); |
| |
| // general tests |
| Variant a; |
| auto b = Variant(5); |
| assert(!b.peek!(real) && b.peek!(int)); |
| // assign |
| a = *b.peek!(int); |
| // comparison |
| assert(a == b, a.type.toString() ~ " " ~ b.type.toString()); |
| auto c = Variant("this is a string"); |
| assert(a != c); |
| // comparison via implicit conversions |
| a = 42; b = 42.0; assert(a == b); |
| |
| // try failing conversions |
| bool failed = false; |
| try |
| { |
| auto d = c.get!(int); |
| } |
| catch (Exception e) |
| { |
| //writeln(stderr, e.toString); |
| failed = true; |
| } |
| assert(failed); // :o) |
| |
| // toString tests |
| a = Variant(42); assert(a.toString() == "42"); |
| a = Variant(42.22); assert(a.toString() == "42.22"); |
| |
| // coerce tests |
| a = Variant(42.22); assert(a.coerce!(int) == 42); |
| a = cast(short) 5; assert(a.coerce!(double) == 5); |
| a = Variant("10"); assert(a.coerce!int == 10); |
| |
| a = Variant(1); |
| assert(a.coerce!bool); |
| a = Variant(0); |
| assert(!a.coerce!bool); |
| |
| a = Variant(1.0); |
| assert(a.coerce!bool); |
| a = Variant(0.0); |
| assert(!a.coerce!bool); |
| a = Variant(float.init); |
| assertThrown!ConvException(a.coerce!bool); |
| |
| a = Variant("true"); |
| assert(a.coerce!bool); |
| a = Variant("false"); |
| assert(!a.coerce!bool); |
| a = Variant(""); |
| assertThrown!ConvException(a.coerce!bool); |
| |
| // Object tests |
| class B1 {} |
| class B2 : B1 {} |
| a = new B2; |
| assert(a.coerce!(B1) !is null); |
| a = new B1; |
| assert(collectException(a.coerce!(B2) is null)); |
| a = cast(Object) new B2; // lose static type info; should still work |
| assert(a.coerce!(B2) !is null); |
| |
| // struct Big { int a[45]; } |
| // a = Big.init; |
| |
| // hash |
| assert(a.toHash() != 0); |
| } |
| |
| // tests adapted from |
| // http://www.dsource.org/projects/tango/browser/trunk/tango/core/Variant.d?rev=2601 |
| @system unittest |
| { |
| Variant v; |
| |
| assert(!v.hasValue); |
| v = 42; |
| assert( v.peek!(int) ); |
| assert( v.convertsTo!(long) ); |
| assert( v.get!(int) == 42 ); |
| assert( v.get!(long) == 42L ); |
| assert( v.get!(ulong) == 42uL ); |
| |
| v = "Hello, World!"; |
| assert( v.peek!(string) ); |
| |
| assert( v.get!(string) == "Hello, World!" ); |
| assert(!is(char[] : wchar[])); |
| assert( !v.convertsTo!(wchar[]) ); |
| assert( v.get!(string) == "Hello, World!" ); |
| |
| // Literal arrays are dynamically-typed |
| v = cast(int[4]) [1,2,3,4]; |
| assert( v.peek!(int[4]) ); |
| assert( v.get!(int[4]) == [1,2,3,4] ); |
| |
| { |
| v = [1,2,3,4,5]; |
| assert( v.peek!(int[]) ); |
| assert( v.get!(int[]) == [1,2,3,4,5] ); |
| } |
| |
| v = 3.1413; |
| assert( v.peek!(double) ); |
| assert( v.convertsTo!(real) ); |
| //@@@ BUG IN COMPILER: DOUBLE SHOULD NOT IMPLICITLY CONVERT TO FLOAT |
| assert( v.convertsTo!(float) ); |
| assert( *v.peek!(double) == 3.1413 ); |
| |
| auto u = Variant(v); |
| assert( u.peek!(double) ); |
| assert( *u.peek!(double) == 3.1413 ); |
| |
| // operators |
| v = 38; |
| assert( v + 4 == 42 ); |
| assert( 4 + v == 42 ); |
| assert( v - 4 == 34 ); |
| assert( Variant(4) - v == -34 ); |
| assert( v * 2 == 76 ); |
| assert( 2 * v == 76 ); |
| assert( v / 2 == 19 ); |
| assert( Variant(2) / v == 0 ); |
| assert( v % 2 == 0 ); |
| assert( Variant(2) % v == 2 ); |
| assert( (v & 6) == 6 ); |
| assert( (6 & v) == 6 ); |
| assert( (v | 9) == 47 ); |
| assert( (9 | v) == 47 ); |
| assert( (v ^ 5) == 35 ); |
| assert( (5 ^ v) == 35 ); |
| assert( v << 1 == 76 ); |
| assert( Variant(1) << Variant(2) == 4 ); |
| assert( v >> 1 == 19 ); |
| assert( Variant(4) >> Variant(2) == 1 ); |
| assert( Variant("abc") ~ "def" == "abcdef" ); |
| assert( Variant("abc") ~ Variant("def") == "abcdef" ); |
| |
| v = 38; |
| v += 4; |
| assert( v == 42 ); |
| v = 38; v -= 4; assert( v == 34 ); |
| v = 38; v *= 2; assert( v == 76 ); |
| v = 38; v /= 2; assert( v == 19 ); |
| v = 38; v %= 2; assert( v == 0 ); |
| v = 38; v &= 6; assert( v == 6 ); |
| v = 38; v |= 9; assert( v == 47 ); |
| v = 38; v ^= 5; assert( v == 35 ); |
| v = 38; v <<= 1; assert( v == 76 ); |
| v = 38; v >>= 1; assert( v == 19 ); |
| v = 38; v += 1; assert( v < 40 ); |
| |
| v = "abc"; |
| v ~= "def"; |
| assert( v == "abcdef", *v.peek!(char[]) ); |
| assert( Variant(0) < Variant(42) ); |
| assert( Variant(42) > Variant(0) ); |
| assert( Variant(42) > Variant(0.1) ); |
| assert( Variant(42.1) > Variant(1) ); |
| assert( Variant(21) == Variant(21) ); |
| assert( Variant(0) != Variant(42) ); |
| assert( Variant("bar") == Variant("bar") ); |
| assert( Variant("foo") != Variant("bar") ); |
| |
| { |
| auto v1 = Variant(42); |
| auto v2 = Variant("foo"); |
| |
| int[Variant] hash; |
| hash[v1] = 0; |
| hash[v2] = 1; |
| |
| assert( hash[v1] == 0 ); |
| assert( hash[v2] == 1 ); |
| } |
| |
| { |
| int[char[]] hash; |
| hash["a"] = 1; |
| hash["b"] = 2; |
| hash["c"] = 3; |
| Variant vhash = hash; |
| |
| assert( vhash.get!(int[char[]])["a"] == 1 ); |
| assert( vhash.get!(int[char[]])["b"] == 2 ); |
| assert( vhash.get!(int[char[]])["c"] == 3 ); |
| } |
| } |
| |
| @system unittest |
| { |
| // check comparisons incompatible with AllowedTypes |
| Algebraic!int v = 2; |
| |
| assert(v == 2); |
| assert(v < 3); |
| static assert(!__traits(compiles, () => v == long.max)); |
| static assert(!__traits(compiles, () => v == null)); |
| static assert(!__traits(compiles, () => v < long.max)); |
| static assert(!__traits(compiles, () => v > null)); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=1558 |
| @system unittest |
| { |
| Variant va=1; |
| Variant vb=-2; |
| assert((va+vb).get!(int) == -1); |
| assert((va-vb).get!(int) == 3); |
| } |
| |
| @system unittest |
| { |
| Variant a; |
| a=5; |
| Variant b; |
| b=a; |
| Variant[] c; |
| c = variantArray(1, 2, 3.0, "hello", 4); |
| assert(c[3] == "hello"); |
| } |
| |
| @system unittest |
| { |
| Variant v = 5; |
| assert(!__traits(compiles, v.coerce!(bool delegate()))); |
| } |
| |
| |
| @system unittest |
| { |
| struct Huge { |
| real a, b, c, d, e, f, g; |
| } |
| |
| Huge huge; |
| huge.e = 42; |
| Variant v; |
| v = huge; // Compile time error. |
| assert(v.get!(Huge).e == 42); |
| } |
| |
| @system unittest |
| { |
| const x = Variant(42); |
| auto y1 = x.get!(const int); |
| // @@@BUG@@@ |
| //auto y2 = x.get!(immutable int)(); |
| } |
| |
| // test iteration |
| @system unittest |
| { |
| auto v = Variant([ 1, 2, 3, 4 ][]); |
| auto j = 0; |
| foreach (int i; v) |
| { |
| assert(i == ++j); |
| } |
| assert(j == 4); |
| } |
| |
| // test convertibility |
| @system unittest |
| { |
| auto v = Variant("abc".dup); |
| assert(v.convertsTo!(char[])); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=5424 |
| @system unittest |
| { |
| interface A { |
| void func1(); |
| } |
| static class AC: A { |
| void func1() { |
| } |
| } |
| |
| A a = new AC(); |
| a.func1(); |
| Variant b = Variant(a); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=7070 |
| @system unittest |
| { |
| Variant v; |
| v = null; |
| } |
| |
| // Class and interface opEquals, https://issues.dlang.org/show_bug.cgi?id=12157 |
| @system unittest |
| { |
| class Foo { } |
| |
| class DerivedFoo : Foo { } |
| |
| Foo f1 = new Foo(); |
| Foo f2 = new DerivedFoo(); |
| |
| Variant v1 = f1, v2 = f2; |
| assert(v1 == f1); |
| assert(v1 != new Foo()); |
| assert(v1 != f2); |
| assert(v2 != v1); |
| assert(v2 == f2); |
| } |
| |
| // Const parameters with opCall, https://issues.dlang.org/show_bug.cgi?id=11361 |
| @system unittest |
| { |
| static string t1(string c) { |
| return c ~ "a"; |
| } |
| |
| static const(char)[] t2(const(char)[] p) { |
| return p ~ "b"; |
| } |
| |
| static char[] t3(int p) { |
| import std.conv : text; |
| return p.text.dup; |
| } |
| |
| Variant v1 = &t1; |
| Variant v2 = &t2; |
| Variant v3 = &t3; |
| |
| assert(v1("abc") == "abca"); |
| assert(v1("abc").type == typeid(string)); |
| assert(v2("abc") == "abcb"); |
| |
| assert(v2(cast(char[])("abc".dup)) == "abcb"); |
| assert(v2("abc").type == typeid(const(char)[])); |
| |
| assert(v3(4) == ['4']); |
| assert(v3(4).type == typeid(char[])); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=12071 |
| @system unittest |
| { |
| static struct Structure { int data; } |
| alias VariantTest = Algebraic!(Structure delegate() pure nothrow @nogc @safe); |
| |
| bool called = false; |
| Structure example() pure nothrow @nogc @safe |
| { |
| called = true; |
| return Structure.init; |
| } |
| auto m = VariantTest(&example); |
| m(); |
| assert(called); |
| } |
| |
| // Ordering comparisons of incompatible types |
| // e.g. https://issues.dlang.org/show_bug.cgi?id=7990 |
| @system unittest |
| { |
| import std.exception : assertThrown; |
| assertThrown!VariantException(Variant(3) < "a"); |
| assertThrown!VariantException("a" < Variant(3)); |
| assertThrown!VariantException(Variant(3) < Variant("a")); |
| |
| assertThrown!VariantException(Variant.init < Variant(3)); |
| assertThrown!VariantException(Variant(3) < Variant.init); |
| } |
| |
| // Handling of unordered types |
| // https://issues.dlang.org/show_bug.cgi?id=9043 |
| @system unittest |
| { |
| import std.exception : assertThrown; |
| static struct A { int a; } |
| |
| assert(Variant(A(3)) != A(4)); |
| |
| assertThrown!VariantException(Variant(A(3)) < A(4)); |
| assertThrown!VariantException(A(3) < Variant(A(4))); |
| assertThrown!VariantException(Variant(A(3)) < Variant(A(4))); |
| } |
| |
| // Handling of empty types and arrays |
| // https://issues.dlang.org/show_bug.cgi?id=10958 |
| @system unittest |
| { |
| class EmptyClass { } |
| struct EmptyStruct { } |
| alias EmptyArray = void[0]; |
| alias Alg = Algebraic!(EmptyClass, EmptyStruct, EmptyArray); |
| |
| Variant testEmpty(T)() |
| { |
| T inst; |
| Variant v = inst; |
| assert(v.get!T == inst); |
| assert(v.peek!T !is null); |
| assert(*v.peek!T == inst); |
| Alg alg = inst; |
| assert(alg.get!T == inst); |
| return v; |
| } |
| |
| testEmpty!EmptyClass(); |
| testEmpty!EmptyStruct(); |
| testEmpty!EmptyArray(); |
| |
| // EmptyClass/EmptyStruct sizeof is 1, so we have this to test just size 0. |
| EmptyArray arr = EmptyArray.init; |
| Algebraic!(EmptyArray) a = arr; |
| assert(a.length == 0); |
| assert(a.get!EmptyArray == arr); |
| } |
| |
| // Handling of void function pointers / delegates |
| // https://issues.dlang.org/show_bug.cgi?id=11360 |
| @system unittest |
| { |
| static void t1() { } |
| Variant v = &t1; |
| assert(v() == Variant.init); |
| |
| static int t2() { return 3; } |
| Variant v2 = &t2; |
| assert(v2() == 3); |
| } |
| |
| // Using peek for large structs |
| // https://issues.dlang.org/show_bug.cgi?id=8580 |
| @system unittest |
| { |
| struct TestStruct(bool pad) |
| { |
| int val1; |
| static if (pad) |
| ubyte[Variant.size] padding; |
| int val2; |
| } |
| |
| void testPeekWith(T)() |
| { |
| T inst; |
| inst.val1 = 3; |
| inst.val2 = 4; |
| Variant v = inst; |
| T* original = v.peek!T; |
| assert(original.val1 == 3); |
| assert(original.val2 == 4); |
| original.val1 = 6; |
| original.val2 = 8; |
| T modified = v.get!T; |
| assert(modified.val1 == 6); |
| assert(modified.val2 == 8); |
| } |
| |
| testPeekWith!(TestStruct!false)(); |
| testPeekWith!(TestStruct!true)(); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=18780 |
| @system unittest |
| { |
| int x = 7; |
| Variant a = x; |
| assert(a.convertsTo!ulong); |
| assert(a.convertsTo!uint); |
| } |
| |
| /** |
| * Applies a delegate or function to the given $(LREF Algebraic) depending on the held type, |
| * ensuring that all types are handled by the visiting functions. |
| * |
| * The delegate or function having the currently held value as parameter is called |
| * with `variant`'s current value. Visiting handlers are passed |
| * in the template parameter list. |
| * It is statically ensured that all held types of |
| * `variant` are handled across all handlers. |
| * `visit` allows delegates and static functions to be passed |
| * as parameters. |
| * |
| * If a function with an untyped parameter is specified, this function is called |
| * when the variant contains a type that does not match any other function. |
| * This can be used to apply the same function across multiple possible types. |
| * Exactly one generic function is allowed. |
| * |
| * If a function without parameters is specified, this function is called |
| * when `variant` doesn't hold a value. Exactly one parameter-less function |
| * is allowed. |
| * |
| * Duplicate overloads matching the same type in one of the visitors are disallowed. |
| * |
| * Returns: The return type of visit is deduced from the visiting functions and must be |
| * the same across all overloads. |
| * Throws: $(LREF VariantException) if `variant` doesn't hold a value and no |
| * parameter-less fallback function is specified. |
| */ |
| template visit(Handlers...) |
| if (Handlers.length > 0) |
| { |
| /// |
| auto visit(VariantType)(VariantType variant) |
| if (isAlgebraic!VariantType) |
| { |
| return visitImpl!(true, VariantType, Handlers)(variant); |
| } |
| } |
| |
| /// |
| @system unittest |
| { |
| Algebraic!(int, string) variant; |
| |
| variant = 10; |
| assert(variant.visit!((string s) => cast(int) s.length, |
| (int i) => i)() |
| == 10); |
| variant = "string"; |
| assert(variant.visit!((int i) => i, |
| (string s) => cast(int) s.length)() |
| == 6); |
| |
| // Error function usage |
| Algebraic!(int, string) emptyVar; |
| auto rslt = emptyVar.visit!((string s) => cast(int) s.length, |
| (int i) => i, |
| () => -1)(); |
| assert(rslt == -1); |
| |
| // Generic function usage |
| Algebraic!(int, float, real) number = 2; |
| assert(number.visit!(x => x += 1) == 3); |
| |
| // Generic function for int/float with separate behavior for string |
| Algebraic!(int, float, string) something = 2; |
| assert(something.visit!((string s) => s.length, x => x) == 2); // generic |
| something = "asdf"; |
| assert(something.visit!((string s) => s.length, x => x) == 4); // string |
| |
| // Generic handler and empty handler |
| Algebraic!(int, float, real) empty2; |
| assert(empty2.visit!(x => x + 1, () => -1) == -1); |
| } |
| |
| @system unittest |
| { |
| Algebraic!(size_t, string) variant; |
| |
| // not all handled check |
| static assert(!__traits(compiles, variant.visit!((size_t i){ })() )); |
| |
| variant = cast(size_t) 10; |
| auto which = 0; |
| variant.visit!( (string s) => which = 1, |
| (size_t i) => which = 0 |
| )(); |
| |
| // integer overload was called |
| assert(which == 0); |
| |
| // mustn't compile as generic Variant not supported |
| Variant v; |
| static assert(!__traits(compiles, v.visit!((string s) => which = 1, |
| (size_t i) => which = 0 |
| )() |
| )); |
| |
| static size_t func(string s) { |
| return s.length; |
| } |
| |
| variant = "test"; |
| assert( 4 == variant.visit!(func, |
| (size_t i) => i |
| )()); |
| |
| Algebraic!(int, float, string) variant2 = 5.0f; |
| // Shouldn' t compile as float not handled by visitor. |
| static assert(!__traits(compiles, variant2.visit!( |
| (int _) {}, |
| (string _) {})())); |
| |
| Algebraic!(size_t, string, float) variant3; |
| variant3 = 10.0f; |
| auto floatVisited = false; |
| |
| assert(variant3.visit!( |
| (float f) { floatVisited = true; return cast(size_t) f; }, |
| func, |
| (size_t i) { return i; } |
| )() == 10); |
| assert(floatVisited == true); |
| |
| Algebraic!(float, string) variant4; |
| |
| assert(variant4.visit!(func, (float f) => cast(size_t) f, () => size_t.max)() == size_t.max); |
| |
| // double error func check |
| static assert(!__traits(compiles, |
| visit!(() => size_t.max, func, (float f) => cast(size_t) f, () => size_t.max)(variant4)) |
| ); |
| } |
| |
| // disallow providing multiple generic handlers to visit |
| // disallow a generic handler that does not apply to all types |
| @system unittest |
| { |
| Algebraic!(int, float) number = 2; |
| // ok, x + 1 valid for int and float |
| static assert( __traits(compiles, number.visit!(x => x + 1))); |
| // bad, two generic handlers |
| static assert(!__traits(compiles, number.visit!(x => x + 1, x => x + 2))); |
| // bad, x ~ "a" does not apply to int or float |
| static assert(!__traits(compiles, number.visit!(x => x ~ "a"))); |
| // bad, x ~ "a" does not apply to int or float |
| static assert(!__traits(compiles, number.visit!(x => x + 1, x => x ~ "a"))); |
| |
| Algebraic!(int, string) maybenumber = 2; |
| // ok, x ~ "a" valid for string, x + 1 valid for int, only 1 generic |
| static assert( __traits(compiles, maybenumber.visit!((string x) => x ~ "a", x => "foobar"[0 .. x + 1]))); |
| // bad, x ~ "a" valid for string but not int |
| static assert(!__traits(compiles, maybenumber.visit!(x => x ~ "a"))); |
| // bad, two generics, each only applies in one case |
| static assert(!__traits(compiles, maybenumber.visit!(x => x + 1, x => x ~ "a"))); |
| } |
| |
| /** |
| * Behaves as $(LREF visit) but doesn't enforce that all types are handled |
| * by the visiting functions. |
| * |
| * If a parameter-less function is specified it is called when |
| * either `variant` doesn't hold a value or holds a type |
| * which isn't handled by the visiting functions. |
| * |
| * Returns: The return type of tryVisit is deduced from the visiting functions and must be |
| * the same across all overloads. |
| * Throws: $(LREF VariantException) if `variant` doesn't hold a value or |
| * `variant` holds a value which isn't handled by the visiting functions, |
| * when no parameter-less fallback function is specified. |
| */ |
| template tryVisit(Handlers...) |
| if (Handlers.length > 0) |
| { |
| /// |
| auto tryVisit(VariantType)(VariantType variant) |
| if (isAlgebraic!VariantType) |
| { |
| return visitImpl!(false, VariantType, Handlers)(variant); |
| } |
| } |
| |
| /// |
| @system unittest |
| { |
| Algebraic!(int, string) variant; |
| |
| variant = 10; |
| auto which = -1; |
| variant.tryVisit!((int i) { which = 0; })(); |
| assert(which == 0); |
| |
| // Error function usage |
| variant = "test"; |
| variant.tryVisit!((int i) { which = 0; }, |
| () { which = -100; })(); |
| assert(which == -100); |
| } |
| |
| @system unittest |
| { |
| import std.exception : assertThrown; |
| Algebraic!(int, string) variant; |
| |
| variant = 10; |
| auto which = -1; |
| variant.tryVisit!((int i){ which = 0; })(); |
| |
| assert(which == 0); |
| |
| variant = "test"; |
| |
| assertThrown!VariantException(variant.tryVisit!((int i) { which = 0; })()); |
| |
| void errorfunc() |
| { |
| which = -1; |
| } |
| |
| variant.tryVisit!((int i) { which = 0; }, errorfunc)(); |
| |
| assert(which == -1); |
| } |
| |
| private template isAlgebraic(Type) |
| { |
| static if (is(Type _ == VariantN!T, T...)) |
| enum isAlgebraic = T.length >= 2; // T[0] == maxDataSize, T[1..$] == AllowedTypesParam |
| else |
| enum isAlgebraic = false; |
| } |
| |
| @system unittest |
| { |
| static assert(!isAlgebraic!(Variant)); |
| static assert( isAlgebraic!(Algebraic!(string))); |
| static assert( isAlgebraic!(Algebraic!(int, int[]))); |
| } |
| |
| private auto visitImpl(bool Strict, VariantType, Handler...)(VariantType variant) |
| if (isAlgebraic!VariantType && Handler.length > 0) |
| { |
| alias AllowedTypes = VariantType.AllowedTypes; |
| |
| |
| /** |
| * Returns: Struct where `indices` is an array which |
| * contains at the n-th position the index in Handler which takes the |
| * n-th type of AllowedTypes. If an Handler doesn't match an |
| * AllowedType, -1 is set. If a function in the delegates doesn't |
| * have parameters, the field `exceptionFuncIdx` is set; |
| * otherwise it's -1. |
| */ |
| auto visitGetOverloadMap() |
| { |
| struct Result { |
| int[AllowedTypes.length] indices; |
| int exceptionFuncIdx = -1; |
| int generalFuncIdx = -1; |
| } |
| |
| Result result; |
| |
| enum int nonmatch = () |
| { |
| foreach (int dgidx, dg; Handler) |
| { |
| bool found = false; |
| foreach (T; AllowedTypes) |
| { |
| found |= __traits(compiles, { static assert(isSomeFunction!(dg!T)); }); |
| found |= __traits(compiles, (T t) { dg(t); }); |
| found |= __traits(compiles, dg()); |
| } |
| if (!found) return dgidx; |
| } |
| return -1; |
| }(); |
| static assert(nonmatch == -1, "No match for visit handler #"~ |
| nonmatch.stringof~" ("~Handler[nonmatch].stringof~")"); |
| |
| foreach (tidx, T; AllowedTypes) |
| { |
| bool added = false; |
| foreach (dgidx, dg; Handler) |
| { |
| // Handle normal function objects |
| static if (isSomeFunction!dg) |
| { |
| alias Params = Parameters!dg; |
| static if (Params.length == 0) |
| { |
| // Just check exception functions in the first |
| // inner iteration (over delegates) |
| if (tidx > 0) |
| continue; |
| else |
| { |
| if (result.exceptionFuncIdx != -1) |
| assert(false, "duplicate parameter-less (error-)function specified"); |
| result.exceptionFuncIdx = dgidx; |
| } |
| } |
| else static if (is(Params[0] == T) || is(Unqual!(Params[0]) == T)) |
| { |
| if (added) |
| assert(false, "duplicate overload specified for type '" ~ T.stringof ~ "'"); |
| |
| added = true; |
| result.indices[tidx] = dgidx; |
| } |
| } |
| else static if (__traits(compiles, { static assert(isSomeFunction!(dg!T)); })) |
| { |
| assert(result.generalFuncIdx == -1 || |
| result.generalFuncIdx == dgidx, |
| "Only one generic visitor function is allowed"); |
| result.generalFuncIdx = dgidx; |
| } |
| // Handle composite visitors with opCall overloads |
| } |
| |
| if (!added) |
| result.indices[tidx] = -1; |
| } |
| |
| return result; |
| } |
| |
| enum HandlerOverloadMap = visitGetOverloadMap(); |
| |
| if (!variant.hasValue) |
| { |
| // Call the exception function. The HandlerOverloadMap |
| // will have its exceptionFuncIdx field set to value != -1 if an |
| // exception function has been specified; otherwise we just through an exception. |
| static if (HandlerOverloadMap.exceptionFuncIdx != -1) |
| return Handler[ HandlerOverloadMap.exceptionFuncIdx ](); |
| else |
| throw new VariantException("variant must hold a value before being visited."); |
| } |
| |
| foreach (idx, T; AllowedTypes) |
| { |
| if (auto ptr = variant.peek!T) |
| { |
| enum dgIdx = HandlerOverloadMap.indices[idx]; |
| |
| static if (dgIdx == -1) |
| { |
| static if (HandlerOverloadMap.generalFuncIdx >= 0) |
| return Handler[HandlerOverloadMap.generalFuncIdx](*ptr); |
| else static if (Strict) |
| static assert(false, "overload for type '" ~ T.stringof ~ "' hasn't been specified"); |
| else static if (HandlerOverloadMap.exceptionFuncIdx != -1) |
| return Handler[HandlerOverloadMap.exceptionFuncIdx](); |
| else |
| throw new VariantException( |
| "variant holds value of type '" |
| ~ T.stringof ~ |
| "' but no visitor has been provided" |
| ); |
| } |
| else |
| { |
| return Handler[ dgIdx ](*ptr); |
| } |
| } |
| } |
| |
| assert(false); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=21253 |
| @system unittest |
| { |
| static struct A { int n; } |
| static struct B { } |
| |
| auto a = Algebraic!(A, B)(B()); |
| assert(a.visit!( |
| (B _) => 42, |
| (a ) => a.n |
| ) == 42); |
| } |
| |
| @system unittest |
| { |
| // validate that visit can be called with a const type |
| struct Foo { int depth; } |
| struct Bar { int depth; } |
| alias FooBar = Algebraic!(Foo, Bar); |
| |
| int depth(in FooBar fb) { |
| return fb.visit!((Foo foo) => foo.depth, |
| (Bar bar) => bar.depth); |
| } |
| |
| FooBar fb = Foo(3); |
| assert(depth(fb) == 3); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=16383 |
| @system unittest |
| { |
| class Foo {this() immutable {}} |
| alias V = Algebraic!(immutable Foo); |
| |
| auto x = V(new immutable Foo).visit!( |
| (immutable(Foo) _) => 3 |
| ); |
| assert(x == 3); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=5310 |
| @system unittest |
| { |
| const Variant a; |
| assert(a == a); |
| Variant b; |
| assert(a == b); |
| assert(b == a); |
| } |
| |
| @system unittest |
| { |
| const Variant a = [2]; |
| assert(a[0] == 2); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=10017 |
| @system unittest |
| { |
| static struct S |
| { |
| ubyte[Variant.size + 1] s; |
| } |
| |
| Variant v1, v2; |
| v1 = S(); // the payload is allocated on the heap |
| v2 = v1; // AssertError: target must be non-null |
| assert(v1 == v2); |
| } |
| |
| // https://issues.dlang.org/show_bug.cgi?id=7069 |
| @system unittest |
| { |
| import std.exception : assertThrown; |
| Variant v; |
| |
| int i = 10; |
| v = i; |
| static foreach (qual; AliasSeq!(Alias, ConstOf)) |
| { |
| assert(v.get!(qual!int) == 10); |
| assert(v.get!(qual!float) == 10.0f); |
| } |
| static foreach (qual; AliasSeq!(ImmutableOf, SharedOf, SharedConstOf)) |
| { |
| assertThrown!VariantException(v.get!(qual!int)); |
| } |
| |
| const(int) ci = 20; |
| v = ci; |
| static foreach (qual; AliasSeq!(ConstOf)) |
| { |
| assert(v.get!(qual!int) == 20); |
| assert(v.get!(qual!float) == 20.0f); |
| } |
| static foreach (qual; AliasSeq!(Alias, ImmutableOf, SharedOf, SharedConstOf)) |
| { |
| assertThrown!VariantException(v.get!(qual!int)); |
| assertThrown!VariantException(v.get!(qual!float)); |
| } |
| |
| immutable(int) ii = ci; |
| v = ii; |
| static foreach (qual; AliasSeq!(ImmutableOf, ConstOf, SharedConstOf)) |
| { |
| assert(v.get!(qual!int) == 20); |
| assert(v.get!(qual!float) == 20.0f); |
| } |
| static foreach (qual; AliasSeq!(Alias, SharedOf)) |
| { |
| assertThrown!VariantException(v.get!(qual!int)); |
| assertThrown!VariantException(v.get!(qual!float)); |
| } |
| |
|