/slib/scanf.scm

https://github.com/ieee8023/UMB-Scheme · Lisp · 140 lines · 103 code · 11 blank · 26 comment · 0 complexity · c69967a776768fddceda0f7e9f7ca593 MD5 · raw file

  1. ;;;;"scanf.scm" implemenation of formated input
  2. ;From: jjb@isye.gatech.edu (John Bartholdi)
  3. ;
  4. ; This code is in the public domain.
  5. ;;;;Heavily hacked by jaffer 4/94
  6. ;;;This implements a Scheme-oriented version of SCANF: returns a list of
  7. ;;;objects read (rather than set!-ing values).
  8. ; Example:
  9. ;(sscanf "I want $100 for 3.14159 kilograms of ice cream."
  10. ; "%7s %*c %d %*s %*f %24s")
  11. ; => ("I want " 100 "kilograms of ice cream.")
  12. (require 'string-port)
  13. (define (scanf:scanf <format-string> . args)
  14. (scanf:fscanf (current-input-port) <format-string>))
  15. (define (scanf:sscanf str <format-string> . args)
  16. (call-with-input-string
  17. str (lambda (p) (scanf:fscanf p <format-string>))))
  18. (define (scanf:fscanf <input-port> <format-string> . args)
  19. (call-with-input-string
  20. <format-string>
  21. (lambda (ip1)
  22. (let loop1 ((items-read '()))
  23. ;; get the next format by reading between "%" characters:
  24. (let ((next-format
  25. (scanf:read-separate-item (lambda (c) (char=? #\% c)) ip1)))
  26. (cond
  27. ((eof-object? next-format)
  28. (reverse items-read))
  29. (else ; interpret next format:
  30. (call-with-input-string
  31. next-format
  32. (lambda (ip2)
  33. (let loop2 ((field-width-chars '())
  34. (report-field? #T))
  35. (let ((c (char-downcase (read-char ip2))))
  36. (cond
  37. ((char=? #\* c)
  38. (loop2 field-width-chars #F))
  39. ((char-numeric? c)
  40. (loop2 (cons c field-width-chars) report-field?))
  41. ((char=? #\c c)
  42. (let ((next-item (read-char <input-port>)))
  43. (if report-field?
  44. (loop1 (cons next-item items-read))
  45. (loop1 items-read))))
  46. (else
  47. (let ((next-item
  48. (if (null? field-width-chars)
  49. ; no fieldwidth given so
  50. (scanf:read-separate-item char-whitespace?
  51. <input-port>)
  52. ; else read block of chars
  53. (let ((field-width
  54. (string->number
  55. (list->string
  56. (reverse field-width-chars)))))
  57. (scanf:read-n-chars field-width
  58. <input-port>)))))
  59. (case c
  60. ((#\e #\f #\g #\i #\o #\d #\u #\x)
  61. (call-with-input-string
  62. next-item
  63. (lambda (ip-str)
  64. (let* ((num (scanf:read-separate-item
  65. char-whitespace? ip-str)))
  66. (set! num
  67. (case c
  68. ((#\e #\f #\g #\i) (string->number num))
  69. ((#\o) (string->number num 8))
  70. ((#\d #\u) (string->number num 10))
  71. ((#\x) (string->number num 16))))
  72. (if (number? num)
  73. (if report-field?
  74. (loop1 (cons num items-read))
  75. (loop1 items-read))
  76. (slib:error
  77. 'SCANF
  78. "Number format (~s) does not match input (~s)"
  79. next-format
  80. next-item))))))
  81. ((#\s)
  82. (if report-field?
  83. (loop1 (cons next-item items-read))
  84. (loop1 items-read)))
  85. (else
  86. (slib:error
  87. 'SCANF
  88. "Unsupported format directive: ~a" c)))))))))))))))))
  89. ; Reads characters from <input-port> and returns them in a
  90. ; string (excluding any character of which
  91. ; (<separator?> c)
  92. ; returns #T). Reads until reach either end of file or else
  93. ; the first separator following a non-separator. If at the
  94. ; end of the file, returns eof-object.
  95. ;
  96. ; <separator?> is a function of one argument, a character, and
  97. ; returns either #T or #F.
  98. (define (scanf:read-separate-item <separator?> <input-port>)
  99. (let ((c (peek-char <input-port>)))
  100. (if (eof-object? c) c
  101. (let loop ((char-list '())
  102. (found-valid-chars? #F))
  103. (let ((c (read-char <input-port>)))
  104. (cond ((eof-object? c)
  105. (list->string (reverse char-list)))
  106. ((<separator?> c)
  107. (if found-valid-chars?
  108. (list->string (reverse char-list))
  109. (loop char-list #F)))
  110. (else ; not a separator:
  111. (loop (cons c char-list) #T))))))))
  112. ; Reads characters from a port until either <n> are read or
  113. ; eof-object? evaluates to #T, then returns all the characters read
  114. ; in a string.
  115. (define (scanf:read-n-chars <n> <input-port>)
  116. (let ((c (peek-char <input-port>)))
  117. (if (eof-object? c) c
  118. (let ((str (make-string <n>)))
  119. (let loop ((count 0))
  120. (cond
  121. ((= <n> count) str)
  122. ((eof-object? (peek-char <input-port>))
  123. (substring str 0 count))
  124. (else
  125. (string-set! str count (read-char <input-port>))
  126. (loop (+ 1 count)))))))))
  127. (define scanf scanf:scanf)
  128. (define sscanf scanf:sscanf)
  129. (define fscanf scanf:fscanf)