diff --git a/src/jesse.erl b/src/jesse.erl index 71d6c63d..7365664d 100644 --- a/src/jesse.erl +++ b/src/jesse.erl @@ -88,13 +88,20 @@ | ?not_found ). +-type setter_fun() :: fun(( jesse_json_path:path() + , json_term() + , json_term() + ) -> json_term()) + | undefined. + -type option() :: {allowed_errors, allowed_errors()} | {default_schema_ver, schema_ver()} | {error_handler, error_handler()} | {external_validator, external_validator()} | {meta_schema_ver, schema_ver()} | {parser_fun, parser_fun()} - | {schema_loader_fun, schema_loader_fun()}. + | {schema_loader_fun, schema_loader_fun()} + | {setter_fun, setter_fun()}. -type options() :: [option()]. diff --git a/src/jesse_lib.erl b/src/jesse_lib.erl index d43ceec0..e5563724 100644 --- a/src/jesse_lib.erl +++ b/src/jesse_lib.erl @@ -27,6 +27,7 @@ -export([ empty_if_not_found/1 , is_array/1 , is_json_object/1 + , is_json_object_empty/1 , is_null/1 ]). @@ -86,3 +87,22 @@ is_null(null) -> is_null(_Value) -> false. +%% @doc check if json object is_empty. +-spec is_json_object_empty(Value :: any()) -> boolean(). +is_json_object_empty({struct, Value}) + when is_list(Value) andalso Value =:= [] -> + true; +is_json_object_empty({Value}) + when is_list(Value) + andalso Value =:= [] -> + true; +%% handle `jsx' empty objects +is_json_object_empty([{}]) -> + true; +?IF_MAPS( +is_json_object_empty(Map) + when erlang:is_map(Map) -> + maps:size(Map) =:= 0; +) +is_json_object_empty(_) -> + false. diff --git a/src/jesse_schema_validator.erl b/src/jesse_schema_validator.erl index 69611ae3..922526ea 100644 --- a/src/jesse_schema_validator.erl +++ b/src/jesse_schema_validator.erl @@ -40,10 +40,11 @@ , Options :: [{Key :: atom(), Data :: any()}] ) -> {ok, jesse:json_term()} | no_return(). -validate(JsonSchema, Value, Options) -> +validate(JsonSchema, Value, Options0) -> + Options = [{with_value, Value} | proplists:delete(with_value, Options0)], State = jesse_state:new(JsonSchema, Options), NewState = validate_with_state(JsonSchema, Value, State), - {result(NewState), Value}. + {result(NewState), jesse_state:get_current_value(NewState)}. %% @doc Validates json `Data' against `JsonSchema' with `State'. %% If the given json is valid, then the latest state is returned to the caller, diff --git a/src/jesse_schema_validator.hrl b/src/jesse_schema_validator.hrl index 2bcc4e17..ff753568 100644 --- a/src/jesse_schema_validator.hrl +++ b/src/jesse_schema_validator.hrl @@ -62,6 +62,7 @@ -define(MULTIPLEOF, <<"multipleOf">>). -define(MAXPROPERTIES, <<"maxProperties">>). -define(MINPROPERTIES, <<"minProperties">>). +-define(DEFAULT, <<"default">>). %% Constant definitions for Json types -define(ANY, <<"any">>). diff --git a/src/jesse_state.erl b/src/jesse_state.erl index fdd13064..843661cd 100644 --- a/src/jesse_state.erl +++ b/src/jesse_state.erl @@ -27,6 +27,7 @@ -export([ add_to_path/2 , get_allowed_errors/1 , get_external_validator/1 + , get_current_value/1 , get_current_path/1 , get_current_schema/1 , get_current_schema_id/1 @@ -37,11 +38,14 @@ , remove_last_from_path/1 , set_allowed_errors/2 , set_current_schema/2 + , set_value/3 , set_error_list/2 , resolve_ref/2 , undo_resolve_ref/2 , canonical_path/2 , combine_id/2 + , validator_options/1 + , validator_option/2, validator_option/3 ]). -export_type([ state/0 @@ -55,6 +59,7 @@ , { allowed_errors :: jesse:allowed_errors() , current_path :: current_path() , current_schema :: jesse:schema() + , current_value :: jesse:json_term() , default_schema_ver :: jesse:schema_ver() , error_handler :: jesse:error_handler() , error_list :: jesse:error_list() @@ -62,6 +67,8 @@ , id :: jesse:schema_id() , root_schema :: jesse:schema() , schema_loader_fun :: jesse:schema_loader_fun() + , setter_fun :: jesse:setter_fun() + , validator_options :: jesse:options() } ). @@ -146,6 +153,16 @@ new(JsonSchema, Options) -> , Options , ?default_schema_loader_fun ), + SetterFun = proplists:get_value( setter_fun + , Options + ), + Value = proplists:get_value( with_value + , Options + ), + ValidatorOptions = proplists:get_value( validator_options + , Options + , [] + ), NewState = #state{ root_schema = JsonSchema , current_path = [] , allowed_errors = AllowedErrors @@ -154,6 +171,9 @@ new(JsonSchema, Options) -> , default_schema_ver = DefaultSchemaVer , schema_loader_fun = LoaderFun , external_validator = ExternalValidator + , setter_fun = SetterFun + , current_value = Value + , validator_options = ValidatorOptions }, set_current_schema(NewState, JsonSchema). @@ -205,14 +225,22 @@ resolve_ref(State, Reference) -> Path = jesse_json_path:parse(Pointer), case load_local_schema(State#state.root_schema, Path) of ?not_found -> - jesse_error:handle_schema_invalid({?schema_not_found, CanonicalReference}, State); + jesse_error:handle_schema_invalid( { ?schema_not_found + , CanonicalReference + } + , State + ); Schema -> set_current_schema(State, Schema) end; false -> case load_schema(State, BaseURI) of ?not_found -> - jesse_error:handle_schema_invalid({?schema_not_found, CanonicalReference}, State); + jesse_error:handle_schema_invalid( { ?schema_not_found + , CanonicalReference + } + , State + ); RemoteSchema -> SchemaVer = jesse_json_path:value(?SCHEMA, RemoteSchema, ?default_schema_ver), @@ -223,7 +251,11 @@ resolve_ref(State, Reference) -> Path = jesse_json_path:parse(Pointer), case load_local_schema(RemoteSchema, Path) of ?not_found -> - jesse_error:handle_schema_invalid({?schema_not_found, CanonicalReference}, State); + jesse_error:handle_schema_invalid( { ?schema_not_found + , CanonicalReference + } + , State + ); Schema -> set_current_schema(NewState, Schema) end @@ -392,3 +424,32 @@ load_schema(#state{schema_loader_fun = LoaderFun}, SchemaURI) -> %% @private get_external_validator(#state{external_validator = Fun}) -> Fun. + +%% @doc Getter for `current_value'. +-spec get_current_value(State :: state()) -> jesse:json_term(). +get_current_value(#state{current_value = Value}) -> + Value. + +-spec set_value(State :: state(), jesse:path(), jesse:json_term()) -> state(). +set_value(#state{setter_fun = undefined}=State, _Path, _Value) -> State; +set_value(#state{current_value = undefined}=State, _Path, _Value) -> State; +set_value( #state{ setter_fun = Setter + , current_value = Value + } = State + , Path + , NewValue + ) -> + State#state{current_value = Setter(Path, NewValue, Value)}. + +-spec validator_options(State :: state()) -> jesse:options(). +validator_options(#state{validator_options = Options}) -> + Options. + +-spec validator_option(Option :: atom(), State :: state()) -> any(). +validator_option(Option, #state{validator_options = Options}) -> + proplists:get_value(Option, Options). + +-spec validator_option(Option :: atom(), State :: state(), Default :: any()) -> + any(). +validator_option(Option, #state{validator_options = Options}, Default) -> + proplists:get_value(Option, Options, Default). diff --git a/src/jesse_validator_draft3.erl b/src/jesse_validator_draft3.erl index 7a2b4ed2..5bbcfd4d 100644 --- a/src/jesse_validator_draft3.erl +++ b/src/jesse_validator_draft3.erl @@ -31,6 +31,7 @@ -include("jesse_schema_validator.hrl"). -type schema_error() :: ?wrong_type_dependency + | ?schema_invalid | ?wrong_type_items. -type schema_error_type() :: schema_error() @@ -348,20 +349,19 @@ check_properties(Value, Properties, State) -> = lists:foldl( fun({PropertyName, PropertySchema}, CurrentState) -> case get_value(PropertyName, Value) of ?not_found -> -%% @doc 5.7. required -%% -%% This attribute indicates if the instance must have a value, and not -%% be undefined. This is false by default, making the instance -%% optional. -%% @end - case get_value(?REQUIRED, PropertySchema) of - true -> - handle_data_invalid( {?missing_required_property - , PropertyName} - , Value - , CurrentState); - _ -> - CurrentState + case get_value(?DEFAULT, PropertySchema) of + ?not_found -> + check_required( PropertySchema + , PropertyName + , Value + , CurrentState + ); + Default -> + check_default( PropertyName + , PropertySchema + , Default + , CurrentState + ) end; Property -> NewState = set_current_schema( CurrentState @@ -583,6 +583,24 @@ check_items_fun(Tuples, State) -> ), set_current_schema(TmpState, get_current_schema(State)). + +%% @doc 5.7. required +%% +%% This attribute indicates if the instance must have a value, and not +%% be undefined. This is false by default, making the instance +%% optional. +%% @private +check_required(PropertySchema, PropertyName, Value, CurrentState) -> + case get_value(?REQUIRED, PropertySchema) of + true -> + handle_data_invalid( {?missing_required_property + , PropertyName} + , Value + , CurrentState); + _ -> + CurrentState + end. + %% @doc 5.8. dependencies %% %% This attribute is an object that defines the requirements of a @@ -904,7 +922,8 @@ validate_ref(Value, Reference, State) -> {error, NewState} -> undo_resolve_ref(NewState, State); {ok, NewState, Schema} -> - ResultState = jesse_schema_validator:validate_with_state(Schema, Value, NewState), + ResultState = + jesse_schema_validator:validate_with_state(Schema, Value, NewState), undo_resolve_ref(ResultState, State) end. @@ -992,7 +1011,11 @@ compare_properties(Value1, Value2) -> %% Wrappers %% @private get_value(Key, Schema) -> - jesse_json_path:value(Key, Schema, ?not_found). + get_value(Key, Schema, ?not_found). + +%% @private +get_value(Key, Schema, Default) -> + jesse_json_path:value(Key, Schema, Default). %% @private unwrap(Value) -> @@ -1041,3 +1064,80 @@ maybe_external_check_value(Value, State) -> Fun -> Fun(Value, State) end. + +%% @private +validator_option(Option, State, Default) -> + jesse_state:validator_option(Option, State, Default). + +%% @private +set_value(PropertyName, Value, State) -> + Path = lists:reverse([PropertyName] ++ jesse_state:get_current_path(State)), + jesse_state:set_value(State, Path, Value). + +%% @private +check_default_for_type(Default, State) -> + validator_option('use_defaults', State, false) + andalso (not jesse_lib:is_json_object(Default) + orelse validator_option( 'apply_defaults_to_empty_objects' + , State + , false + ) + orelse not jesse_lib:is_json_object_empty(Default)). + +%% @private +check_default(PropertyName, PropertySchema, Default, State) -> + Type = get_value(?TYPE, PropertySchema, ?not_found), + case is_valid_default(Type, Default, State) of + true -> + set_default(PropertyName, PropertySchema, Default, State); + false -> + State + end. + +%% @private +is_valid_default(?not_found, _Default, _State) -> + false; +is_valid_default(Type, Default, State) + when is_binary(Type) -> + check_default_for_type(Default, State) + andalso is_type_valid(Default, Type, State); +is_valid_default(Types, Default, State) + when is_list(Types) -> + check_default_for_type(Default, State) + andalso lists:any( fun(Type) -> + is_type_valid(Default, Type, State) + end + , Types + ); +is_valid_default(_, _Default, _State) -> false. + +%% @private +set_default(PropertyName, PropertySchema, Default, State) -> + State1 = set_value(PropertyName, Default, State), + State2 = add_to_path(State1, PropertyName), + case validate_schema(Default, PropertySchema, State2) of + {true, State4} -> + jesse_state:remove_last_from_path(State4); + _ -> + State + end. + +%% @doc Validate a value against a schema in a given state. +%% Used by all combinators to run validation on a schema. +%% @private +validate_schema(Value, Schema, State0) -> + try + case jesse_lib:is_json_object(Schema) of + true -> + State1 = set_current_schema(State0, Schema), + State2 = jesse_schema_validator:validate_with_state( Schema + , Value + , State1 + ), + {true, State2}; + false -> + handle_schema_invalid(?schema_invalid, State0) + end + catch + throw:Errors -> {false, Errors} + end. diff --git a/src/jesse_validator_draft4.erl b/src/jesse_validator_draft4.erl index 49c77903..8d8e0d2e 100644 --- a/src/jesse_validator_draft4.erl +++ b/src/jesse_validator_draft4.erl @@ -377,17 +377,26 @@ check_properties(Value, Properties, State) -> TmpState = lists:foldl( fun({PropertyName, PropertySchema}, CurrentState) -> case get_value(PropertyName, Value) of - ?not_found -> - CurrentState; - Property -> - NewState = set_current_schema( CurrentState - , PropertySchema - ), - check_value( PropertyName - , Property - , PropertySchema - , NewState - ) + ?not_found -> + case get_value(?DEFAULT, PropertySchema) of + ?not_found -> + CurrentState; + Default -> + check_default( PropertyName + , PropertySchema + , Default + , CurrentState + ) + end; + Property -> + NewState = set_current_schema( CurrentState + , PropertySchema + ), + check_value( PropertyName + , Property + , PropertySchema + , NewState + ) end end , State @@ -1226,7 +1235,8 @@ validate_ref(Value, Reference, State) -> {error, NewState} -> undo_resolve_ref(NewState, State); {ok, NewState, Schema} -> - ResultState = jesse_schema_validator:validate_with_state(Schema, Value, NewState), + ResultState = + jesse_schema_validator:validate_with_state(Schema, Value, NewState), undo_resolve_ref(ResultState, State) end. @@ -1314,7 +1324,11 @@ compare_properties(Value1, Value2) -> %% Wrappers %% @private get_value(Key, Schema) -> - jesse_json_path:value(Key, Schema, ?not_found). + get_value(Key, Schema, ?not_found). + +%% @private +get_value(Key, Schema, Default) -> + jesse_json_path:value(Key, Schema, Default). %% @private unwrap(Value) -> @@ -1355,6 +1369,10 @@ add_to_path(State, Property) -> remove_last_from_path(State) -> jesse_state:remove_last_from_path(State). +%% @private +validator_option(Option, State, Default) -> + jesse_state:validator_option(Option, State, Default). + %% @private valid_datetime(DateTimeBin) -> case rfc3339:parse(DateTimeBin) of @@ -1371,3 +1389,56 @@ maybe_external_check_value(Value, State) -> Fun -> Fun(Value, State) end. + +%% @private +set_value(PropertyName, Value, State) -> + Path = lists:reverse([PropertyName] ++ jesse_state:get_current_path(State)), + jesse_state:set_value(State, Path, Value). + +%% @private +check_default_for_type(Default, State) -> + validator_option('use_defaults', State, false) + andalso (not jesse_lib:is_json_object(Default) + orelse validator_option( 'apply_defaults_to_empty_objects' + , State + , false + ) + orelse not jesse_lib:is_json_object_empty(Default)). + +%% @private +check_default(PropertyName, PropertySchema, Default, State) -> + Type = get_value(?TYPE, PropertySchema, ?not_found), + case is_valid_default(Type, Default, State) of + true -> + set_default(PropertyName, PropertySchema, Default, State); + false -> + State + end. + +%% @private +is_valid_default(?not_found, _Default, _State) -> + false; +is_valid_default(Type, Default, State) + when is_binary(Type) -> + check_default_for_type(Default, State) + andalso is_type_valid(Default, Type); +is_valid_default(Types, Default, State) + when is_list(Types) -> + check_default_for_type(Default, State) + andalso lists:any( fun(Type) -> + is_type_valid(Default, Type) + end + , Types + ); +is_valid_default(_, _Default, _State) -> false. + +%% @private +set_default(PropertyName, PropertySchema, Default, State) -> + State1 = set_value(PropertyName, Default, State), + State2 = add_to_path(State1, PropertyName), + case validate_schema(Default, PropertySchema, State2) of + {true, State4} -> + jesse_state:remove_last_from_path(State4); + _ -> + State + end. diff --git a/test/jesse_schema_validator_tests.erl b/test/jesse_schema_validator_tests.erl index 44c67cbd..3a5ce9c2 100644 --- a/test/jesse_schema_validator_tests.erl +++ b/test/jesse_schema_validator_tests.erl @@ -21,6 +21,76 @@ -module(jesse_schema_validator_tests). -include_lib("eunit/include/eunit.hrl"). +setter_test() -> + Schema = {[ + {<<"type">>, <<"object">>}, + {<<"properties">>, {[ + {<<"bar">>, {[ + {<<"type">>, <<"string">>}, + {<<"minLength">>, 4}, + {<<"default">>, <<"awesome">>} + ]}} + ]}} + ]}, + + Default = {[{<<"bar">>, <<"awesome">>}]}, + Value = {[]}, + Fun = fun([K], V, {L1}) -> + {[{K, V} | proplists:delete(K, L1)]} + end, + Options = [ {setter_fun, Fun} + , {validator_options, [use_defaults]} + ], + + [ ?assertEqual( {ok, Value} + , jesse_schema_validator:validate(Schema, Value, []) + ) + , ?assertEqual( {ok, Default} + , jesse_schema_validator:validate(Schema, Value, Options) + ) + ]. + +invalid_default_test() -> + BadSchema = {[ + {<<"type">>, <<"object">>}, + {<<"properties">>, {[ + {<<"bar">>, {[ + {<<"type">>, <<"string">>}, + {<<"minLength">>, 4}, + {<<"default">>, <<"bad">>} + ]}} + ]}} + ]}, + + GoodSchema = {[ + {<<"type">>, <<"object">>}, + {<<"properties">>, {[ + {<<"bar">>, {[ + {<<"type">>, <<"string">>}, + {<<"minLength">>, 4}, + {<<"default">>, <<"awesome">>} + ]}} + ]}} + ]}, + + WithDefault = {[{<<"bar">>, <<"good">>}]}, + WithoutDefault = {[]}, + + ?assertEqual( + {ok, WithoutDefault}, + jesse_schema_validator:validate(BadSchema, WithoutDefault, []) + ), + + ?assertEqual( + {ok, WithDefault}, + jesse_schema_validator:validate(BadSchema, WithDefault, []) + ), + + ?assertEqual( + {ok, WithoutDefault}, + jesse_schema_validator:validate(GoodSchema, WithoutDefault, []) + ). + data_invalid_test() -> IntegerSchema = {[{<<"type">>, <<"integer">>}]},