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