forked from gregr/first-order-miniKanren
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmicrok-fo.rkt
79 lines (74 loc) · 2.45 KB
/
microk-fo.rkt
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
#lang racket
(provide
(all-from-out "common.rkt")
(struct-out disj)
(struct-out conj)
(struct-out relate)
(struct-out ==)
(struct-out =/=)
(struct-out symbolo)
(struct-out stringo)
(struct-out numbero)
(struct-out not-symbolo)
(struct-out not-stringo)
(struct-out not-numbero)
(struct-out mplus)
(struct-out bind)
(struct-out pause)
step
mature
mature?)
(require "common.rkt")
;; first-order microKanren
(struct disj (g1 g2) #:prefab)
(struct conj (g1 g2) #:prefab)
(struct relate (thunk description) #:prefab)
(struct == (t1 t2) #:prefab)
(struct =/= (t1 t2) #:prefab)
(struct symbolo (t) #:prefab)
(struct stringo (t) #:prefab)
(struct numbero (t) #:prefab)
(struct not-symbolo (t) #:prefab)
(struct not-stringo (t) #:prefab)
(struct not-numbero (t) #:prefab)
(struct bind (bind-s bind-g) #:prefab)
(struct mplus (mplus-s1 mplus-s2) #:prefab)
(struct pause (pause-state pause-goal) #:prefab)
(define (mature? s) (or (not s) (pair? s)))
(define (mature s)
(if (mature? s) s (mature (step s))))
(define (start st g)
(match g
((disj g1 g2)
(step (mplus (pause st g1)
(pause st g2))))
((conj g1 g2)
(step (bind (pause st g1) g2)))
((relate thunk _)
(pause st (thunk)))
((== t1 t2) (state->stream (unify t1 t2 st)))
((=/= t1 t2) (state->stream (disunify t1 t2 st)))
((symbolo t) (state->stream (typify t symbol? st)))
((stringo t) (state->stream (typify t string? st)))
((numbero t) (state->stream (typify t number? st)))
((not-symbolo t) (state->stream (distypify t symbol? st)))
((not-stringo t) (state->stream (distypify t string? st)))
((not-numbero t) (state->stream (distypify t number? st)))))
(define (step s)
(match s
((mplus s1 s2)
(let ((s1 (if (mature? s1) s1 (step s1))))
(cond ((not s1) s2)
((pair? s1)
(cons (car s1)
(mplus s2 (cdr s1))))
(else (mplus s2 s1)))))
((bind s g)
(let ((s (if (mature? s) s (step s))))
(cond ((not s) #f)
((pair? s)
(step (mplus (pause (car s) g)
(bind (cdr s) g))))
(else (bind s g)))))
((pause st g) (start st g))
(_ s)))