-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
29 changed files
with
1,939 additions
and
3,105 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
|
||
``` | ||
deb12user@GODLIKE:~/metta-wam/tests/zebra$ time mettalog fish_riddle_2_no_states.metta | ||
!(include dv-pl) | ||
Deterministic: () | ||
!(query &self (because red house keeps dogs the $norwegian is the fish owner)) | ||
Deterministic: (Succeed ((quote (because red house keeps dogs the norwegian is the fish owner)))) | ||
!(query &self (because blue house keeps dogs the $brit is the fish owner)) | ||
Deterministic: (Succeed ((quote (because blue house keeps dogs the brit is the fish owner)))) | ||
!(query &self (because red house keeps cats the $norwegian is the fish owner)) | ||
Deterministic: (Succeed ((quote (because red house keeps cats the norwegian is the fish owner)))) | ||
% 4,329,368 inferences, 0.375 CPU in 0.377 seconds (100% CPU, 11540291 Lips) | ||
[()] | ||
[(Succeed ((quote (because red house keeps dogs the norwegian is the fish owner))))] | ||
[(Succeed ((quote (because blue house keeps dogs the brit is the fish owner))))] | ||
[(Succeed ((quote (because red house keeps cats the norwegian is the fish owner))))] | ||
; DEBUG Exit code of METTA_CMD: 7 | ||
real 0m3.445s | ||
user 0m3.356s | ||
sys 0m0.106s | ||
``` | ||
|
||
``` | ||
(venv) deb12user@GODLIKE:~/metta-wam/tests/zebra$ time metta fish_riddle_2_no_states.metta | ||
[(Succeed ((quote (because red house keeps dogs the norwegian is the fish owner)) (quote (because red house keeps dogs the norwegian is the fish owner))))] | ||
[(Succeed ((quote (because blue house keeps dogs the brit is the fish owner)) (quote (because blue house keeps dogs the brit is the fish owner))))] | ||
[(Succeed ((quote (because red house keeps cats the norwegian is the fish owner)) (quote (because red house keeps cats the norwegian is the fish owner))))] | ||
real 2m52.320s | ||
user 2m52.054s | ||
sys 0m0.260s | ||
``` | ||
|
||
|
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,266 @@ | ||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ||
;; Backward chaining logic | ||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ||
;!(import! &self dv-pl) | ||
;; Predicate signatures for backward chaining logic | ||
(: cur= (-> Atom Atom)) ; Current goal | ||
(: retVal= (-> Atom Atom)) ; Return value | ||
(: goal= (-> Atom Atom)) ; Final goal | ||
|
||
;; Interface for backward chaining (quotes the arguement) | ||
(: backward-chain (-> Atom Atom Atom Atom Atom)) | ||
(= (backward-chain $info $goal $kb $rb) | ||
(backward-chain-q $info (quote $goal) $kb $rb)) | ||
|
||
;; Handle specific cases during backward chaining | ||
(: backward-chain-q (-> Atom Atom Atom Atom Atom)) | ||
(= (backward-chain-q $info (quote $goal) $kb $rb) | ||
(case (quote $goal) ( | ||
((quote (is $a $b)) (let $a $b (quote $goal))) ; Assignment | ||
((quote (bool $expr)) (if $expr (quote $goal) (empty))) ; Boolean evaluation | ||
((quote (eval= $a $expr)) (let $a $expr (quote $goal))) ; Expression evaluation | ||
((quote (nonvar $var)) (if (== Variable (get-metatype $var)) (empty) (quote $goal))) ; Non-variable check | ||
((quote (var $var)) (if (== Variable (get-metatype $var)) (quote $goal) (empty))) ; Variable check | ||
((quote (true)) (quote $goal)) ; Always succeeds | ||
((quote (fail)) (empty)) ; Always fails | ||
((quote (cut $signal)) (if (equalz $info $signal) (quote $goal) (quote $goal))) ; Cut operator for pruning | ||
((quote (naf $expr)) (if (has-match $kb $expr) (empty) (quote $goal))) ; Negation as failure | ||
((quote (qnaf $expr)) (let (Failed $_1) (backward-chain $info2 $expr $kb $rb) (quote $goal))) ; Negation as failure at query level | ||
($_2 (backward-chain-q2 $info (quote $goal) $kb $rb)) ; Match against knowledge base | ||
))) | ||
|
||
(= (backward-chain-q2 $info (quote $goal) $kb $rb) | ||
(match $kb $goal (quote $goal))) | ||
;; Recursive case for backward chaining | ||
(= (backward-chain-q2 $old_info (quote $goal) $kb $rb) | ||
(function | ||
(return ;; if $info is bound that means a clause has called a cut (thus we return empty) | ||
(if (== Variable (get-metatype $old_info)) | ||
(let $r (match $rb ($goal :- $body) | ||
(match-body $info $body $kb $rb $goal)) | ||
(if (== Variable (get-metatype $info)) ;; if not cut | ||
$r ;; then return normal | ||
(return $r))) ;; else return early | ||
(empty))))) | ||
|
||
|
||
;; Match predicate: checks if a goal has at least one match in a space | ||
;; This is less efficient, as it processes the entire space to find matches | ||
(: has-match (-> Atom Atom Bool)) | ||
(= (has-match $space $g) | ||
(let $m (collapse (match $space $g True)) ; Attempt to match $g in $space | ||
(if (== $m ()) ; If no match is found | ||
False ; Return False | ||
True ; Otherwise, return True | ||
) | ||
) | ||
) | ||
|
||
;; Predicate to check if a function definition exists in a given space | ||
(= (has-fundef $space $g) | ||
(let $m (collapse (match $space (= $g $_1) True)) ; Match $g as a function in $space | ||
(if (== $m ()) ; If no match is found | ||
False ; Return False | ||
True ; Otherwise, return True | ||
) | ||
) | ||
) | ||
|
||
;; Chain through each element in the body and return the goal | ||
(: match-body (-> Atom Atom Atom Atom Atom Atom)) | ||
(= (match-body $info $body $kb $rb $goal) | ||
(if (== $body ()) | ||
(quote $goal) ; Base case: no more elements to match | ||
(let* ( | ||
(($cur $rest) (decons-atom $body)) ; Deconstruct body | ||
($debugging False) ;; Print debug or not | ||
($bugcheck False) ;; Catch a bug where we dont return the quoted goals | ||
(() (if $debugging (println! (quote (IN (cur= $cur) (goal= $goal)))) ())) ; Debugging: log input | ||
($_12 (if $bugcheck | ||
(let* (($retVal (backward-chain $info $cur $kb $rb)) ; Recursive chaining | ||
($m (collapse (equalz (quote $cur) $retVal))) ; Check match | ||
(() (if (== $m ()) (println! (quote (BAD!!!!!!!! (cur= $cur) (retVal= $retVal))) ()))) | ||
) ()) ())) | ||
((quote $cur) (backward-chain $info $cur $kb $rb)) ; Recursive chaining | ||
(() (if $debugging (println! (quote (OUT (cur= $cur) (goal= $goal)))) ())) ; Debugging: log output | ||
) | ||
(match-body $info $rest $kb $rb $goal) ; Continue matching | ||
) | ||
) | ||
) | ||
|
||
|
||
;; Predicate for general atom equality | ||
(: equalz (-> Atom Atom Bool)) ; Compares two atoms | ||
(= (equalz $A $A) True) ; Atoms are equal if they are identical | ||
|
||
(: quote-equalz (-> Atom Atom Bool)) | ||
(= (quote-equalz $A $A) True) | ||
(= (quote-equalz $A $B) (equalz (quote $A) $B)) | ||
(= (quote-equalz $A $B) (equalz $A (quote $B))) | ||
|
||
(: is-equals (-> %Undefined %Undefined% Bool)) ; Compares two atoms | ||
(: is-equals (-> $t $t Bool)) ; Compares two things | ||
(= (is-equals $A $A) True) ; Atoms are equal if they are identical | ||
|
||
;; Define natural numbers using Zero (Z) and Successor (S) | ||
(: Z Nat) ; Base case for natural numbers (0) | ||
(: S (-> Nat Nat)) ; Successor function for incrementing natural numbers | ||
|
||
;; Predicate for natural number equality | ||
(: nat-equal (-> Nat Nat Bool)) ; Compares two natural numbers | ||
(= (nat-equal $A $A) True) ; Two numbers are equal if they are identical | ||
|
||
|
||
;; Query execution logic | ||
(: query (-> Atom Atom Atom)) | ||
(= (query $kb $goal) | ||
(let $m (collapse (backward-chain $info $goal $kb $kb)) | ||
(if (== $m ()) (Failed (quote $goal)) (Succeed $m))) | ||
) | ||
|
||
(: query-f (-> Atom Atom Atom)) | ||
(= (query-f $kb $goal) | ||
(let $m (collapse (backward-chain $info $goal $kb $kb)) | ||
(if (== $m ()) Passed-Negation (Failed $m))) | ||
) | ||
|
||
|
||
;!(equalz $a (+ 1 1)) | ||
;!(is-equals 2 (+ 1 1)) | ||
;!(is-equals $b (+ 1 1)) | ||
|
||
|
||
;; Predicate: Check if an element is a member of a list | ||
(member $Elem ($Cons $Elem $_1)) ; Base case: element found | ||
((member $Elem ($Cons $_1 $Tail)) :- | ||
((member $Elem $Tail))) ; Recursive case: continue checking | ||
|
||
|
||
;; Zero-based natural number indexing in lists | ||
(nat-nth Z ($Cons $Elem $_1) $Elem) ; Base case: first element | ||
((nat-nth (S $N) ($Cons $_1 $Rest) $Elem) :- | ||
((nat-nth $N $Rest $Elem))) ; Recursive case: move to next element | ||
|
||
;; Append two lists | ||
(cons-append Nil $List2 $List2) ; Base case: append empty list | ||
((cons-append ($Cons $H $T) $List2 ($Cons $H $T2)) :- | ||
((cons-append $T $List2 $T2))) ; Recursive case: append remaining elements | ||
|
||
;; Predicate to check equality of two values | ||
(same $x $x) | ||
|
||
;; Predicate to get the last element of a list | ||
(last $x ($Cons $x Nil)) ; Base case: last element | ||
((last $x ($Cons $_1 $rest)) :- | ||
((last $x $rest))) ; Recursive case: move to next element | ||
|
||
;; Calculate length of a list | ||
(length Nil 0) ; Base case: empty list has length 0 | ||
((length ($Cons $_1 $rest) $out) :- | ||
((length $rest $out2) | ||
(is $out (+ $out2 1)))) ; Recursive case: increment length | ||
|
||
;; Calculate sum of natural numbers from 0 to $n | ||
(sum 0 0) ; Base case: sum to 0 is 0 | ||
((sum $n $s) :- ( | ||
(bool (> $n 0)) ; Ensure $n is positive | ||
(is $n1 (- $n 1)) ; Decrement $n | ||
(sum $n1 $s1) ; Calculate sum for smaller range | ||
(is $s (+ $n $s1)))) ; Add $n to sum of smaller range | ||
|
||
;; Predicate to append lists | ||
(append Nil $L $L) ; Base case: appending an empty list | ||
((append ($Cons $H $T) $L ($Cons $H $R)) :- ( | ||
(append $T $L $R) ; Recursive case: append remaining elements | ||
)) | ||
|
||
|
||
|
||
;; Negation as failure test | ||
((bachelor $x) :- ( | ||
(man $x) ; $x must be a man | ||
(naf (married $x)) ; $x must not be married | ||
)) | ||
|
||
; Facts: Men and marital status | ||
(man John) (man Tim) | ||
(married Tim) | ||
|
||
;; Predicate to define a woman | ||
((woman $x) :- ( | ||
(naf (man $x)) ; $x is a woman if not a man | ||
)) | ||
|
||
;;;;;;;;;;;;;;;;; | ||
;; TEST Queries ;; | ||
;;;;;;;;;;;;;;;;; | ||
|
||
(: traced-eval (-> Atom Atom)) | ||
(= (traced-eval $arg) (traced-eval-q (quote $arg))) | ||
(: traced-eval-q (-> Expression Atom)) | ||
(= (traced-eval-q (quote $arg)) | ||
(let* ( | ||
(() (println! (quote (====>EVAL $arg)))) | ||
($r $arg) | ||
(() (println! (quote (RET<=== $r))))) | ||
$r)) | ||
|
||
(: progn (-> Atom Atom)) | ||
(= (progn $body) (progn-q (quote $body))) | ||
(: progn-q (-> Expression Atom)) | ||
(= (progn-q (quote $body)) | ||
(if (== (quote $body) (quote ())) () | ||
(let* ( | ||
(($head $rest) (decons-atom $body)) ; Deconstruct body | ||
($r (traced-eval-q (quote $head)))) | ||
(if (== (quote $rest) (quote ())) | ||
$r (progn-q (quote $rest)))))) | ||
|
||
(= (test_normal_queries $kb) | ||
|
||
(progn-q (quote ( | ||
|
||
; Query for the last element in a list | ||
(query $kb (last $x ($Cons 1 ($Cons 2 ($Cons 3 Nil))))) | ||
|
||
; Query for the length of a list | ||
(query $kb (length ($Cons 1 ($Cons 2 ($Cons 3 Nil))) $out)) | ||
|
||
; Query for the sum of numbers up to 3 | ||
(query $kb (sum 3 $x)) | ||
|
||
; Query for appending lists | ||
(query $kb (append ($Cons 1 ($Cons 2 Nil)) ($Cons 3 ($Cons 4 Nil)) $x)) | ||
(query $kb (append ($Cons 1 ($Cons 2 Nil)) $Out ($Cons 1 ($Cons 2 ($Cons 3 ($Cons 4 Nil)))))) | ||
(query $kb (append $What $Out ($Cons 1 ($Cons 2 ($Cons 3 ($Cons 4 Nil)))))) | ||
|
||
;; Test list operations | ||
(query $kb (append ($Cons 1 Nil) ($Cons 2 ($Cons 3 Nil)) $Result)) ; Appending lists | ||
(query $kb (append ($Cons 1 ($Cons 2 Nil)) $Rest ($Cons 1 ($Cons 2 ($Cons 3 ($Cons 4 Nil)))))) ; Partial list append | ||
|
||
;; Negation as failure tests with bachelors and marital status | ||
(query $kb (bachelor $x)) ; Finds bachelors who are not married | ||
(query-f $kb (woman Tim)) ; Tests failure for a man labeled as a woman | ||
(query $kb (woman Jane)) ; Should succeed for Jane if defined as a woman | ||
(query-f $kb (woman $x)) ; Negative query for undefined women | ||
|
||
;; Check if an element is a member of a list | ||
(query $kb (member 3 ($Cons 1 ($Cons 2 ($Cons 3 Nil))))) ; Should succeed | ||
(query-f $kb (member 5 ($Cons 1 ($Cons 2 ($Cons 3 Nil))))) ; Should fail, 5 is not in the list | ||
(query $kb (member $x ($Cons 1 ($Cons 2 ($Cons 3 Nil))))) ; Should find members one by one | ||
|
||
;; Query for basic arithmetic | ||
(query $kb (is $x (+ 1 1))) ; $x should be assigned 2 | ||
|
||
;; Query for natural number indexing | ||
(query $kb (nat-nth (S (S Z)) (SomethingAtThird $1 $2) Third)) ; Should find the third element | ||
|
||
)))) | ||
|
||
;!(import! &self dv-pl) | ||
; !(test_normal_queries) | ||
; !(test_normal_queries) | ||
|
||
|
||
|
Oops, something went wrong.