factor/extra/validators/validators.factor

143 lines
3.6 KiB
Factor
Raw Normal View History

! Copyright (C) 2006, 2008 Slava Pestov
! See http://factorcode.org/license.txt for BSD license.
USING: kernel continuations sequences math namespaces sets
math.parser assocs regexp fry unicode.categories sequences
2008-05-26 01:48:02 -04:00
arrays hashtables words combinators mirrors classes quotations
xmode.catalog ;
IN: validators
: v-default ( str def -- str )
over empty? spin ? ;
: v-required ( str -- str )
dup empty? [ "required" throw ] when ;
: v-optional ( str quot -- str )
over empty? [ 2drop f ] [ call ] if ; inline
: v-min-length ( str n -- str )
over length over < [
[ "must be at least " % # " characters" % ] "" make
throw
] [
drop
] if ;
: v-max-length ( str n -- str )
over length over > [
[ "must be no more than " % # " characters" % ] "" make
throw
] [
drop
] if ;
: v-number ( str -- n )
dup string>number [ ] [ "must be a number" throw ] ?if ;
2008-05-26 01:48:02 -04:00
: v-integer ( str -- n )
v-number dup integer? [ "must be an integer" throw ] unless ;
: v-min-value ( x n -- x )
2dup < [
[ "must be at least " % # ] "" make throw
] [
drop
] if ;
: v-max-value ( x n -- x )
2dup > [
[ "must be no more than " % # ] "" make throw
] [
drop
] if ;
: v-regexp ( str what regexp -- str )
>r over r> matches?
[ drop ] [ "invalid " prepend throw ] if ;
: v-email ( str -- str )
#! From http://www.regular-expressions.info/email.html
60 v-max-length
"e-mail"
R' [A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,4}'i
v-regexp ;
: v-url ( str -- str )
"URL"
R' (ftp|http|https)://(\w+:?\w*@)?(\S+)(:[0-9]+)?(/|/([\w#!:.?+=&%@!\-/]))?'
v-regexp ;
: v-captcha ( str -- str )
dup empty? [ "must remain blank" throw ] unless ;
: v-one-line ( str -- str )
2008-05-26 01:48:02 -04:00
v-required
dup "\r\n" intersect empty?
[ "must be a single line" throw ] unless ;
: v-one-word ( str -- str )
2008-05-26 01:48:02 -04:00
v-required
dup [ alpha? ] all?
[ "must be a single word" throw ] unless ;
2008-05-26 01:48:02 -04:00
: v-username ( str -- str )
2 v-min-length 16 v-max-length v-one-word ;
: v-password ( str -- str )
6 v-min-length 40 v-max-length v-one-line ;
: v-mode ( str -- str )
dup mode-names member? [
"not a valid syntax mode" throw
] unless ;
SYMBOL: validation-messages
2008-05-26 01:48:02 -04:00
SYMBOL: named-validation-messages
2008-05-26 01:48:02 -04:00
: init-validation ( -- )
V{ } clone validation-messages set
H{ } clone named-validation-messages set ;
: (validation-message) ( obj -- )
validation-messages get push ;
: (validation-message-for) ( obj name -- )
2008-05-26 01:48:02 -04:00
named-validation-messages get set-at ;
TUPLE: validation-message message ;
C: <validation-message> validation-message
: validation-message ( string -- )
<validation-message> (validation-message) ;
: validation-message-for ( string name -- )
[ <validation-message> ] dip (validation-message-for) ;
2008-05-26 01:48:02 -04:00
TUPLE: validation-error message value ;
C: <validation-error> validation-error
2008-05-26 01:48:02 -04:00
: validation-error ( message -- )
f <validation-error> (validation-message) ;
2008-05-26 01:48:02 -04:00
: validation-error-for ( message value name -- )
[ <validation-error> ] dip (validation-message-for) ;
: validation-failed? ( -- ? )
2008-05-26 01:48:02 -04:00
validation-messages get [ validation-error? ] contains?
named-validation-messages get [ nip validation-error? ] assoc-contains?
or ;
: define-validators ( class validators -- )
>hashtable "validators" set-word-prop ;
: validate ( value name quot -- result )
2008-05-26 01:48:02 -04:00
'[ drop @ ] [ -rot validation-error-for f ] recover ; inline
: required-values ( assoc -- )
2008-05-26 01:48:02 -04:00
[ swap [ v-required ] validate drop ] assoc-each ;
: validate-values ( assoc validators -- assoc' )
2008-05-26 01:48:02 -04:00
swap '[ [ [ dup , at ] keep ] dip validate ] assoc-map ;