384 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			384 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
 | |
| 
 | |
| #include "Yap.h"
 | |
| 
 | |
| #include "py4yap.h"
 | |
| 
 | |
| #include <frameobject.h>
 | |
| 
 | |
| void YAPPy_ThrowError__(const char *file, const char *function, int lineno,
 | |
|                         yap_error_number type, term_t where, ...) {
 | |
|   va_list ap;
 | |
|   char tmpbuf[MAXPATHLEN];
 | |
|   YAP_Term wheret = YAP_GetFromSlot(where);
 | |
|   PyObject *ptype;
 | |
| 
 | |
|   if ((ptype = PyErr_Occurred()) == NULL) {
 | |
|     PyErr_Print();
 | |
|   }
 | |
| 
 | |
|   va_start(ap, where);
 | |
|   char *format = va_arg(ap, char *);
 | |
|   if (format != NULL) {
 | |
| #if HAVE_VSNPRINTF
 | |
|     (void)vsnprintf(tmpbuf, MAXPATHLEN - 1, format, ap);
 | |
| #else
 | |
|     (void)vsprintf(tnpbuf, format, ap);
 | |
| #endif
 | |
|     // fprintf(stderr, "warning: ");
 | |
|     Yap_ThrowError__(file, function, lineno, type, wheret, tmpbuf);
 | |
|   } else {
 | |
|     //    Yap_ThrowError__(file, function, lineno, type, wheret);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static Term repr_term(PyObject *pVal) {
 | |
|   Term t = MkAddressTerm(pVal);
 | |
|   return Yap_MkApplTerm(FunctorObj, 1, &t);
 | |
| }
 | |
| 
 | |
| 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;
 | |
|   Py_INCREF(e);
 | |
|   return PyObject_SetAttrString(dic, s, e) == 0;
 | |
| }
 | |
| 
 | |
| static Term python_to_term__(PyObject *pVal) {
 | |
|   if (pVal == Py_None) {
 | |
|     // fputs("<<*** ",stderr);Yap_DebugPlWrite(YAP_GetFromSlot(t));   fputs("
 | |
|     // >>***\n",stderr);
 | |
|     //return YAP_MkVarTerm();
 | |
|     // fputs("<<*** ",stderr);Yap_DebugPlWrite(YAP_GetFromSlot(t));   fputs("
 | |
|     return MkAtomTerm(Yap_LookupAtom("none"));
 | |
|     // >>***\n",stderr);
 | |
|   } else if (PyBool_Check(pVal)) {
 | |
|     if(PyObject_IsTrue(pVal)) return TermTrue;
 | |
|     return TermFalse;
 | |
|   } else if (PyLong_Check(pVal)) {
 | |
|     return MkIntegerTerm(PyLong_AsLong(pVal));
 | |
| #if PY_MAJOR_VERSION < 3
 | |
|   } else if (PyInt_Check(pVal)) {
 | |
|     return MkIntegerTerm(PyInt_AsLong(pVal));
 | |
| #endif
 | |
|   } else if (PyFloat_Check(pVal)) {
 | |
|     return MkFloatTerm(PyFloat_AsDouble(pVal));
 | |
|   } else if (PyComplex_Check(pVal)) {
 | |
|     Term t[2];
 | |
|     t[0] = MkFloatTerm(PyComplex_RealAsDouble(pVal));
 | |
|     t[1] = MkFloatTerm(PyComplex_ImagAsDouble(pVal));
 | |
|     return Yap_MkApplTerm(FunctorI, 2, t);
 | |
|  
 | |
|   }
 | |
|   else if (PyUnicode_Check(pVal)) {
 | |
| #if PY_MAJOR_VERSION < 3
 | |
|     size_t sz = PyUnicode_GetSize(pVal) + 1;
 | |
|     wchar_t *s = malloc(sizeof(wchar_t) * sz);
 | |
|     sz = PyUnicode_AsWideChar((PyUnicodeObject *)pVal, a, sz - 1);
 | |
|     free(ptr);
 | |
| #else
 | |
|     const char *s = PyUnicode_AsUTF8(pVal);
 | |
| #endif
 | |
| #if 0
 | |
|     if (false && Yap_AtomInUse(s))
 | |
|       rc = rc && PL_unify_atom_chars(t, s);
 | |
|     else
 | |
| #endif
 | |
|       if  (pyStringToString)
 | |
|         return MkStringTerm(s);
 | |
|       else
 | |
| 	return MkAtomTerm(Yap_LookupAtom(s));
 | |
|   }
 | |
|   else if (PyByteArray_Check(pVal)) {
 | |
|     return MkStringTerm(PyByteArray_AsString(pVal));
 | |
| #if PY_MAJOR_VERSION < 3
 | |
|   }
 | |
|   else if (PyString_Check(pVal)) {
 | |
|     return MkStringTerm(PyString_AsString(pVal));
 | |
| #endif
 | |
|   }
 | |
|   else if (PyTuple_Check(pVal)) {
 | |
|     Py_ssize_t sz = PyTuple_Size(pVal);
 | |
|     const char *s;
 | |
|     s = Py_TYPE(pVal)->tp_name;
 | |
|     if (s == NULL)
 | |
|       s = "t";
 | |
|     if (sz == 0) {
 | |
|       return MkAtomTerm(YAP_LookupAtom(Py_TYPE(pVal)->tp_name));
 | |
|     }
 | |
|     else {
 | |
|       Functor f = Yap_MkFunctor(Yap_LookupAtom(s), sz);
 | |
|       Term t = Yap_MkNewApplTerm(f, sz);
 | |
|       long i;
 | |
|       CELL *ptr = RepAppl(t) + 1;
 | |
|       for (i = 0; i < sz; i++) {
 | |
|         PyObject *p = PyTuple_GetItem(pVal, i);
 | |
|         if (p == NULL) {
 | |
|           PyErr_Clear();
 | |
|           return false;
 | |
|         }
 | |
|         *ptr++ = python_to_term__(p);
 | |
|       }
 | |
|       return t;
 | |
|     }
 | |
|     // PL_reset_term_refs(to);
 | |
|     // 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)
 | |
|       return repr_term(pVal);
 | |
|     Term t = TermNil;
 | |
|     for (i = sz; i > 0; ) {
 | |
|       -- i;
 | |
|       PyObject *p = PyList_GetItem(pVal, i);
 | |
|       if (p == NULL) {
 | |
|         PyErr_Clear();
 | |
|         return false;
 | |
|       }
 | |
|       if (!python_to_term__(p))
 | |
|         return false;
 | |
| 
 | |
|       t = MkPairTerm(python_to_term__(p), t);
 | |
|     }
 | |
|     return t;
 | |
|   }
 | |
|   else if (PyDict_Check(pVal)) {
 | |
|     Py_ssize_t pos = 0, tot = PyDict_Size(pVal);
 | |
|     PyObject *key, *value;
 | |
|     Term f, *opt = &f, t, to;
 | |
|     if (tot == 0)
 | |
|       return MkAtomTerm( Yap_LookupAtom("{}"));
 | |
|     while (PyDict_Next(pVal, &pos, &key, &value)) {
 | |
|       Term t0[2];
 | |
|       t0[0] = python_to_term__(key);
 | |
|       t0[1] = python_to_term__(value);
 | |
|       to = Yap_MkApplTerm(FunctorModule, 2, t0);
 | |
|       if (pos < tot) {
 | |
|         t = Yap_MkNewApplTerm(FunctorComma, 2);
 | |
|         CELL *pt = RepAppl(t) + 1;
 | |
|         pt[0] = to;
 | |
|         *opt = t;
 | |
| 	opt = pt+1;
 | |
|       } else {
 | |
|         if (pos == 0) {
 | |
|           return repr_term(pVal);
 | |
|         } 
 | |
| 
 | |
|         *opt = to;
 | |
| 	break;
 | |
|       }
 | |
|     }
 | |
|     return Yap_MkApplTerm(FunctorBraces, 1, &f);
 | |
|   }
 | |
|   return repr_term(pVal);
 | |
| 
 | |
| }
 | |
| 
 | |
| foreign_t python_to_term(PyObject *pVal, term_t t) {
 | |
|   Term o = python_to_term__(pVal);
 | |
|   return YAP_Unify(o,YAP_GetFromSlot(t));
 | |
| }
 | |
| 
 | |
| // extern bool Yap_do_low_level_trace;
 | |
| 
 | |
| X_API YAP_Term pythonToYAP(PyObject *pVal) {
 | |
|   // Yap_do_low_level_trace=1;
 | |
|   /* fputs(" ***    ", stderr); */
 | |
|   /* PyObject_Print(pVal, stderr, 0); */
 | |
|   /* fputs("***>>\n", stderr); */
 | |
|   if (pVal == NULL)
 | |
|     Yap_ThrowError(SYSTEM_ERROR_INTERNAL, 0, NULL);
 | |
|   yhandle_t h0 = Yap_CurrentHandle();
 | |
|   Term t =  python_to_term__(pVal);
 | |
|   /* fputs("<< ***    ", stderr); */
 | |
|   /* Yap_DebugPlWrite(t); */
 | |
|   /* fputs(" ***\n", stderr); */
 | |
|   // Py_DECREF(pVal);
 | |
|   Yap_CloseHandles(h0);
 | |
|   return t;
 | |
| }
 | |
| 
 | |
| 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) {
 | |
|   bool rc = true;
 | |
|   PyErr_Print();
 | |
|   term_t inp = Yap_CurrentHandle();
 | |
|   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]
 | |
|       PL_reset_term_refs(inp);
 | |
| 
 | |
|     rc = python_to_term(exp, t);
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   case PL_STRING: {             
 | |
|     char *s = NULL;
 | |
|     size_t l;
 | |
|     PL_get_string_chars(t, &s,&l);
 | |
|     if (!context)
 | |
|       context = py_Main;
 | |
|     if (PyObject_SetAttrString(context, s, exp) == 0) {
 | |
|       PL_reset_term_refs(inp);
 | |
|       ;
 | |
|       rc = true;
 | |
|     } else {
 | |
|       PyErr_Print();
 | |
|       PL_reset_term_refs(inp);
 | |
|       rc = false;
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case PL_ATOM: {
 | |
|     char *s = NULL;
 | |
|     PL_get_atom_chars(t, &s);
 | |
|     if (!context)
 | |
|       context = py_Main;
 | |
|     if (PyObject_SetAttrString(context, s, exp) == 0){
 | |
|       rc = true;
 | |
|     } else {
 | |
|       rc = false;
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case PL_INTEGER:
 | |
|   case PL_FLOAT:
 | |
|     // domain or type erro?
 | |
|     rc = false;
 | |
|     break;
 | |
|   default: {
 | |
|     term_t tail = PL_new_term_ref(), arg = PL_new_term_ref();
 | |
|     size_t len, i;
 | |
| 
 | |
|     if (PL_is_pair(t)) {
 | |
|       if (PL_skip_list(t, tail, &len) &&
 | |
|           PL_get_nil(tail)) { //             true list
 | |
| 
 | |
|         if (PySequence_Check(exp) && PySequence_Length(exp) == len)
 | |
| 
 | |
|           for (i = 0; i < len; i++) {
 | |
|             PyObject *p;
 | |
|             if (!PL_get_list(t, arg, t)) {
 | |
|               p = Py_None;
 | |
|               rc = false;
 | |
|             } else {
 | |
|               if ((p = PySequence_GetItem(exp, i)) == NULL)
 | |
|                 p = Py_None;
 | |
|               rc  = rc &&  python_assign(arg, p, context);
 | |
|             }
 | |
|           }
 | |
|       } else {
 | |
|         while(PL_is_pair(t)) {
 | |
|           context = find_obj(context, t, false);
 | |
|         }
 | |
|         rc = python_assign(t, exp, context);
 | |
|       }
 | |
| 
 | |
|     } else {
 | |
|       functor_t fun;
 | |
| 
 | |
|       if (!PL_get_functor(t, &fun)) {
 | |
|         rc = false;
 | |
|       }
 | |
| 
 | |
|       if (fun == FUNCTOR_sqbrackets2) {
 | |
|         //  tail is the object o
 | |
|         if (!PL_get_arg(2, t, tail)) {
 | |
|           rc =  false;
 | |
|         } else {
 | |
|           PyObject *o = term_to_python(tail, true, context, false);
 | |
|           // t now refers to the index
 | |
|           if (!PL_get_arg(1, t, t) || !PL_get_list(t, t, tail) ||
 | |
|               !PL_get_nil(tail)) {
 | |
|             rc = false;
 | |
|           } else {
 | |
|             PyObject *i = term_to_python(t, true, NULL, false);
 | |
|             // check numeric
 | |
|             if (PySequence_Check(o) && PyLong_Check(i)) {
 | |
|               long int j;
 | |
|               j = PyLong_AsLong(i);
 | |
|               rc =  PySequence_SetItem(o, j, exp) == 0;
 | |
|             }
 | |
| #if PY_MAJOR_VERSION < 3
 | |
|             if (PySequence_Check(o) && PyInt_Check(i)) {
 | |
|               long int j;
 | |
|               j = PyInt_AsLong(i);
 | |
|               szzb  rc =  PySequence_SetItem(o, i, exp) == 0;
 | |
|             } else
 | |
| #endif
 | |
|               if (PyDict_Check(o)) {
 | |
|                 if (PyDict_SetItem(o, i, exp) == 0) {
 | |
|                   PL_reset_term_refs(inp);
 | |
|                   ;
 | |
|                   rc = true;
 | |
|                 }
 | |
|               }
 | |
|             if (PyObject_SetAttr(o, i, exp) == 0) {
 | |
|               PL_reset_term_refs(inp);
 | |
|               ;
 | |
|               rc = true;
 | |
|             }
 | |
|           } 
 | |
|         } 
 | |
|       } else {
 | |
| 
 | |
|         atom_t s;
 | |
|         int n, i;
 | |
|         PL_get_name_arity(t, &s, &n);
 | |
|         PyObject *o = term_to_python(t, true, context, true);
 | |
|         PyErr_Print();
 | |
|         if (PySequence_Check(o) && PySequence_Length(o) == n) {
 | |
|           for (i = 1; i <= n; i++) {
 | |
|             PyObject *p;
 | |
|             if (!PL_get_arg(i, t, arg)) {
 | |
|               o = NULL;
 | |
|               p = Py_None;
 | |
|             }
 | |
| 	    
 | |
|             if ((p = PySequence_GetItem(exp, i - 1)) == NULL)
 | |
|               p = Py_None;
 | |
|             rc = python_assign(arg, p, NULL);
 | |
| 
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   }
 | |
| 
 | |
|   PyErr_Print();
 | |
|   PL_reset_term_refs(inp);
 | |
|   
 | |
|    
 | |
|   return rc;
 | |
| }
 |