Validation
Mantis provides a validation system that helps ensure your data meets specific criteria.
Basic Usage
Here's a simple example validating user input:
v
module main
import khalyomede.mantis.validation { validate, Rule, Value, Min }
fn main() {
data := {
"age": Value(21)
}
rules := {
"age": [
Rule(Min{18})
]
}
validate(data, rules) or {
// err is a ValidationError
eprintln(err.msg())
exit(1)
}
println("The age is correct.")
}
Struct validation
You can get back a struct when validating data. Useful for form validation for example.
v
module main
import validation { validate_struct, Rule, Value, Min, Max }
struct ContactForm {
email string
message string
}
fn main() {
email := "john.doe@example.com"
message := "Tech companies 2024 survey."
contact_form := validate_struct[ContactForm]({
"email": Value(email)
"message": Value(message)
}, {
"email": [
Rule(Min{3})
]
"message": [
Rule(Min{3})
Rule(Max{2000})
]
}) or { panic(err) }
assert contact_form.email == "john.doe@example.com"
assert contact_form.message == "Tech companies 2024 survey."
}
Custom rules
You can create your own rules.
They must implement two method: validate(Value) bool
and message(string) string
.
v
module main
import khalyomede.mantis.validation { Rule, Value }
// Custom email validation rule
struct Email {}
fn (rule Email) validate(value Value) bool {
return match value {
string {
// Simple email check for demonstration
value.contains('@') && value.contains('.')
}
else { false }
}
}
fn (rule Email) message(key string) string {
return "The ${key} must be a valid email address."
}
fn main() {
email := "john.doe@company.com"
data := {
"email": Value(email)
}
rules := {
"email": [
Rule(Email{})
]
}
validation.validate(data, rules) or {
eprintln(err.msg())
exit(1)
}
println("The email is correct.")
}
Combining multiple rules
You can chain multiple rules for a field. The catched error will correspond to the first rule that did not pass.
v
module main
import khalyomede.mantis.validation { validate, Rule, Value, Min, Max }
fn main() {
post := {
"title": Value("Structure and Interpretation of Computer Programs")
}
rules := {
"title": [
Rule(Min{5})
Rule(Max{100})
]
}
validate(post, rules) or {
eprintln(err.msg())
exit(1)
}
println("The post data are correct.")
}
HTTP Form validation example
v
module main
import khalyomede.mantis.http { create_app, App, Response }
import khalyomede.mantis.http.route
import khalyomede.mantis.validation { Rule, Value, Min, Max }
// Custom validation rule for passwords
struct StrongPassword {}
// 8 characters, with lower and uppercase
fn (rule StrongPassword) validate(value Value) bool {
return match value {
string {
value.len >= 8 &&
value.to_lower() != value &&
value.to_upper() != value
}
else { false }
}
}
fn (rule StrongPassword) message(key string) string {
return "The ${key} must be at least 8 characters and contain both uppercase and lowercase letters."
}
fn main() {
app := create_app(
routes: [
route.post(
name: "register"
path: "/register"
callback: fn (app App) !Response {
// Get form data
email := app.request.form("email") or { "" }
password := app.request.form("password") or { "" }
// Define validation rules
rules := {
"password": [
Rule(StrongPassword{})
]
}
input := {
"email": Value(email)
"password": Value(password)
}
// Validate input
validation.validate(input, rules)!
// If we reach here, validation passed
return app.response.html(
content: "Registration successful!"
status: .created
)
}
)
]
)
app.serve() or { panic(err) }
}
Built-in Rules
Min
Ensures the value has a minimum value.
v
module main
import khalyomede.mantis.validation { validate, Rule, Value, Min }
fn main() {
validate({
"age": Value(21)
}, {
"age": [
Rule(Min{18})
]
}) or {
eprintln(err.msg())
exit(1)
}
println("Age is correct.")
}
Max
Ensures the value does not pass over the given value.
v
module main
import khalyomede.mantis.validation { validate, Rule, Value, Max }
fn main() {
validate({
"age": Value(31)
}, {
"age": [
Rule(Max{100})
]
}) or {
eprintln(err.msg())
}
println("The age is correct.")
}