%%%----------------------------------------------------------------------------
-module(hope_kv_list).
+-include_lib("hope_kv_list.hrl").
+
-behavior(hope_gen_dictionary).
-export_type(
, fold/3
, of_kv_list/1
, to_kv_list/1
- , validate_unique_presence/2 % Assume default optional parameter(s)
- , validate_unique_presence/3 % Specify optional parameter(s)
+ , find_unique_presence_violations/2 % No optional keys
+ , find_unique_presence_violations/3 % Specify optional keys
+ , validate_unique_presence/2 % No optional keys
+ , validate_unique_presence/3 % Specify optional keys
]).
-type t(K, V) ::
[{K, V}].
+-type presence_violations(A) ::
+ % This is a hack to effectively parametarize the types of record fields.
+ % IMPORTANT: Make sure that the order of fields matches the definition of
+ % #hope_kv_list_presence_violations
+ { hope_kv_list_presence_violations
+ , [A] % keys_missing
+ , [A] % keys_duplicated
+ , [A] % keys_unsupported
+ }.
+
-type presence_error(A) ::
{keys_missing , [A]}
| {keys_duplicated , [A]}
-spec validate_unique_presence(t(K, _V), [K], [K]) ->
hope_result:t(ok, [presence_error(K)]).
validate_unique_presence(T, KeysRequired, KeysOptional) ->
- KeysSupported = KeysRequired ++ KeysOptional,
- KeysGiven = [K || {K, _V} <- T],
- KeysGivenUnique = lists:usort(KeysGiven),
- KeysDups = lists:usort(KeysGiven -- KeysGivenUnique),
- KeysMissing = KeysRequired -- KeysGivenUnique,
- KeysUnsupported = KeysGivenUnique -- KeysSupported,
- case {KeysDups, KeysMissing, KeysUnsupported}
- of {[], [], []} ->
+ case find_unique_presence_violations(T, KeysRequired, KeysOptional)
+ of #hope_kv_list_presence_violations
+ { keys_missing = []
+ , keys_duplicated = []
+ , keys_unsupported = []
+ } ->
{ok, ok}
- ; {Dups, Missing, Unsupported} ->
- ErrorDups =
- case Dups
+ ; #hope_kv_list_presence_violations
+ { keys_missing = KeysMissing
+ , keys_duplicated = KeysDuplicated
+ , keys_unsupported = KeysUnsupported
+ } ->
+ ErrorMissing =
+ case KeysMissing
of [] -> []
- ; [_|_] -> [{keys_duplicated, Dups}]
+ ; [_|_] -> [{keys_missing, KeysMissing}]
end,
- ErrorMissing =
- case Missing
+ ErrorDups =
+ case KeysDuplicated
of [] -> []
- ; [_|_] -> [{keys_missing, Missing}]
+ ; [_|_] -> [{keys_duplicated, KeysDuplicated}]
end,
ErrorUnsupported =
- case Unsupported
+ case KeysUnsupported
of [] -> []
- ; [_|_] -> [{keys_unsupported, Unsupported}]
+ ; [_|_] -> [{keys_unsupported, KeysUnsupported}]
end,
Errors = ErrorDups ++ ErrorMissing ++ ErrorUnsupported,
{error, Errors}
end.
+-spec find_unique_presence_violations(t(K, _V), [K]) ->
+ presence_violations(K).
+find_unique_presence_violations(T, KeysRequired) ->
+ KeysOptional = [],
+ find_unique_presence_violations(T, KeysRequired, KeysOptional).
+
+-spec find_unique_presence_violations(t(K, _V), [K], [K]) ->
+ presence_violations(K).
+find_unique_presence_violations(T, KeysRequired, KeysOptional) ->
+ KeysSupported = KeysRequired ++ KeysOptional,
+ KeysGiven = [K || {K, _V} <- T],
+ KeysGivenUnique = lists:usort(KeysGiven),
+ KeysDuplicated = lists:usort(KeysGiven -- KeysGivenUnique),
+ KeysMissing = KeysRequired -- KeysGivenUnique,
+ KeysUnsupported = KeysGivenUnique -- KeysSupported,
+ #hope_kv_list_presence_violations
+ { keys_missing = KeysMissing
+ , keys_duplicated = KeysDuplicated
+ , keys_unsupported = KeysUnsupported
+ }.
+
%% ============================================================================
%% Helpers
-module(hope_kv_list_SUITE).
+-include_lib("hope_kv_list.hrl").
+
%% Callbacks
-export(
[ all/0
DictUnsup = [{a, 1}, {b, 2}, {c, 3}, {d, 4}],
DictDups = [{a, 1}, {b, 2}, {c, 3}, {a, 4}],
DictMissing = [{a, 1}, {b, 2}],
+
{ok, ok} =
hope_kv_list:validate_unique_presence(DictOk, KeysRequired),
+ #hope_kv_list_presence_violations
+ { keys_missing = []
+ , keys_duplicated = []
+ , keys_unsupported = []
+ } =
+ hope_kv_list:find_unique_presence_violations(DictOk, KeysRequired),
+
{error, [{keys_unsupported, [d]}]} =
hope_kv_list:validate_unique_presence(DictUnsup, KeysRequired),
+ #hope_kv_list_presence_violations
+ { keys_missing = []
+ , keys_duplicated = []
+ , keys_unsupported = [d]
+ } =
+ hope_kv_list:find_unique_presence_violations(DictUnsup, KeysRequired),
+
{error, [{keys_duplicated, [a]}]} =
hope_kv_list:validate_unique_presence(DictDups, KeysRequired),
+ #hope_kv_list_presence_violations
+ { keys_missing = []
+ , keys_duplicated = [a]
+ , keys_unsupported = []
+ } =
+ hope_kv_list:find_unique_presence_violations(DictDups, KeysRequired),
+
{error, [{keys_missing, [c]}]} =
- hope_kv_list:validate_unique_presence(DictMissing, KeysRequired).
+ hope_kv_list:validate_unique_presence(DictMissing, KeysRequired),
+ #hope_kv_list_presence_violations
+ { keys_missing = [c]
+ , keys_duplicated = []
+ , keys_unsupported = []
+ } =
+ hope_kv_list:find_unique_presence_violations(DictMissing, KeysRequired).