1 ;;; rng-xsd.el --- W3C XML Schema datatypes library for RELAX NG
3 ;; Copyright (C) 2003, 2007-2013 Free Software Foundation, Inc.
6 ;; Keywords: XML, RelaxNG
8 ;; This file is part of GNU Emacs.
10 ;; GNU Emacs is free software: you can redistribute it and/or modify
11 ;; it under the terms of the GNU General Public License as published by
12 ;; the Free Software Foundation, either version 3 of the License, or
13 ;; (at your option) any later version.
15 ;; GNU Emacs is distributed in the hope that it will be useful,
16 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
17 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 ;; GNU General Public License for more details.
20 ;; You should have received a copy of the GNU General Public License
21 ;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
25 ;; The main entry point is `rng-xsd-compile'. The validator
26 ;; knows to use this for the datatype library with URI
27 ;; http://www.w3.org/2001/XMLSchema-datatypes because it
28 ;; is the value of the rng-dt-compile property on that URI
31 ;; W3C XML Schema Datatypes are specified by
32 ;; http://www.w3.org/TR/xmlschema-2/
33 ;; Guidelines for using them with RELAX NG are described in
34 ;; http://relaxng.org/xsd.html
43 (put 'http
://www.w3.org
/2001/XMLSchema-datatypes
48 (defun rng-xsd-compile (name params
)
49 "Provides W3C XML Schema as a RELAX NG datatypes library.
50 NAME is a symbol giving the local name of the datatype. PARAMS is a
51 list of pairs (PARAM-NAME . PARAM-VALUE) where PARAM-NAME is a symbol
52 giving the name of the parameter and PARAM-VALUE is a string giving
53 its value. If NAME or PARAMS are invalid, it calls rng-dt-error
54 passing it arguments in the same style as format; the value from
55 rng-dt-error will be returned. Otherwise, it returns a list. The
56 first member of the list is t if any string is a legal value for the
57 datatype and nil otherwise. The second argument is a symbol; this
58 symbol will be called as a function passing it a string followed by
59 the remaining members of the list. The function must return an object
60 representing the value of the datatype that was represented by the
61 string, or nil if the string is not a representation of any value.
62 The object returned can be any convenient non-nil value, provided
63 that, if two strings represent the same value, the returned objects
65 (let ((convert (get name
'rng-xsd-convert
)))
67 (rng-dt-error "There is no XSD datatype named %s" name
)
68 (rng-xsd-compile1 name params convert
))))
72 (defun rng-xsd-compile1 (name params convert
)
74 (cons (equal convert
'(identity))
75 (cond ((eq name
'string
) convert
)
76 ((eq name
'normalizedString
)
77 (cons 'rng-xsd-replace-space convert
))
78 ((and (not (eq name
'string
))
79 (or (memq 'identity convert
)
80 (memq 'rng-xsd-convert-any-uri convert
)
81 (memq 'rng-xsd-check-pattern convert
)))
82 (cons 'rng-xsd-collapse-space convert
))
84 (let* ((param (car params
))
85 (param-name (car param
))
86 (param-value (cdr param
)))
87 (cond ((memq param-name
88 '(minExclusive maxExclusive minInclusive maxInclusive
))
89 (let ((limit (apply (car convert
)
92 (less-than-fun (get name
'rng-xsd-less-than
)))
94 (rng-dt-error "Minimum value %s is not valid"
97 (rng-dt-error "Values of type %s are not ordered"
100 (rng-xsd-compile1 name
102 (cons (get param-name
105 (cons limit convert
))))))))
106 ((memq param-name
'(length minLength maxLength
))
107 (let ((limit (rng-xsd-string-to-non-negative-integer param-value
))
108 (length-fun (get name
'rng-xsd-length
)))
110 (rng-dt-error "Length %s is not valid" param-value
))
112 (rng-dt-error "Values of type %s do not have a length"
115 (rng-xsd-compile1 name
117 (cons (get param-name
120 (cons limit convert
))))))))
121 ((memq param-name
'(fractionDigits totalDigits
))
122 (let ((n (rng-xsd-string-to-non-negative-integer param-value
)))
124 (rng-dt-error "Number of digits %s is not valid"
127 (rng-xsd-compile1 name
129 (cons (get param-name
131 (cons n convert
)))))))
132 ((eq param-name
'pattern
)
134 (rng-xsd-compile1 name
136 (cons 'rng-xsd-check-pattern
139 (xsdre-translate param-value
)
142 (xsdre-invalid-regexp
143 (rng-dt-error "Invalid regular expression (%s)"
145 ((memq param-name
'(enumeration whiteSpace
))
146 (rng-dt-error "Facet %s cannot be used in RELAX NG" param-name
))
147 (t (rng-dt-error "Unknown facet %s" param-name
))))))
149 (defun rng-xsd-string-to-non-negative-integer (str)
150 (and (rng-xsd-convert-integer str
)
151 (let ((n (string-to-number str
)))
156 (defun rng-xsd-collapse-space (str convert
&rest args
)
157 (apply convert
(cons (mapconcat 'identity
(split-string str
"[ \t\n\r]+")
161 (defun rng-xsd-replace-space (str convert
&rest args
)
165 (while (and (setq i
(string-match "[\r\n\t]" str i
))
166 (or copied
(setq copied
(copy-sequence str
)))
172 (put 'minExclusive
'rng-xsd-check
'rng-xsd-check-min-exclusive
)
173 (put 'minInclusive
'rng-xsd-check
'rng-xsd-check-min-inclusive
)
174 (put 'maxExclusive
'rng-xsd-check
'rng-xsd-check-max-exclusive
)
175 (put 'maxInclusive
'rng-xsd-check
'rng-xsd-check-max-inclusive
)
176 (put 'length
'rng-xsd-check
'rng-xsd-check-length
)
177 (put 'minLength
'rng-xsd-check
'rng-xsd-check-min-length
)
178 (put 'maxLength
'rng-xsd-check
'rng-xsd-check-max-length
)
179 (put 'fractionDigits
'rng-xsd-check
'rng-xsd-check-fraction-digits
)
180 (put 'totalDigits
'rng-xsd-check
'rng-xsd-check-total-digits
)
182 (defun rng-xsd-check-min-exclusive (str less-than-fun limit convert
&rest args
)
183 (let ((obj (apply convert
(cons str args
))))
185 (funcall less-than-fun limit obj
)
188 (defun rng-xsd-check-min-inclusive (str less-than-fun limit convert
&rest args
)
189 (let ((obj (apply convert
(cons str args
))))
191 (or (funcall less-than-fun limit obj
)
195 (defun rng-xsd-check-max-exclusive (str less-than-fun limit convert
&rest args
)
196 (let ((obj (apply convert
(cons str args
))))
198 (funcall less-than-fun obj limit
)
201 (defun rng-xsd-check-max-inclusive (str less-than-fun limit convert
&rest args
)
202 (let ((obj (apply convert
(cons str args
))))
204 (or (funcall less-than-fun obj limit
)
208 (defun rng-xsd-check-min-length (str length-fun limit convert
&rest args
)
209 (let ((obj (apply convert
(cons str args
))))
211 (>= (funcall length-fun obj
) limit
)
214 (defun rng-xsd-check-max-length (str length-fun limit convert
&rest args
)
215 (let ((obj (apply convert
(cons str args
))))
217 (<= (funcall length-fun obj
) limit
)
220 (defun rng-xsd-check-length (str length-fun len convert
&rest args
)
221 (let ((obj (apply convert
(cons str args
))))
223 (= (funcall length-fun obj
) len
)
226 (defun rng-xsd-check-fraction-digits (str n convert
&rest args
)
227 (let ((obj (apply convert
(cons str args
))))
229 (<= (length (aref obj
2)) n
)
232 (defun rng-xsd-check-total-digits (str n convert
&rest args
)
233 (let ((obj (apply convert
(cons str args
))))
235 (<= (+ (length (aref obj
1))
236 (length (aref obj
2)))
240 (defun rng-xsd-check-pattern (str regexp convert
&rest args
)
241 (and (let ((case-fold-search nil
)) (string-match regexp str
))
242 (apply convert
(cons str args
))))
245 (defun rng-xsd-convert-boolean (string)
246 (and (string-match "\\`[ \t\n\r]*\\(?:\\(true\\|1\\)\\|false\\|0\\)[ \t\n\r]*\\'" string
)
247 (if (match-beginning 1) 'true
'false
)))
249 (defun rng-xsd-convert-decimal (string)
250 "Convert a string representing a decimal to an object representing it values.
251 A decimal value is represented by a vector [SIGN INTEGER-DIGITS
252 FRACTION-DIGITS] where SIGN is 1 or -1, INTEGER-DIGITS is a string
253 containing zero or more digits, with no leading zero, and
254 FRACTION-DIGITS is a string containing zero or more digits with no
255 trailing digits. For example, -0021.0430 would be represented by [-1
257 (and (string-match "\\`[ \t\n\r]*\\([-+]\\)?\\(0*\\([1-9][0-9]*\\)?\\(\\.\\([0-9]*[1-9]\\)?0*\\)?\\)[ \t\n\r]*\\'" string
)
258 (let ((digits (match-string 2 string
)))
259 (and (not (string= digits
"."))
260 (not (string= digits
""))))
261 (let ((integer-digits (match-string 3 string
)))
262 (vector (if (and (equal (match-string 1 string
) "-")
267 (or integer-digits
"")
268 (or (match-string 5 string
) "")))))
270 (defun rng-xsd-convert-integer (string)
271 (and (string-match "\\`[ \t\n\r]*\\([-+]\\)?\\(?:0*\\([1-9][0-9]*\\)\\|0+\\)[ \t\n\r]*\\'" string
)
272 (let ((integer-digits (match-string 2 string
)))
273 (vector (if (and (equal (match-string 1 string
) "-")
278 (or integer-digits
"")
281 (defun rng-xsd-decimal< (n1 n2
)
282 (< (rng-xsd-compare-decimal n1 n2
) 0))
284 (defun rng-xsd-compare-decimal (n1 n2
)
285 "Return a < 0, 0, > 0 according as n1 < n2, n1 = n2 or n1 > n2."
286 (let* ((sign1 (aref n1
0))
288 (sign (- sign1 sign2
)))
291 (let* ((int1 (aref n1
1))
295 (lencmp (- len1 len2
)))
297 (if (string= int1 int2
)
298 (rng-xsd-strcmp (aref n1
2) (aref n2
2))
299 (rng-xsd-strcmp int1 int2
))
303 (defconst rng-xsd-float-regexp
304 (concat "\\`[ \r\n\t]*\\(?:"
306 "[-+]?\\(?:[0-9]+\\(?:\\.[0-9]*\\)?\\|\\.[0-9]+\\)"
307 "\\(?:[eE][-+]?[0-9]+\\)?"
314 (defun rng-xsd-convert-float (string)
315 (cond ((not (string-match rng-xsd-float-regexp string
)) nil
)
317 (float (string-to-number (match-string 1 string
))))
318 ((match-beginning 2) 1.0e
+INF
)
319 ((match-beginning 3) -
1.0e
+INF
)
320 ;; Don't use a NaN float because we want NaN to be equal to NaN
321 ((match-beginning 4) 'NaN
)))
323 (defun rng-xsd-float< (f1 f2
)
324 (and (not (eq f1
'NaN
))
328 (defun rng-xsd-convert-token (string regexp
)
329 (and (string-match regexp string
)
330 (match-string 1 string
)))
332 (defun rng-xsd-convert-hex-binary (string)
333 (and (string-match "\\`[ \r\n\t]*\\(\\(?:[0-9A-Fa-f][0-9A-Fa-f]\\)*\\)[ \r\n\t]*\\'"
335 (downcase (match-string 1 string
))))
337 (defun rng-xsd-hex-binary-length (obj)
340 (defconst rng-xsd-base64-binary-regexp
341 (let ((S "[ \t\r\n]*")
343 (B16 "[AEIMQUYcgkosw048]")
344 (B64 "[A-Za-z0-9+/]"))
345 (concat "\\`" S
"\\(?:\\(?:" B64 S
"\\)\\{4\\}\\)*"
346 "\\(?:" B64 S B64 S B16 S
"=" S
347 "\\|" B64 S B04 S
"=" S
"=" S
"\\)?\\'")))
349 (defun rng-xsd-convert-base64-binary (string)
350 (and (string-match rng-xsd-base64-binary-regexp string
)
351 (replace-regexp-in-string "[ \t\r\n]+" "" string t t
)))
353 (defun rng-xsd-base64-binary-length (obj)
354 (let ((n (* (/ (length obj
) 4) 3)))
356 (string= (substring obj -
1) "="))
357 (- n
(if (string= (substring obj -
2) "==")
362 (defun rng-xsd-convert-any-uri (string)
363 (and (string-match "\\`\\(?:[^%]\\|%[0-9a-fA-F][0-9a-fA-F]\\)?*\\'" string
)
364 (string-match "\\`[^#]*\\(?:#[^#]*\\)?\\'" string
)
365 (string-match "\\`\\(?:[a-zA-Z][-+.A-Za-z0-9]*:.+\\|[^:]*\\(?:[#/?].*\\)?\\)\\'" string
)
368 (defun rng-xsd-make-date-time-regexp (template)
369 "Returns a regular expression matching a ISO 8601 date/time.
370 The template is a string with Y standing for years field, M standing
371 for months, D standing for day of month, T standing for a literal T, t
372 standing for time and - standing for a literal hyphen. A time zone is
373 always allowed at the end. Regardless of the fields appearing in the
374 template, the regular expression will have twelve groups matching the
375 year sign, year, month, day of month, hours, minutes, integer seconds,
376 fractional seconds (including leading period), time zone, time zone
377 sign, time zone hours, time zone minutes."
379 (len (length template
))
383 (setq c
(aref template i
))
385 (cons (cond ((eq c ?Y
)
388 "\\(-\\)?\\(\\(?:[1-9][0-9]*\\)?[0-9]\\{4\\}\\)")
403 "\\([0-9][0-9]\\):\\([0-9][0-9]\\):\\([0-9][0-9]\\)\\(\\.[0-9]*\\)?")
408 (setq last
(1+ last
))
409 ;; Add dummy fields that can never much but keep the group
411 (setq parts
(cons "\\(\\'X\\)?" parts
)))
412 (setq parts
(cons "\\(Z\\|\\([-+]\\)\\([0-9][0-9]\\):\\([0-5][0-9]\\)\\)?[ \t\n\r]*\\'"
414 (setq parts
(cons "\\`[ \t\n\r]*" (nreverse parts
)))
416 (setq first
(1- first
))
417 (setq parts
(cons "\\(X\\)?" parts
)))
418 (apply 'concat parts
)))
420 (defconst rng-xsd-seconds-per-day
(* 24 60 60))
421 (defconst rng-xsd-days-in-month
[31 28 31 30 31 30 31 31 30 31 30 31])
423 (defun rng-xsd-days-in-month (year month
)
424 (if (and (= month
2) (rng-xsd-leap-year-p year
))
426 (aref rng-xsd-days-in-month
(1- month
))))
428 (defconst rng-xsd-months-to-days
429 (let ((v (make-vector 12 nil
))
433 (setq total
(+ total
(aref rng-xsd-days-in-month i
)))
438 (defun rng-xsd-convert-date-time (string regexp
)
439 "Converts an XML Schema date/time to a list.
440 Returns nil if invalid. REGEXP is a regexp for parsing the date time
441 as returned by `rng-xsd-make-date-time-regexp'. The list has 4 members
442 \(HAS-TIME-ZONE DAY SECOND SECOND-FRACTION), where HAS-TIME-ZONE is t
443 or nil depending on whether a time zone was specified, DAY is an
444 integer giving a day number (with Jan 1 1AD being day 1), SECOND is the
445 second within that day, and SECOND-FRACTION is a float giving the
446 fractional part of the second."
447 (and (string-match regexp string
)
448 (let ((year-sign (match-string 1 string
))
449 (year (match-string 2 string
))
450 (month (match-string 3 string
))
451 (day (match-string 4 string
))
452 (hour (match-string 5 string
))
453 (minute (match-string 6 string
))
454 (second (match-string 7 string
))
455 (second-fraction (match-string 8 string
))
456 (has-time-zone (match-string 9 string
))
457 (time-zone-sign (match-string 10 string
))
458 (time-zone-hour (match-string 11 string
))
459 (time-zone-minute (match-string 12 string
)))
460 (setq year-sign
(if year-sign -
1 1))
464 (string-to-number year
))
467 (if month
(string-to-number month
) 1))
469 (if day
(string-to-number day
) 1))
471 (if hour
(string-to-number hour
) 0))
473 (if minute
(string-to-number minute
) 0))
475 (if second
(string-to-number second
) 0))
476 (setq second-fraction
478 (float (string-to-number second-fraction
))
480 (setq has-time-zone
(and has-time-zone t
))
482 (if (equal time-zone-sign
"-") -
1 1))
484 (if time-zone-hour
(string-to-number time-zone-hour
) 0))
485 (setq time-zone-minute
486 (if time-zone-minute
(string-to-number time-zone-minute
) 0))
490 (<= day
(rng-xsd-days-in-month year month
))
493 (<= second
60) ; leap second
494 (<= time-zone-hour
23)
495 (<= time-zone-minute
59)
498 (list (rng-xsd-date-to-days year month day
)
499 (rng-xsd-time-to-seconds hour minute second
)
501 (* (rng-xsd-time-to-seconds time-zone-hour
504 (- time-zone-sign
))))))))
506 (defun rng-xsd-leap-year-p (year)
507 (and (= (% year
4) 0)
508 (or (/= (% year
100) 0)
509 (= (% year
400) 0))))
511 (defun rng-xsd-time-to-seconds (hour minute second
)
517 (defconst rng-xsd-max-tz
(rng-xsd-time-to-seconds 14 0 0))
519 (defun rng-xsd-date-time< (dt1 dt2
)
520 (cond ((eq (car dt1
) (car dt2
))
521 (rng-xsd-number-list< (cdr dt1
) (cdr dt2
)))
523 (rng-xsd-number-list< (cdr dt1
)
524 (rng-xsd-add-seconds (cdr dt2
)
525 (- rng-xsd-max-tz
))))
527 (rng-xsd-number-list< (rng-xsd-add-seconds (cdr dt1
)
531 (defun rng-xsd-add-seconds (date offset
)
532 (let ((day (nth 0 date
))
533 (second (+ (nth 1 date
) offset
))
534 (fraction (nth 2 date
)))
537 (+ second rng-xsd-seconds-per-day
)
539 ((>= second rng-xsd-seconds-per-day
)
541 (- second rng-xsd-seconds-per-day
)
543 (t (list day second fraction
)))))
545 (defun rng-xsd-number-list< (numbers1 numbers2
)
546 (while (and numbers1
(= (car numbers1
) (car numbers2
)))
547 (setq numbers1
(cdr numbers1
))
548 (setq numbers2
(cdr numbers2
)))
550 (< (car numbers1
) (car numbers2
))))
552 (defun rng-xsd-date-to-days (year month day
)
553 "Return a unique day number where Jan 1 1 AD is day 1"
555 (+ (rng-xsd-days-in-years (- year
1))
556 (rng-xsd-day-number-in-year year month day
))
557 (- (+ (- (rng-xsd-days-in-years (- 3 year
))
558 (rng-xsd-days-in-years 3))
559 (- (if (rng-xsd-leap-year-p year
) 366 365)
560 (rng-xsd-day-number-in-year year month day
))))))
562 (defun rng-xsd-days-in-years (years)
563 "The number of days in YEARS years where the first year is 1AD."
569 (defun rng-xsd-day-number-in-year (year month day
)
572 (aref rng-xsd-months-to-days
(- month
2)))
575 (rng-xsd-leap-year-p year
))
579 (defconst rng-xsd-duration-regexp
580 "\\`[ \t\r\n]*\\(-\\)?P\
581 \\([0-9]+Y\\)?\\([0-9]+M\\)?\\([0-9]+D\\)?\
582 \\(?:T\\([0-9]+H\\)?\\([0-9]+M\\)?\
583 \\(\\([0-9]+\\(?:\\.[0-9]*\\)?\\|\\.[0-9]+\\)S\\)?\\)?\
587 (defun rng-xsd-convert-duration (string)
588 (and (string-match rng-xsd-duration-regexp string
)
589 (let ((last (substring string -
1)))
590 (not (or (string= last
"P")
591 (string= last
"T"))))
592 ;; years months days hours minutes seconds
593 (let ((v (make-vector 6 0))
594 (sign (if (match-beginning 1) -
1 1))
597 (let ((start (match-beginning (+ i
2))))
603 (1- (match-end (+ i
2)))))))))
605 ;; Force seconds to be float so that equal works properly.
606 (aset v
5 (float (aref v
5)))
609 (defconst rng-xsd-min-seconds-per-month
(* 28 rng-xsd-seconds-per-day
))
611 (defun rng-xsd-duration< (d1 d2
)
612 (let* ((months1 (rng-xsd-duration-months d1
))
613 (months2 (rng-xsd-duration-months d2
))
614 (seconds1 (rng-xsd-duration-seconds d1
))
615 (seconds2 (rng-xsd-duration-seconds d2
)))
616 (cond ((< months1 months2
)
617 (if (< (- seconds1 seconds2
) rng-xsd-min-seconds-per-month
)
619 (rng-xsd-months-seconds< months1 seconds1 months2 seconds2
)))
621 (if (< (- seconds2 seconds1
) rng-xsd-min-seconds-per-month
)
623 (rng-xsd-months-seconds< months1 seconds1 months2 seconds2
)))
624 (t (< seconds1 seconds2
)))))
626 (defconst xsd-duration-reference-dates
627 '((1696 .
9) (1697 .
2) (1903 .
3) (1903 .
7)))
629 (defun rng-xsd-months-seconds< (months1 seconds1 months2 seconds2
)
631 (ref-dates xsd-duration-reference-dates
))
632 (while (let* ((ref-date (car ref-dates
))
633 (ref-year (car ref-date
))
634 (ref-month (cdr ref-date
)))
635 (unless (< (+ (rng-xsd-month-seconds months1
639 (+ (rng-xsd-month-seconds months2
645 (setq ref-dates
(cdr ref-dates
)))))
649 (defun rng-xsd-month-seconds (months ref-year ref-month
)
650 "Return the seconds in a number of months starting on a reference date.
651 Returns a floating point number."
652 (* (rng-xsd-month-days (abs months
) ref-year ref-month
)
653 (float rng-xsd-seconds-per-day
)
654 (if (< months
0) -
1.0 1.0)))
656 (defconst rng-xsd-years-per-gregorian-cycle
400)
657 (defconst rng-xsd-months-per-gregorian-cycle
658 (* rng-xsd-years-per-gregorian-cycle
12))
659 (defconst rng-xsd-leap-years-per-gregorian-cycle
(- 100 (- 4 1)))
660 (defconst rng-xsd-days-per-gregorian-cycle
661 (+ (* 365 rng-xsd-years-per-gregorian-cycle
)
662 rng-xsd-leap-years-per-gregorian-cycle
))
664 (defun rng-xsd-month-days (months ref-year ref-month
)
665 "Return the days in a number of months starting on a reference date.
666 MONTHS must be an integer >= 0."
668 (setq months
(mod months rng-xsd-months-per-gregorian-cycle
))
669 ;; This may be rather slow, but it is highly unlikely
670 ;; ever to be used in real life.
673 (+ (rng-xsd-days-in-month ref-year ref-month
)
676 (if (eq ref-month
12)
678 (setq ref-year
(1+ ref-year
))
681 (setq months
(1- months
)))
682 (+ (* (/ months rng-xsd-months-per-gregorian-cycle
)
683 rng-xsd-days-per-gregorian-cycle
)
686 (defun rng-xsd-duration-months (d)
690 (defun rng-xsd-duration-seconds (d)
691 (+ (* (+ (* (+ (* (aref d
2)
699 (defun rng-xsd-convert-qname (string)
700 (and (string-match "\\`[ \r\n\t]*\\([_[:alpha:]][-._[:alnum:]]*\\(:[_[:alpha:]][-._[:alnum:]]*\\)?\\)[ \r\n\t]*\\'" string
)
701 (let ((colon (match-beginning 2))
702 (context (apply (car rng-dt-namespace-context-getter
)
703 (cdr rng-dt-namespace-context-getter
))))
705 (let* ((prefix (substring string
708 (binding (assoc prefix
(cdr context
))))
715 (match-string 1 string
))))))
717 (defun rng-xsd-convert-list (string convert
&rest args
)
718 (let* ((tokens (split-string string
"[ \t\n\r]+"))
721 (let ((obj (apply convert
722 (cons (car tem
) args
))))
725 (setq tem
(cdr tem
)))
729 ;; Fortuitously this returns nil if the list is empty
730 ;; which is what we want since the list types
731 ;; have to have one or more members.
734 (defun rng-xsd-strcmp (s1 s2
)
735 (cond ((string= s1 s2
) 0)
739 (put 'string
'rng-xsd-convert
'(identity))
740 (put 'string
'rng-xsd-length
'length
)
741 (put 'string
'rng-xsd-matches-anything t
)
743 (put 'normalizedString
'rng-xsd-convert
'(identity))
744 (put 'normalizedString
'rng-xsd-length
'length
)
745 (put 'normalizedString
'rng-xsd-matches-anything t
)
747 (put 'token
'rng-xsd-convert
'(identity))
748 (put 'token
'rng-xsd-length
'length
)
749 (put 'token
'rng-xsd-matches-anything t
)
751 (put 'hexBinary
'rng-xsd-convert
'(rng-xsd-convert-hex-binary))
752 (put 'hexBinary
'rng-xsd-length
'rng-xsd-hex-binary-length
)
754 (put 'base64Binary
'rng-xsd-convert
'(rng-xsd-convert-base64-binary))
755 (put 'base64Binary
'rng-xsd-length
'rng-xsd-base64-binary-length
)
757 (put 'boolean
'rng-xsd-convert
'(rng-xsd-convert-boolean))
759 (put 'float
'rng-xsd-convert
'(rng-xsd-convert-float))
760 (put 'float
'rng-xsd-less-than
'rng-xsd-float
<)
762 (put 'double
'rng-xsd-convert
'(rng-xsd-convert-float))
763 (put 'double
'rng-xsd-less-than
'rng-xsd-float
<)
765 (put 'decimal
'rng-xsd-convert
'(rng-xsd-convert-decimal))
766 (put 'decimal
'rng-xsd-less-than
'rng-xsd-decimal
<)
768 (put 'integer
'rng-xsd-convert
'(rng-xsd-convert-integer))
769 (put 'integer
'rng-xsd-less-than
'rng-xsd-decimal
<)
771 (defun rng-xsd-def-integer-type (name min max
)
772 (put name
'rng-xsd-less-than
'rng-xsd-decimal
<)
775 (cdr (rng-xsd-compile 'integer
776 (append (and min
`((minInclusive .
,min
)))
777 (and max
`((maxInclusive .
,max
))))))))
779 (defun rng-xsd-def-token-type (name regexp
)
780 (put name
'rng-xsd-convert
(list 'rng-xsd-convert-token
781 (concat "\\`[\r\n\t ]*\\("
783 "\\)[\r\n\t ]*\\'")))
784 (put name
'rng-xsd-length
'length
))
786 (rng-xsd-def-token-type 'NMTOKEN
"[-.:_[:alnum:]]+")
787 (rng-xsd-def-token-type 'Name
"[:_[:alpha:]][-.:_[:alnum:]]*")
788 (rng-xsd-def-token-type 'NCName
"[_[:alpha:]][-._[:alnum:]]*")
789 (rng-xsd-def-token-type 'language
790 "[a-zA-Z]\\{1,8\\}\\(?:-[a-zA-Z0-9]\\{1,8\\}\\)*")
792 (put 'ENTITY
'rng-xsd-convert
(get 'NCName
'rng-xsd-convert
))
793 (put 'ENTITY
'rng-xsd-length
'length
)
794 (put 'ID
'rng-xsd-convert
(get 'NCName
'rng-xsd-convert
))
795 (put 'ID
'rng-xsd-length
'length
)
796 (put 'IDREF
'rng-xsd-convert
(get 'NCName
'rng-xsd-convert
))
797 (put 'IDREF
'rng-xsd-length
'length
)
799 (defun rng-xsd-def-list-type (name member-name
)
800 (put name
'rng-xsd-convert
(cons 'rng-xsd-convert-list
801 (get member-name
'rng-xsd-convert
)))
802 (put name
'rng-xsd-length
'length
))
804 (rng-xsd-def-list-type 'NMTOKENS
'NMTOKEN
)
805 (rng-xsd-def-list-type 'IDREFS
'IDREF
)
806 (rng-xsd-def-list-type 'ENTITIES
'ENTITY
)
808 (put 'anyURI
'rng-xsd-convert
'(rng-xsd-convert-any-uri))
809 (put 'anyURI
'rng-xsd-length
'length
)
811 (put 'QName
'rng-xsd-convert
'(rng-xsd-convert-qname))
812 (put 'NOTATION
'rng-xsd-convert
'(rng-xsd-convert-qname))
814 (defconst rng-xsd-long-max
"9223372036854775807")
815 (defconst rng-xsd-long-min
"-9223372036854775808")
816 (defconst rng-xsd-int-max
"2147483647")
817 (defconst rng-xsd-int-min
"-2147483648")
818 (defconst rng-xsd-short-max
"32767")
819 (defconst rng-xsd-short-min
"-32768")
820 (defconst rng-xsd-byte-max
"127")
821 (defconst rng-xsd-byte-min
"-128")
822 (defconst rng-xsd-unsigned-long-max
"18446744073709551615")
823 (defconst rng-xsd-unsigned-int-max
"4294967295")
824 (defconst rng-xsd-unsigned-short-max
"65535")
825 (defconst rng-xsd-unsigned-byte-max
"255")
827 (rng-xsd-def-integer-type 'nonNegativeInteger
"0" nil
)
828 (rng-xsd-def-integer-type 'positiveInteger
"1" nil
)
829 (rng-xsd-def-integer-type 'nonPositiveInteger nil
"0")
830 (rng-xsd-def-integer-type 'negativeInteger nil
"-1")
831 (rng-xsd-def-integer-type 'long rng-xsd-long-min rng-xsd-long-max
)
832 (rng-xsd-def-integer-type 'int rng-xsd-int-min rng-xsd-int-max
)
833 (rng-xsd-def-integer-type 'short rng-xsd-short-min rng-xsd-short-max
)
834 (rng-xsd-def-integer-type 'byte rng-xsd-byte-min rng-xsd-byte-max
)
835 (rng-xsd-def-integer-type 'unsignedLong
"0" rng-xsd-unsigned-long-max
)
836 (rng-xsd-def-integer-type 'unsignedInt
"0" rng-xsd-unsigned-int-max
)
837 (rng-xsd-def-integer-type 'unsignedShort
"0" rng-xsd-unsigned-short-max
)
838 (rng-xsd-def-integer-type 'unsignedByte
"0" rng-xsd-unsigned-byte-max
)
840 (defun rng-xsd-def-date-time-type (name template
)
841 (put name
'rng-xsd-convert
(list 'rng-xsd-convert-date-time
842 (rng-xsd-make-date-time-regexp template
)))
843 (put name
'rng-xsd-less-than
'rng-xsd-date-time
<))
845 (rng-xsd-def-date-time-type 'dateTime
"Y-M-DTt")
846 (rng-xsd-def-date-time-type 'time
"t")
847 (rng-xsd-def-date-time-type 'date
"Y-M-D")
848 (rng-xsd-def-date-time-type 'gYearMonth
"Y-M")
849 (rng-xsd-def-date-time-type 'gYear
"Y")
850 (rng-xsd-def-date-time-type 'gMonthDay
"--M-D")
851 (rng-xsd-def-date-time-type 'gDay
"---D")
852 (rng-xsd-def-date-time-type 'gMonth
"--M")
854 (put 'duration
'rng-xsd-convert
'(rng-xsd-convert-duration))
855 (put 'duration
'rng-xsd-less-than
'rng-xsd-duration
<)
859 ;;; rng-xsd.el ends here