You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Very cool project, thank you. Given the integration with ktor, are you planning any methods to validate incoming form, query or path parameters? This library assumes the object can be constructed, then performs validation checks (field is greater than zero, string matches a regex pattern, etc). But due to bugs, malicious actors, and incorrect input, lots of times the supplied data is insufficient to instantiate an object instance, and attempting to do so throws an error before we can start any validation.
For my current project I wrote some custom validators (see example at bottom) that walk through all of the possible input parameters and individually check them all, but this has a few problems:
only way to instantiate this is via a manual function call (not with @Validate decorator),
the collection constraints like .isContaining don't work on ktor's Parameters - almost all constraints are custom
all params are strings, so numeric comparisons require conversion first
regarding above, validation throws an error (instead of ConstraintViolation) when parameter is not the correct type ("eleven" not "11", bad date format, etc) unless you specifically test for that (and must "fail fast" if you also plan to validate the resulting value)
building a validator for the object itself is duplicitous, since the input parameters have already been tested
validating optional params is burdensome - have to check if it exists, after that all references require !!
essentially, the validation result is just a boolean value (Success/Failure), not the resulting object (to call methods on, pass to another object, etc)
hard to say if this is any easier/safer than just using a bunch of require {} in the object init
I haven't thought this through entirely, but maybe some kind of workflow where the validator would attempt to create an object from the submitted params, if it succeeds, then we perform the normal object validation (via @Validate), but if the object creation fails then we redirect to an alternate set of constraints on the individual parameters to help determine why the instantiation failed (param was missing, wrong type, unparsable, etc)
Another idea would be to have constraints that attempt to coerce a string value to another type (LocalDate, URI, etc). I think this is preferable to Regex matching due to different date patterns in different locales (same with phone numbers, postal codes, etc), and the complexity of a regex validators for URIs, emails, etc. Also works for enums.
Again, thanks for a cool library, looking forward to seeing it grow.
example custom validator:
// validate incoming query params for / route (allows user to bookmark report with certain defaults)val validateOrdersQueryParams =Validator<Parameters>(akkurateConfig) {
val params :Parameters=this.unwrap()
if (params.contains("start_date")) this.constrain { noError { LocalDate.parse(it["start_date"]!!) } } otherwise { "start_date can't be parsed" }
if (params.contains("end_date")) this.constrain { noError { LocalDate.parse(it["end_date"]!!) } } otherwise { "end_date can't be parsed" }
if (params.contains("start_date") && params.contains("end_date")) this.constrain {
!(LocalDate.parse(it["start_date"]!!).isAfter(LocalDate.parse(it["end_date"]!!)))
} otherwise { "start_date is after end_date" }
if (params.contains("sales_min")) {
this.constrain { noError { it["sales_min"]!!.toInt() } } otherwise { "sales_min is non-numeric" }
this.constrain { it["sales_min"]!!.toInt() >=0 } otherwise { "sales_min is out of range" }
}
}
// attempts to run a function and returns true/false whether it completes without errorfunnoError(testFunction: () ->Any) : Boolean {
val result:Result<Boolean> =try {
testFunction()
Result.success(true)
} catch (e:Exception) {
Result.failure(e)
}
returnwhen {
result.isSuccess ->true
result.isFailure ->falseelse->false
}
}
The text was updated successfully, but these errors were encountered:
Very cool project, thank you. Given the integration with ktor, are you planning any methods to validate incoming form, query or path parameters? This library assumes the object can be constructed, then performs validation checks (field is greater than zero, string matches a regex pattern, etc). But due to bugs, malicious actors, and incorrect input, lots of times the supplied data is insufficient to instantiate an object instance, and attempting to do so throws an error before we can start any validation.
For my current project I wrote some custom validators (see example at bottom) that walk through all of the possible input parameters and individually check them all, but this has a few problems:
.isContaining
don't work on ktor'sParameters
- almost all constraints are custom!!
require {}
in the objectinit
I haven't thought this through entirely, but maybe some kind of workflow where the validator would attempt to create an object from the submitted params, if it succeeds, then we perform the normal object validation (via @Validate), but if the object creation fails then we redirect to an alternate set of constraints on the individual parameters to help determine why the instantiation failed (param was missing, wrong type, unparsable, etc)
Another idea would be to have constraints that attempt to coerce a string value to another type (LocalDate, URI, etc). I think this is preferable to Regex matching due to different date patterns in different locales (same with phone numbers, postal codes, etc), and the complexity of a regex validators for URIs, emails, etc. Also works for enums.
Again, thanks for a cool library, looking forward to seeing it grow.
example custom validator:
The text was updated successfully, but these errors were encountered: