diff --git a/include/proper.hrl b/include/proper.hrl index 803408ae..3893d515 100644 --- a/include/proper.hrl +++ b/include/proper.hrl @@ -47,9 +47,9 @@ %%------------------------------------------------------------------------------ -import(proper_types, [integer/2, float/2, atom/0, binary/0, binary/1, - bitstring/0, bitstring/1, list/1, vector/2, union/1, - weighted_union/1, tuple/1, loose_tuple/1, exactly/1, - fixed_list/1, function/2, map/2, any/0]). + bitstring/0, bitstring/1, list/1, map/1, map/2, + vector/2, union/1, weighted_union/1, tuple/1, loose_tuple/1, + exactly/1, fixed_list/1, fixed_map/1, function/2, any/0]). %%------------------------------------------------------------------------------ @@ -59,7 +59,8 @@ -import(proper_types, [integer/0, non_neg_integer/0, pos_integer/0, neg_integer/0, range/2, float/0, non_neg_float/0, number/0, boolean/0, arity/0, byte/0, char/0, list/0, - tuple/0, map/0, string/0, term/0, timeout/0, wunion/1]). + tuple/0, map/0, merge_maps/2, string/0, term/0, timeout/0, + wunion/1]). -import(proper_types, [int/0, nat/0, largeint/0, real/0, bool/0, choose/2, elements/1, oneof/1, frequency/1, return/1, default/2, orderedlist/1, function0/1, function1/1, function2/1, diff --git a/include/proper_internal.hrl b/include/proper_internal.hrl index ecb7133f..bcea1119 100644 --- a/include/proper_internal.hrl +++ b/include/proper_internal.hrl @@ -69,6 +69,7 @@ -define(PROPERTY_PREFIX, "prop_"). +-define(var(Value), io:format("~s = ~p\n", [??Value, Value])). %%------------------------------------------------------------------------------ %% Constants diff --git a/src/proper_gen.erl b/src/proper_gen.erl index 6f603917..4310b506 100644 --- a/src/proper_gen.erl +++ b/src/proper_gen.erl @@ -42,8 +42,8 @@ binary_rev/1, binary_len_gen/1, bitstring_gen/1, bitstring_rev/1, bitstring_len_gen/1, list_gen/2, distlist_gen/3, vector_gen/2, union_gen/1, weighted_union_gen/1, tuple_gen/1, loose_tuple_gen/2, - loose_tuple_rev/2, exactly_gen/1, fixed_list_gen/1, function_gen/2, - any_gen/1, native_type_gen/2, safe_weighted_union_gen/1, + loose_tuple_rev/2, exactly_gen/1, fixed_list_gen/1, fixed_map_gen/1, + function_gen/2, any_gen/1, native_type_gen/2, safe_weighted_union_gen/1, safe_union_gen/1]). %% Public API types @@ -344,6 +344,9 @@ clean_instance({'$to_part',ImmInstance}) -> clean_instance(ImmInstance); clean_instance(ImmInstance) when is_list(ImmInstance) -> clean_instance_list(ImmInstance); +clean_instance(ImmInstance) when is_map(ImmInstance) -> + %% maps:map only changes the values, this handles both keys and values + maps:from_list(clean_instance_list(maps:to_list(ImmInstance))); clean_instance(ImmInstance) when is_tuple(ImmInstance) -> list_to_tuple(clean_instance_list(tuple_to_list(ImmInstance))); clean_instance(ImmInstance) -> ImmInstance. @@ -576,6 +579,15 @@ fixed_list_gen({ProperHead,ImproperTail}) -> fixed_list_gen(ProperFields) -> [generate(F) || F <- ProperFields]. +%% @private +-spec fixed_map_gen(map()) -> imm_instance(). +fixed_map_gen(Map) when is_map(Map) -> + maps:from_list([ + {generate(KeyOrType), generate(ValueOrType)} + || + {KeyOrType, ValueOrType} <- maps:to_list(Map) + ]). + %% @private -spec function_gen(arity(), proper_types:type()) -> function(). function_gen(Arity, RetType) -> diff --git a/src/proper_shrink.erl b/src/proper_shrink.erl index 1d90c2ba..065e40aa 100644 --- a/src/proper_shrink.erl +++ b/src/proper_shrink.erl @@ -33,6 +33,7 @@ -export([number_shrinker/4, union_first_choice_shrinker/3, union_recursive_shrinker/3]). -export([split_shrinker/3, remove_shrinker/3]). +-export([map_remove_shrinker/3, map_key_shrinker/3, map_value_shrinker/3]). -export_type([state/0, shrinker/0]). @@ -397,6 +398,107 @@ elements_shrinker(Instance, Type, elements_shrinker(Instance, Type, {inner,Indices,GetElemType,{shrunk,N,InnerState}}). +-spec map_remove_shrinker( + proper_gen:imm_instance(), proper_types:type(), state() +) -> {[proper_gen:imm_instance()], state()}. +map_remove_shrinker(Instance, Type, init) when is_map(Instance) -> + GetKeys = proper_types:get_prop(get_keys, Type), + Keys = GetKeys(Instance), + map_remove_shrinker(Instance, Type, {shrunk, 1, {keys, ordsets:new(), Keys}}); +map_remove_shrinker(Instance, _Type, {keys, _Checked, []}) when is_map(Instance) -> + {[], done}; +map_remove_shrinker(Instance, Type, {keys, Checked, [Key | Rest]}) when is_map(Instance) -> + Remove = proper_types:get_prop(remove, Type), + {[Remove(Key, Instance)], {keys, ordsets:add_element(Key, Checked), Rest}}; +map_remove_shrinker(Instance, Type, {shrunk, 1, {keys, Checked, ToCheck}}) when is_map(Instance) -> + %% GetKeys = proper_types:get_prop(get_keys, Type), + %% Keys = ordsets:from_list(GetKeys(Instance)), + %% NewToCheck = ordsets:subtract(Keys, Checked), + map_remove_shrinker(Instance, Type, {keys, Checked, ToCheck}). + +-spec map_value_shrinker( + proper_gen:imm_instance(), proper_types:type(), state() +) -> {[proper_gen:imm_instance()], state()}. +map_value_shrinker(Instance, _Type, init) when map_size(Instance) =:= 0 -> + {[], done}; +map_value_shrinker(Instance, Type, init) when is_map(Instance) -> + GetKeys = proper_types:get_prop(get_keys, Type), + TypeMap = proper_types:get_prop(internal_types, Type), + Keys = GetKeys(Instance), + ValueTypeMap = maps:map(fun(Key, Value) -> + {_KeyType, ValueType} = get_map_field_candidates(Key, Value, TypeMap), + ValueType + end, Instance), + map_value_shrinker(Instance, Type, {inner, Keys, ValueTypeMap, init}); +map_value_shrinker(Instance, _Type, {inner, [], _ValueTypeMap, init}) when is_map(Instance) -> + {[], done}; +map_value_shrinker( + Instance, Type, {inner, [_Key | Rest], ValueTypeMap, done} +) when is_map(Instance) -> + map_value_shrinker(Instance, Type, {inner, Rest, ValueTypeMap, init}); +map_value_shrinker( + Instance, Type, {inner, Keys = [Key | _], ValueTypeMap, InnerState} +) when is_map(Instance) -> + Retrieve = proper_types:get_prop(retrieve, Type), + Update = proper_types:get_prop(update, Type), + Value = Retrieve(Key, Instance), + ValueType = Retrieve(Key, ValueTypeMap), + {NewValues, NewInnerState} = shrink(Value, ValueType, InnerState), + NewInstances = [Update(Key, NewValue, Instance) || NewValue <- NewValues], + {NewInstances, {inner, Keys, ValueTypeMap, NewInnerState}}; +map_value_shrinker( + Instance, Type, {shrunk, N, {inner, ToCheck, ValueTypeMap, InnerState}} +) when is_map(Instance) -> + map_value_shrinker( + Instance, Type, {inner, ToCheck, ValueTypeMap, {shrunk, N, InnerState}} + ). + +-spec map_key_shrinker( + proper_gen:imm_instance(), proper_types:type(), state() +) -> {[proper_gen:imm_instance()], state()}. +map_key_shrinker(Instance, Type, init) when is_map(Instance) -> + GetKeys = proper_types:get_prop(get_keys, Type), + TypeMap = proper_types:get_prop(internal_types, Type), + Keys = GetKeys(Instance), + KeyTypeMap = maps:map(fun(Key, Value) -> + {KeyType, _ValueType} = get_map_field_candidates(Key, Value, TypeMap), + KeyType + end, Instance), + map_key_shrinker(Instance, Type, {inner, Keys, KeyTypeMap, init}); +map_key_shrinker(Instance, _Type, {inner, [], _ValueTypeMap, init}) when is_map(Instance) -> + {[], done}; +map_key_shrinker( + Instance, Type, {inner, [_Key | Rest], KeyTypeMap, done} +) when is_map(Instance) -> + map_key_shrinker(Instance, Type, {inner, Rest, KeyTypeMap, init}); +map_key_shrinker( + Instance, Type, {inner, Keys = [Key | _], KeyTypeMap, InnerState} +) when is_map(Instance) -> + Retrieve = proper_types:get_prop(retrieve, Type), + Update = proper_types:get_prop(update, Type), + Remove = proper_types:get_prop(remove, Type), + Value = Retrieve(Key, Instance), + KeyType = Retrieve(Key, KeyTypeMap), + {NewKeys, NewInnerState} = shrink(Key, KeyType, InnerState), + InstanceWithoutKey = Remove(Key, Instance), + NewInstances = [ + Update(NewKey, Value, InstanceWithoutKey) || NewKey <- NewKeys + ], + {NewInstances, {inner, Keys, KeyTypeMap, NewInnerState}}; +map_key_shrinker( + Instance, Type, {shrunk, N, {inner, ToCheck, KeyTypeMap, InnerState}} +) when is_map(Instance) -> + map_key_shrinker( + Instance, Type, {inner, ToCheck, KeyTypeMap, {shrunk, N, InnerState}} + ). + +get_map_field_candidates(Key, Value, TypeMap) -> + Candidates = maps:filter(fun(KeyType, ValueType) -> + proper_types:is_instance(Key, KeyType) andalso + proper_types:is_instance(Value, ValueType) + end, TypeMap), + {KeyType, ValueType, _} = maps:next(maps:iterator(Candidates)), + {KeyType, ValueType}. %%------------------------------------------------------------------------------ %% Custom shrinkers diff --git a/src/proper_types.erl b/src/proper_types.erl index 676aff22..126b0abd 100644 --- a/src/proper_types.erl +++ b/src/proper_types.erl @@ -141,14 +141,14 @@ -export([integer/2, float/2, atom/0, binary/0, binary/1, bitstring/0, bitstring/1, list/1, vector/2, union/1, weighted_union/1, tuple/1, - loose_tuple/1, exactly/1, fixed_list/1, function/2, map/0, map/2, - any/0, shrink_list/1, safe_union/1, safe_weighted_union/1]). + loose_tuple/1, exactly/1, fixed_list/1, fixed_map/1, function/2, map/0, + map/1, map/2, any/0, shrink_list/1, safe_union/1, safe_weighted_union/1]). -export([integer/0, non_neg_integer/0, pos_integer/0, neg_integer/0, range/2, float/0, non_neg_float/0, number/0, boolean/0, byte/0, char/0, nil/0, list/0, tuple/0, string/0, wunion/1, term/0, timeout/0, arity/0]). -export([int/0, nat/0, largeint/0, real/0, bool/0, choose/2, elements/1, oneof/1, frequency/1, return/1, default/2, orderedlist/1, function0/1, - function1/1, function2/1, function3/1, function4/1, + function1/1, function2/1, function3/1, function4/1, merge_maps/1, weighted_default/2]). -export([resize/2, non_empty/1, noshrink/1]). @@ -239,7 +239,7 @@ | 'combine' | 'alt_gens' | 'shrink_to_parts' | 'size_transform' | 'is_instance' | 'shrinkers' | 'noshrink' | 'internal_type' | 'internal_types' - | 'get_length' | 'split' | 'join' | 'get_indices' + | 'get_length' | 'split' | 'join' | 'get_indices' | 'get_keys' | 'remove' | 'retrieve' | 'update' | 'constraints' | 'parameters' | 'env' | 'subenv' | 'user_nf' | 'is_user_nf' | 'matcher'. @@ -258,7 +258,7 @@ | {'shrinkers', [proper_shrink:shrinker()]} | {'noshrink', boolean()} | {'internal_type', raw_type()} - | {'internal_types', tuple() | maybe_improper_list(type(),type() | [])} + | {'internal_types', tuple() | map() | maybe_improper_list(type(),type() | [])} %% The items returned by 'remove' must be of this type. | {'get_length', fun((proper_gen:imm_instance()) -> length())} %% If this is a container type, this should return the number of elements @@ -277,6 +277,10 @@ proper_gen:imm_instance()) -> [index()])} %% If this is a container type, this should return a list of indices we %% can use to remove or insert elements from the given instance. + | {'get_keys', fun((proper_types:type(), + proper_gen:imm_instance()) -> [term()])} + %% Simliar to `get_indices' but for mapping types where the keys of the + %% type is not necessarily the same as the keys of the instance. | {'remove', fun((index(),proper_gen:imm_instance()) -> proper_gen:imm_instance())} | {'retrieve', fun((index(), proper_gen:imm_instance() | tuple() @@ -312,6 +316,8 @@ cook_outer(RawType) when is_tuple(RawType) -> tuple(tuple_to_list(RawType)); cook_outer(RawType) when is_list(RawType) -> fixed_list(RawType); %% CAUTION: this must handle improper lists +cook_outer(RawType) when is_map(RawType) -> + fixed_map(RawType); cook_outer(RawType) -> %% default case (integers, floats, atoms, binaries, ...) exactly(RawType). @@ -1113,12 +1119,93 @@ function_is_instance(Type, X) -> %% TODO: what if it's not a function we produced? andalso equal_types(RetType, proper_gen:get_ret_type(X)). +%% @doc A map whose keys and values are defined by the given `Map'. +%% +%% Shrinks towards the empty map. That is, all keys are assumed to be optional. +%% +%% Also written simply as a {@link maps. map}. +-spec map(#{Key::raw_type() => Value::raw_type()}) -> proper_types:type(). +map(Map) when is_map(Map) -> + MapType = fixed_map(Map), + Shrinkers = get_prop(shrinkers, MapType), + add_props([ + {remove,fun maps:remove/2}, + {shrinkers, [ + fun proper_shrink:map_remove_shrinker/3, + fun proper_shrink:map_key_shrinker/3 + | Shrinkers + ]} + ], MapType). + %% @doc A map whose keys are defined by the generator `K' and values %% by the generator `V'. -spec map(K::raw_type(), V::raw_type()) -> proper_types:type(). map(K, V) -> ?LET(L, list({K, V}), maps:from_list(L)). +%% @doc A map merged from the given map generators. +-spec merge_maps([Map::raw_type()]) -> proper_types:type(). +merge_maps(RawMaps) when is_list(RawMaps) -> + ?LET(Maps, RawMaps, lists:foldl(fun maps:merge/2, #{}, Maps)). + +%% @doc A map whose keys and values are defined by the given `Map'. +%% Also written simply as a {@link maps. map}. +-spec fixed_map(#{Key::raw_type() => Value::raw_type()}) -> proper_types:type(). +fixed_map(Map) when is_map(Map) -> + %% maps:map only changes the values, this handles both keys and values + WithValueTypes = maps:from_list([ + {cook_outer(Key), cook_outer(Value)} + || {Key, Value} <- maps:to_list(Map) + ]), + ?CONTAINER([ + {generator, {typed, fun map_gen/1}}, + {is_instance, {typed, fun map_is_instance/2}}, + {shrinkers, [fun proper_shrink:map_value_shrinker/3]}, + {internal_types, WithValueTypes}, + {get_length, fun maps:size/1}, + {join, fun maps:merge/2}, + {get_keys, fun maps:keys/1}, + {retrieve, fun maps:get/2}, + {update, fun maps:put/3} + ]). + +map_gen(Type) -> + Map = get_prop(internal_types, Type), + proper_gen:fixed_map_gen(Map). + +map_is_instance(Type, X) when is_map(X) -> + Map = get_prop(internal_types, Type), + map_all( + fun (Key, ValueType) when is_map_key(Key, X) -> + is_instance(maps:get(Key, X), ValueType); + (KeyOrType, ValueType) -> + case is_raw_type(KeyOrType) of + true -> + map_all(fun(Key, Value) -> + case is_instance(Key, KeyOrType) of + true -> is_instance(Value, ValueType); + false -> true %% Ignore other keys + end + end, X); + false -> + %% The key not a type and not in `X' + false + end + end, + Map + ); +map_is_instance(_Type, _X) -> + false. + +map_all(Fun, Map) when is_function(Fun, 2) andalso is_map(Map) -> + map_all_internal(Fun, maps:next(maps:iterator(Map)), true). + +map_all_internal(Fun, _, false) when is_function(Fun, 2) -> + false; +map_all_internal(Fun, none, Result) when is_function(Fun, 2) andalso is_boolean(Result) -> + Result; +map_all_internal(Fun, {Key, Value, NextIterator}, true) when is_function(Fun, 2) -> + map_all_internal(Fun, NextIterator, Fun(Key, Value)). %% @doc All Erlang terms (that PropEr can produce). For reasons of efficiency, %% functions are never produced as instances of this type.
diff --git a/src/proper_typeserver.erl b/src/proper_typeserver.erl index c5ea8112..cd25649b 100644 --- a/src/proper_typeserver.erl +++ b/src/proper_typeserver.erl @@ -1648,6 +1648,8 @@ convert(_Mod, {type,_,nonempty_string,[]}, State, _Stack, _VarDict) -> {ok, {simple,proper_types:non_empty(proper_types:string())}, State}; convert(_Mod, {type,_,map,any}, State, _Stack, _VarDict) -> {ok, {simple,proper_types:map()}, State}; +convert(Mod, {type,_,map,Fields}, State, Stack, VarDict) -> + convert_map(Mod, Fields, State, Stack, VarDict); convert(_Mod, {type,_,tuple,any}, State, _Stack, _VarDict) -> {ok, {simple,proper_types:tuple()}, State}; convert(Mod, {type,_,tuple,ElemForms}, State, Stack, VarDict) -> @@ -1787,6 +1789,82 @@ convert_normal_rec_list(RecFun, RecArgs, NonEmpty) -> NewRecArgs = clean_rec_args(RecArgs), {NewRecFun, NewRecArgs}. +-spec convert_map(mod_name(), [Field], state(), stack(), var_dict()) -> + rich_result2(ret_type(), state()) +when + Field :: {type, erl_anno:anno(), map_field_assoc, [abs_type()]} + | {type, erl_anno:anno(), map_field_exact, [abs_type()]}. +convert_map(Mod, Fields, State1, Stack, VarDict) -> + {AbstractRequiredFields, AbstractOptionalFields} = lists:partition( + fun ({type, _, map_field_exact, _FieldType}) -> + true; + ({type, _, map_field_assoc, _FieldType}) -> + false + end, + Fields + ), + case process_map_fields(required, Mod, AbstractRequiredFields, State1, Stack, VarDict) of + {ok, RawRequired, State2} -> + case process_map_fields(optional, Mod, AbstractOptionalFields, State2, Stack, VarDict) of + {ok, RawOptional, State3} -> + % {ok, {rec, fun map_rec_fun/2, [ + % {false, {maps, type, from_list, [RequiredFields]}}, + % {false, {maps, type, from_list, [OptionalFields]}} + % ]}, State3}; + % ?var(Stack), + % io:format("Required ~P \n", [RequiredFields, 10]), + % io:format("Optional ~P \n", [OptionalFields, 10]), + % ?var(proper_gen:safe_generate(Required)), + % ?var(proper_gen:safe_generate(Optional)), + % Required = proper_types:fixed_map(maps:from_list(RequiredFields)), + % Optional = proper_types:map(maps:from_list(OptionalFields)), + % {ok, {simple, proper_types:merge_maps([Required, Optional])}, State3}; + MapType = ?LET( + {Required, Optional}, + {RawRequired, RawOptional}, + maps:merge(Required, Optional) + ), + {ok, {simple, MapType}, State3}; + {error, Reason} -> + {error, Reason} + end; + {error, Reason} -> + {error, Reason} + end. + +process_map_fields(MapKind, Mod, AbstractFields, State, Stack, VarDict) -> + Process = + fun ({type, _, _, RawFieldTypes}, {ok, Fields, State1}) when + length(RawFieldTypes) =:= 2 + -> + case process_list( + Mod, RawFieldTypes, State1, [map | Stack], VarDict + ) of + {ok, FieldTypes, State2} -> + case combine_ret_types(FieldTypes, {map, MapKind}) of + {simple, FinType} -> + {ok, [{simple, FinType} | Fields], State2}; + {rec, _RecFun, RecArgs} = Type -> + case at_toplevel(RecArgs, Stack) of + true -> + base_case_error(Stack); + false -> + {ok, [Type | Fields], State2} + end + end; + {error, Reason} -> + {error, Reason} + end; + (_FieldTypes, {error, Reason}) -> + {error, Reason} + end, + case lists:foldl(Process, {ok, [], State}, AbstractFields) of + {ok, ReverseFields, NewState} -> + {ok, lists:reverse(ReverseFields), NewState}; + {error, Reason} -> + {error, Reason} + end. + -spec convert_tuple(mod_name(), [abs_type()], boolean(), state(), stack(), var_dict()) -> rich_result2(ret_type(),state()). convert_tuple(Mod, ElemForms, ToList, State, Stack, VarDict) -> @@ -2216,8 +2294,8 @@ partition_rec_args(FullTypeRef, RecArgs, OnlyInstanceAccepting) -> proper_arith:partition(SameType, RecArgs). %% Tuples can be of 0 arity, unions of 1 and wunions at least of 2. --spec combine_ret_types([ret_type()], {'tuple',boolean()} | 'union' - | 'wunion') -> ret_type(). +-spec combine_ret_types([ret_type()], {'tuple', boolean()} | + {'map', 'required' | 'optional'} | 'union' | 'wunion') -> ret_type(). combine_ret_types(RetTypes, EnclosingType) -> case lists:all(fun is_simple_ret_type/1, RetTypes) of true -> @@ -2225,6 +2303,7 @@ combine_ret_types(RetTypes, EnclosingType) -> Combine = case EnclosingType of {tuple,false} -> fun proper_types:tuple/1; {tuple,true} -> fun proper_types:fixed_list/1; + {map, _} -> combine_simple_map_fun(EnclosingType); union -> fun proper_types:union/1 end, FinTypes = [T || {simple,T} <- RetTypes], @@ -2247,7 +2326,9 @@ combine_ret_types(RetTypes, EnclosingType) -> {union_rec_fun(RecFunInfo),clean_rec_args(FlatRecArgs)}; wunion -> {wunion_rec_fun(RecFunInfo), - clean_rec_args(FlatRecArgs)} + clean_rec_args(FlatRecArgs)}; + {map, MapKind} -> + {map_rec_fun(RecFunInfo, MapKind), clean_rec_args(FlatRecArgs)} end, {rec, NewRecFun, NewRecArgs} end. @@ -2287,6 +2368,37 @@ wunion_rec_fun({NumTypes,_NumRecs,RecArgLens,RecFuns}) -> proper_types:wunion(WeightedChoices) end. +map_rec_fun({_NumTypes, _NumRecs, RecArgLens, RecFuns}, required) -> + fun(AllGFs, Size) -> + GFsList = proper_arith:unflatten(AllGFs, RecArgLens), + ArgsList = [[GenFuns, Size] || GenFuns <- GFsList], + ZipFun = fun erlang:apply/2, + RawFields = lists:zipwith(ZipFun, RecFuns, ArgsList), + ?LET(Fields, RawFields, proper_type:fixed_map(maps:from_list(Fields))) + end; +map_rec_fun({_NumTypes, NumRecs, RecArgLens, RecFuns}, optional) -> + fun(AllGFs, TopSize) -> + Size = TopSize div NumRecs, + GFsList = proper_arith:unflatten(AllGFs, RecArgLens), + ArgsList = [[GenFuns, Size] || GenFuns <- GFsList], + ZipFun = fun erlang:apply/2, + RawFields = lists:zipwith(ZipFun, RecFuns, ArgsList), + ?LET(Fields, RawFields, proper_type:map(maps:from_list(Fields))) + end. + +combine_simple_map_fun({map, MapKind}) -> + GenFun = case MapKind of + required -> fun proper_types:fixed_map/1; + optional -> fun proper_types:map/1 + end, + fun(FieldTypes) -> + ?LET( + Fields, + FieldTypes, + GenFun(maps:from_list(lists:map(fun list_to_tuple/1, Fields))) + ) + end. + -spec add_ret_type(ret_type(), {[rec_fun()],[rec_args()],non_neg_integer()}) -> {[rec_fun()],[rec_args()],non_neg_integer()}. add_ret_type({simple,FinType}, {RecFuns,RecArgsList,NumRecs}) -> diff --git a/test/proper_exported_types_test.erl b/test/proper_exported_types_test.erl index 7b7e801b..cd455368 100644 --- a/test/proper_exported_types_test.erl +++ b/test/proper_exported_types_test.erl @@ -27,6 +27,8 @@ -module(proper_exported_types_test). -export([not_handled/0]). +-include("proper_internal.hrl"). + %% %% Checks that the automatic translation of types to generators can handle all %% types (structured + opaque) which are exported by some module of PropEr. @@ -40,9 +42,6 @@ %% %% Still, the test is currently not 100% there. %% TODOs: -%% - Eliminate the 12 errors that `proper_typeserver:demo_translate_type/2` -%% currently returns. (Three of these errors are due to the incomplete -%% handling of maps.) %% - Handle symbolic instances (the {'$call', ...} case below). %% @@ -63,8 +62,11 @@ pick_instance({M,T,A,{ok,Gen}}) -> %% catch _ -> io:format("~p~n", [{M,T,A}]), Inst end; _ -> case proper_typeserver:demo_is_instance(Inst, M, stringify(T, A)) of - true -> ok; - false -> {M,T,A,Inst,Gen} + true -> ok; + false -> {M,T,A,Inst,Gen}; + {error, Reason} -> + ?var(Inst), + error(Reason, [{M,T,A,{ok,Gen}}]) end end. diff --git a/test/proper_tests.erl b/test/proper_tests.erl index 6960872c..f6691135 100644 --- a/test/proper_tests.erl +++ b/test/proper_tests.erl @@ -410,6 +410,11 @@ simple_types_with_data() -> {[], [[]], [], [[a],[1,2,3]], "[]"}, {fixed_list([neg_integer(),pos_integer()]), [[-12,32],[-1,1]], [-1,1], [[0,0]], none}, + {map(#{key => value, pos_integer() => neg_integer()}), + [#{key => value, 1 => -1}], #{}, [not_a_map], none}, + {fixed_map(#{key => value, some_number => neg_integer()}), + [#{key => value, some_number => -3}], #{key => value, some_number => -1}, + [not_a_map], none}, {[atom(),integer(),atom(),float()], [[forty_two,42,forty_two,42.0]], ['',0,'',0.0], [[proper,is,licensed],[under,the,gpl]], none}, {[42 | list(integer())], [[42],[42,44,22]], [42], [[],[11,12]], none}, @@ -773,7 +778,7 @@ cant_generate_test_() -> [?_test(assert_cant_generate(Type)) || Type <- impossible_types()]. proper_exported_types_test_() -> - [?_assertEqual({[],12}, proper_exported_types_test:not_handled())]. + [?_assertEqual({[],0}, proper_exported_types_test:not_handled())]. %%------------------------------------------------------------------------------ %% Verify that failing constraints are correctly reported