#include "py4yap.h" static foreign_t repr_term(PyObject *pVal, term_t t) { term_t to = PL_new_term_ref(), t1 = PL_new_term_ref(); PL_put_pointer(t1, pVal); PL_cons_functor(to, FUNCTOR_pointer1, t1); Py_INCREF(pVal); return PL_unify(t, to); } foreign_t assign_to_symbol(term_t t, PyObject *e); foreign_t assign_to_symbol(term_t t, PyObject *e) { char *s = NULL; if (!PL_get_atom_chars(t, &s)) { return false; } PyObject *dic; if (!lookupPySymbol(s, NULL, &dic)) dic = py_Main; return PyObject_SetAttrString(dic, s, e) == 0; } foreign_t python_to_term(PyObject *pVal, term_t t) { bool rc = true; term_t to = PL_new_term_ref(); // fputs(" <<*** ",stderr); PyObject_Print(pVal,stderr,0); // fputs("<<***\n",stderr); if (pVal == Py_None) { // fputs("<<*** ",stderr);Yap_DebugPlWrite(YAP_GetFromSlot(t)); fputs(" // >>***\n",stderr); rc = PL_unify_atom(t, ATOM_none); // fputs("<<*** ",stderr);Yap_DebugPlWrite(YAP_GetFromSlot(t)); fputs(" // >>***\n",stderr); } else if (PyBool_Check(pVal)) { rc = rc && PL_unify_bool(t, PyObject_IsTrue(pVal)); } else if (PyLong_Check(pVal)) { rc = rc && PL_unify_int64(t, PyLong_AsLong(pVal)); #if PY_MAJOR_VERSION < 3 } else if (PyInt_Check(pVal)) { rc = rc && PL_unify_int64(t, PyInt_AsLong(pVal)); #endif } else if (PyFloat_Check(pVal)) { rc = rc && PL_unify_float(t, PyFloat_AsDouble(pVal)); } else if (PyComplex_Check(pVal)) { term_t t1 = PL_new_term_ref(), t2 = PL_new_term_ref(); if (!PL_put_float(t1, PyComplex_RealAsDouble(pVal)) || !PL_put_float(t2, PyComplex_ImagAsDouble(pVal)) || !PL_cons_functor(to, FUNCTOR_complex2, t1, t2)) { rc = false; } else { rc = rc && PL_unify(t, to); } } else if (PyUnicode_Check(pVal)) { atom_t tmp_atom; #if PY_MAJOR_VERSION < 3 size_t sz = PyUnicode_GetSize(pVal) + 1; wchar_t *ptr = malloc(sizeof(wchar_t) * sz); sz = PyUnicode_AsWideChar((PyUnicodeObject *)pVal, ptr, sz - 1); tmp_atom = PL_new_atom_wchars(sz, ptr); free(ptr); #else const char *s = PyUnicode_AsUTF8(pVal); tmp_atom = PL_new_atom(s); #endif rc = rc && PL_unify_atom(t, tmp_atom); } else if (PyByteArray_Check(pVal)) { atom_t tmp_atom = PL_new_atom(PyByteArray_AsString(pVal)); rc = rc && PL_unify_atom(t, tmp_atom); #if PY_MAJOR_VERSION < 3 } else if (PyString_Check(pVal)) { atom_t tmp_atom = PL_new_atom(PyString_AsString(pVal)); rc = rc && PL_unify_atom(t, tmp_atom); #endif } else if (PyTuple_Check(pVal)) { Py_ssize_t i, sz = PyTuple_Size(pVal); functor_t f; const char *s; if (sz == 0) { rc = rc && PL_unify_atom(t, ATOM_brackets); } else { if ((s = (Py_TYPE(pVal)->tp_name))) { if (!strcmp(s, "H")) { pVal = PyTuple_GetItem(pVal, 0); if (pVal == NULL) { pVal = Py_None; PyErr_Clear(); } } if (s[0] == '$') { char *ns = malloc(strlen(s) + 5); strcpy(ns, "__"); strcat(ns, s + 1); strcat(ns, "__"); f = PL_new_functor(PL_new_atom(ns), sz); } else { f = PL_new_functor(PL_new_atom(s), sz); } } else { f = PL_new_functor(ATOM_t, sz); } if (PL_unify_functor(t, f)) { for (i = 0; i < sz; i++) { if (!PL_get_arg(i + 1, t, to)) rc = false; PyObject *p = PyTuple_GetItem(pVal, i); if (p == NULL) { PyErr_Clear(); p = Py_None; } rc = rc && python_to_term(p, to); } } else { rc = false; } // fputs(" ||*** ",stderr); Yap_DebugPlWrite(YAP_GetFromSlot(t)); fputs(" // ||***\n",stderr); } } else if (PyList_Check(pVal)) { Py_ssize_t i, sz = PyList_GET_SIZE(pVal); if (sz == 0) { rc = rc && PL_unify_nil(t); } else { for (i = 0; i < sz; i++) { PyObject *obj; if (!PL_unify_list(t, to, t)) { rc = false; break; } if ((obj = PyList_GetItem(pVal, i)) == NULL) { obj = Py_None; } rc = rc && python_to_term(obj, to); } rc = rc && PL_unify_nil(t); } // fputs("[***] ", stderr); // Yap_DebugPlWrite(yt); fputs("[***]\n", stderr); } else if (PyDict_Check(pVal)) { Py_ssize_t pos = 0; term_t to = PL_new_term_ref(), ti = to; int left = PyDict_Size(pVal); PyObject *key, *value; if (left == 0) { rc = rc && PL_unify_atom(t, ATOM_curly_brackets); } else { while (PyDict_Next(pVal, &pos, &key, &value)) { term_t tkey = PL_new_term_ref(), tval = PL_new_term_ref(), tint, tnew = PL_new_term_ref(); /* do something interesting with the values... */ if (!python_to_term(key, tkey)) { continue; } if (!python_to_term(value, tval)) { continue; } /* reuse */ tint = tkey; if (!PL_cons_functor(tint, FUNCTOR_colon2, tkey, tval)) { rc = false; continue; } if (--left) { if (!PL_cons_functor(tint, FUNCTOR_comma2, tint, tnew)) PL_reset_term_refs(tkey); rc = false; } if (!PL_unify(ti, tint)) { rc = false; } ti = tnew; PL_reset_term_refs(tkey); } rc = rc && PL_unify(t, to); } } else { rc = rc && repr_term(pVal, t); } PL_reset_term_refs(to); return rc; } X_API YAP_Term pythonToYAP(PyObject *pVal) { term_t t = PL_new_term_ref(); if (pVal == NULL || !python_to_term(pVal, t)) { PL_reset_term_refs(t); return 0; } YAP_Term tt = YAP_GetFromSlot(t); PL_reset_term_refs(t); Py_DECREF(pVal); return tt; } PyObject *py_Local, *py_Global; /** * assigns the Python RHS to a Prolog term LHS, ie LHS = RHS * * @param root Python environment * @param t left hand side, in Prolog, may be * - a Prolog variable, exports the term to Prolog, A <- RHS * - Python variable A, A <- RHS * - Python variable $A, A <- RHS * - Python string "A", A <- RHS * - Python array range * @param e the right-hand side * * @return -1 on failure. * * Note that this is an auxiliary routine to the Prolog *python_assign. */ bool python_assign(term_t t, PyObject *exp, PyObject *context) { context = find_obj(context, t, false); // Yap_DebugPlWriteln(yt); switch (PL_term_type(t)) { case PL_VARIABLE: { if (context == NULL) // prevent a.V= N*N[N-1] return python_to_term(exp, t); } case PL_ATOM: { char *s = NULL; PL_get_atom_chars(t, &s); if (!context) context = py_Main; return PyObject_SetAttrString(context, s, exp) == 0; } case PL_STRING: case PL_INTEGER: case PL_FLOAT: // domain or type erro? return false; default: { term_t tail = PL_new_term_ref(), arg = PL_new_term_ref(); size_t len, i; if (PL_skip_list(t, tail, &len) && PL_get_nil(tail)) { // true list bool o = true; if (PySequence_Check(exp) && PySequence_Length(exp) == len) for (i = 0; i < len; i++) { PyObject *p; if (!PL_get_list(t, arg, t)) { PL_reset_term_refs(tail); o = false; p = Py_None; } if ((p = PySequence_GetItem(exp, i)) == NULL) p = Py_None; if (!python_assign(arg, p, context)) { PL_reset_term_refs(tail); o = false; } } PL_reset_term_refs(tail); return o; } else { functor_t fun; if (!PL_get_functor(t, &fun)) { PL_reset_term_refs(tail); return false; } if (fun == FUNCTOR_sqbrackets2) { if (!PL_get_arg(2, t, tail)) { PL_reset_term_refs(tail); return false; } PyObject *o = term_to_python(tail, true, context); if (!PL_get_arg(2, t, tail) && !PL_get_nil(tail)) { PL_reset_term_refs(tail); return false; } if (!PL_get_arg(1, t, t)) { PL_reset_term_refs(tail); return false; } PL_reset_term_refs(tail); PyObject *i = term_to_python(t, true, NULL); if (!i) { return false; } if (PyList_Check(i)) { i = PyList_GetItem(i, 0); if (i == NULL) i = Py_None; long int j; if (PyList_Check(o)) { #if PY_MAJOR_VERSION < 3 if (PyInt_Check(i)) j = PyInt_AsLong(i); else #endif if (PyLong_Check(i)) j = PyLong_AsLong(i); else return NULL; return PyList_SetItem(o, j, exp) == 0; } if (PyDict_Check(i)) { return PyDict_SetItem(o, i, exp) == 0; } return PyObject_SetAttr(o, i, exp) == 0; } } } } } return NULL; }