๐Ÿ“ฆ Bogdanp / racket-chief

๐Ÿ“„ env.rkt ยท 56 lines
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56#lang racket/base

(require racket/match)

(provide
 read-env)

(define vardef-re
  #rx"(?m:([^=]+)=(.*))\n?")

(define (read-env-var in)
  (define-values (line col pos)
    (port-next-location in))

  (match (regexp-match vardef-re in)
    [#f
     (if (and line col)
         (error 'read-env-var "invalid variable definition at ~a:~a" line col)
         (error 'read-env-var "invalid variable definition at pos ~a" pos))]

    [(list _ name value)
     (values name value)]))

(define (read-env [in (current-input-port)])
  (let loop ([vars (hash)])
    (match (peek-char in)
      [(? eof-object?) vars]

      [#\#
       (read-line in)
       (loop vars)]

      [_
       (define-values (name value)
         (read-env-var in))

       (loop (hash-set vars name value))])))

(module+ test
  (require rackunit)

  (check-equal?
   (read-env (open-input-string "A=42"))
   (hash #"A" #"42"))

  (check-equal?
   (read-env (open-input-string #<<ENV
A=42
B=20
# a comment
A=1
ENV
                                ))
   (hash #"A" #"1"
         #"B" #"20")))