aboutsummaryrefslogtreecommitdiff
path: root/stdlib/source/lux/macro/syntax/common/reader.lux
blob: 689e166d0dfbef507ebde521777e62df855b114e (plain)
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
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
(.module: {#.doc "Commons syntax readers."}
  [lux #*
   [abstract
    monad]
   [control
    ["p" parser ("#\." monad)
     ["s" code (#+ Parser)]]]
   [data
    ["." name ("#\." equivalence)]
    ["." product]
    ["." maybe]
    [collection
     ["." list]]]
   ["." meta]]
  ["." //])

(def: #export declaration
  {#.doc (doc "A reader for declaration syntax."
              "Such as:"
              quux
              (foo bar baz))}
  (Parser //.Declaration)
  (p.either (p.and s.local-identifier
                   (p\wrap (list)))
            (s.form (p.and s.local-identifier
                           (p.some s.local-identifier)))))

(def: #export annotations
  {#.doc "Reader for the common annotations syntax used by def: statements."}
  (Parser //.Annotations)
  (s.record (p.some (p.and s.tag s.any))))

(def: (flat-list^ _)
  (-> Any (Parser (List Code)))
  (p.either (do p.monad
              [_ (s.tag! (name-of #.Nil))]
              (wrap (list)))
            (s.form (do p.monad
                      [_ (s.tag! (name-of #.Cons))
                       [head tail] (s.tuple (p.and s.any s.any))
                       tail (s.local (list tail) (flat-list^ []))]
                      (wrap (#.Cons head tail))))))

(template [<name> <type> <tag> <then>]
  [(def: <name>
     (Parser <type>)
     (<| s.tuple
         (p.after s.any)
         s.form
         (do p.monad
           [_ (s.tag! (name-of <tag>))]
           <then>)))]

  [tuple-meta^ (List Code) #.Tuple (flat-list^ [])]
  [text-meta^  Text        #.Text  s.text]
  )

(def: (find-definition-args meta-data)
  (-> (List [Name Code]) (List Text))
  (<| (maybe.default (list))
      (: (Maybe (List Text)))
      (case (list.find (|>> product.left (name\= ["lux" "func-args"])) meta-data)
        (^multi (#.Some [_ value])
                [(p.run tuple-meta^ (list value))
                 (#.Right [_ args])]
                [(p.run (p.some text-meta^) args)
                 (#.Right [_ args])])
        (#.Some args)

        _
        #.None)))

(def: #export typed-input
  {#.doc "Reader for the common typed-argument syntax used by many macros."}
  (Parser //.Typed-Input)
  (s.record (p.and s.any s.any)))

(def: #export type-variables
  {#.doc "Reader for the common type var/param used by many macros."}
  (Parser (List Text))
  (p.some s.local-identifier))