Skip to content

Commit

Permalink
Merge pull request #743 from trueagi-io/jetta_add
Browse files Browse the repository at this point in the history
updates to jetta gateway and tests
  • Loading branch information
Necr0x0Der authored Jul 16, 2024
2 parents bb1fcc0 + ac2f394 commit 69a698f
Show file tree
Hide file tree
Showing 3 changed files with 87 additions and 20 deletions.
74 changes: 54 additions & 20 deletions python/sandbox/jetta/compile.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,32 +12,64 @@

default_url_base = 'http://0.0.0.0:9090/contexts'

def jetta(metta, j_space, code, url=default_url_base):
# TODO: check if `code` is a string (otherwise repr it)?
# TODO: accept metta expressions?
r = requests.post(url + "/" + j_space, data=code)
class JettaServerError(RuntimeError):
pass

def jetta(j_space_id: str, code: str, url=None):
"""
The very basic caller to Jetta server with purely Python interface.
"""
if url is None:
url = default_url_base
r = requests.post(url + "/" + j_space_id, data=code)
if r.status_code != 200:
return None
raise JettaServerError(f"Status code: {r.status_code}")
r = json.loads(r.content.decode())
if not r['isSuccess']:
return r['messages']
# metta.parse_single(r['result'])
raise JettaServerError(r['messages'])
if r['type'] == 'java.lang.Integer':
r['result'] = int(r['result'])
return r['result']

def call_in_jetta(metta, j_space, func, *args):
return [ValueAtom(jetta(metta, j_space,
f"({func} " + " ".join([repr(a) for a in args]) + ")"))]
def _err_msg(expr, msg):
if not isinstance(expr, Atom):
expr = ValueAtom(str(expr))
if not isinstance(msg, Atom):
msg = ValueAtom(str(msg))
return [E(S('Error'), expr, E(S('JettaCompileError'), msg))]

def jetta_unwrap_atom(j_space_a: Atom, code_a: Atom,
url_a=ValueAtom(None)):
"""
The caller to Jetta server with atom wrapping and unwrapping.
This is needed to pass MeTTa expressions to `jetta` as well
as to convert JettaServerError into ordinary MeTTa Error
atom without Python error log.
"""
j_space = j_space_a.get_object().value
if isinstance(code_a, GroundedAtom):
code = code_a.get_object().value
else:
code = repr(code_a)
url = url_a.get_object().value
try:
result = jetta(j_space, code, url)
return [Atoms.UNIT if result is None else ValueAtom(result)]
except JettaServerError as e:
return _err_msg(code, e)
#return [E(S('Error'), ValueAtom(code),
# E(S('JettaCompileError'), ValueAtom(str(e))))]

def compile(metta: MeTTa, j_space, func, arity=None):
j_space = j_space.get_object().content
def compile(metta: MeTTa, j_space_a, func_a, arity=None):
j_space = j_space_a.get_object().content
if arity is not None:
arity = arity.get_object().content
if isinstance(func, GroundedAtom):
func = str(func.get_object().content)
if isinstance(func_a, GroundedAtom):
func = str(func_a.get_object().content)
elif not isinstance(func_a, SymbolAtom):
return _err_msg(func_a, "compile expects a function name")
else:
func = repr(func)
func = repr(func_a)
typ = metta.space().query(
E(S(':'), S(func), V('t'))
)
Expand All @@ -47,7 +79,10 @@ def compile(metta: MeTTa, j_space, func, arity=None):
if len(typ) == 0:
typ = ""
if arity is None:
raise RuntimeError("If type is not defined, arity should be provided")
return _err_msg(E(S('compile'), j_space_a, func_a),
"If type is not defined, arity should be provided" )
#return [E(S('Error'), E(S('compile'), j_space_a, func_a),
# E(S('JettaCompileError'), ValueAtom("If type is not defined, arity should be provided")))]
else:
typ = typ[0]['t']
arity = len(typ.get_children()) - 2
Expand All @@ -63,10 +98,10 @@ def compile(metta: MeTTa, j_space, func, arity=None):
repr(res[0]['__r']) + ")"
code = typ + "\n" + code
# TODO: check if compilation is successful
jetta(metta, j_space, code)
jetta(j_space, code)
#TODO: doesn't work for passing expressions (e.g. lambdas)
funcAtom = OperationAtom(func,
lambda *args: call_in_jetta(metta, j_space, func, *args),
lambda *args: jetta_unwrap_atom(j_space_a, E(func_a, *args)),
unwrap=False)
metta.register_atom(func+'-gnd', funcAtom)
return [Atoms.UNIT]
Expand All @@ -79,8 +114,7 @@ def jetta_space(url=default_url_base):
@register_atoms(pass_metta=True)
def jettaspace_atoms(metta: MeTTa):
newJSpaceAtom = OperationAtom('new-jetta-space', jetta_space)
jettaAtom = OperationAtom('jetta',
lambda *args: jetta(metta, *args))
jettaAtom = OperationAtom('jetta', jetta_unwrap_atom, unwrap=False)
compileAtom = OperationAtom('compile',
lambda *args: compile(metta, *args), unwrap=False)
return {
Expand Down
24 changes: 24 additions & 0 deletions python/sandbox/jetta/test_basic_jetta.metta
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,30 @@
(jetta &jspace "(+ 11 12)")
23)

; checking that we can pass an atom to `jetta`;
; `(foo 11 4)` is not reduced in this case, because
; `foo` was not defined and passed to `jetta` in a text code;
; FIXME? Should `jetta` accept the code as `Atom` type to
; avoid reducing such expressions in MeTTa?
!(assertEqual
(jetta &jspace (foo 11 4))
16)

; Testing errors
!(assertEqualToResult
(case (jetta "WRON SPACE" "(foo 11 4)")
(((Error $1 (JettaCompileError $2)) $2)))
("Status code: 500"))

; error: cannot resolve symbol
!(assertEqual
(< 0
((py-dot
(case (jetta &jspace "(foof 11 4)")
(((Error $1 (JettaCompileError $2)) $2)))
"find") "CannotResolveSymbol"))
True)

; TODO: fix the behavior (one or multiple results)
;(jetta &jspace
; "(foo 11 3)
Expand Down
9 changes: 9 additions & 0 deletions python/sandbox/jetta/test_expr_compile.metta
Original file line number Diff line number Diff line change
Expand Up @@ -24,3 +24,12 @@
!(assertEqual
(my-goo-gnd 4 5)
0)

; error: compile expects a function name
!(assertEqual
(< 0
((py-dot
(case (compile &jspace (ffff a b))
(((Error $1 (JettaCompileError $2)) $2)))
"find") "function name"))
True)

0 comments on commit 69a698f

Please sign in to comment.