| 
									
										
										
										
											2008-12-28 08:41:49 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * $Id$ | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2008-12-28 08:41:49 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Campbell Barton | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "bpy_util.h"
 | 
					
						
							|  |  |  | #include "BLI_dynstr.h"
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2009-04-19 13:37:59 +00:00
										 |  |  | #include "BKE_report.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-28 08:41:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-25 13:48:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BKE_context.h"
 | 
					
						
							|  |  |  | bContext*	__py_context = NULL; | 
					
						
							|  |  |  | bContext*	BPy_GetContext(void) { return __py_context; }; | 
					
						
							|  |  |  | void		BPy_SetContext(bContext *C) { __py_context= C; }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-28 08:41:49 +00:00
										 |  |  | /* for debugging */ | 
					
						
							|  |  |  | void PyObSpit(char *name, PyObject *var) { | 
					
						
							|  |  |  | 	fprintf(stderr, "<%s> : ", name); | 
					
						
							|  |  |  | 	if (var==NULL) { | 
					
						
							|  |  |  | 		fprintf(stderr, "<NIL>"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		PyObject_Print(var, stderr, 0); | 
					
						
							| 
									
										
										
										
											2009-08-15 16:36:25 +00:00
										 |  |  | 		fprintf(stderr, " ref:%d ", (int)var->ob_refcnt); | 
					
						
							| 
									
										
										
										
											2009-06-22 04:26:48 +00:00
										 |  |  | 		fprintf(stderr, " ptr:%p", (void *)var); | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		fprintf(stderr, " type:"); | 
					
						
							|  |  |  | 		if(Py_TYPE(var)) | 
					
						
							|  |  |  | 			fprintf(stderr, "%s", Py_TYPE(var)->tp_name); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			fprintf(stderr, "<NIL>"); | 
					
						
							| 
									
										
										
										
											2008-12-28 08:41:49 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	fprintf(stderr, "\n"); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-02-28 13:27:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-02 20:18:29 +00:00
										 |  |  | void PyLineSpit(void) { | 
					
						
							| 
									
										
										
										
											2009-03-11 17:28:37 +00:00
										 |  |  | 	char *filename; | 
					
						
							| 
									
										
										
										
											2009-03-02 20:18:29 +00:00
										 |  |  | 	int lineno; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PyErr_Clear(); | 
					
						
							|  |  |  | 	BPY_getFileAndNum(&filename, &lineno); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	fprintf(stderr, "%s:%d\n", filename, lineno); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-28 13:27:45 +00:00
										 |  |  | void BPY_getFileAndNum(char **filename, int *lineno) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	PyObject *getframe, *frame; | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 	PyObject *f_lineno= NULL, *co_filename= NULL; | 
					
						
							| 
									
										
										
										
											2009-02-28 13:27:45 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if (filename)	*filename= NULL; | 
					
						
							|  |  |  | 	if (lineno)		*lineno = -1; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	getframe = PySys_GetObject("_getframe"); // borrowed
 | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 	if (getframe==NULL) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:17:49 +00:00
										 |  |  | 		PyErr_Clear(); | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	frame = PyObject_CallObject(getframe, NULL); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:17:49 +00:00
										 |  |  | 	if (frame==NULL) { | 
					
						
							|  |  |  | 		PyErr_Clear(); | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:17:49 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if (filename) { | 
					
						
							|  |  |  | 		co_filename= PyObject_GetAttrStringArgs(frame, 1, "f_code", "co_filename"); | 
					
						
							|  |  |  | 		if (co_filename==NULL) { | 
					
						
							|  |  |  | 			PyErr_SetString(PyExc_SystemError, "Could not access sys._getframe().f_code.co_filename"); | 
					
						
							|  |  |  | 			Py_DECREF(frame); | 
					
						
							|  |  |  | 			return; | 
					
						
							| 
									
										
										
										
											2009-02-28 13:27:45 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		*filename = _PyUnicode_AsString(co_filename); | 
					
						
							|  |  |  | 		Py_DECREF(co_filename); | 
					
						
							| 
									
										
										
										
											2009-02-28 13:27:45 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 	if (lineno) { | 
					
						
							|  |  |  | 		f_lineno= PyObject_GetAttrString(frame, "f_lineno"); | 
					
						
							|  |  |  | 		if (f_lineno==NULL) { | 
					
						
							|  |  |  | 			PyErr_SetString(PyExc_SystemError, "Could not access sys._getframe().f_lineno"); | 
					
						
							|  |  |  | 			Py_DECREF(frame); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		*lineno = (int)PyLong_AsSsize_t(f_lineno); | 
					
						
							|  |  |  | 		Py_DECREF(f_lineno); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-03-21 16:03:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Py_DECREF(frame); | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Would be nice if python had this built in */ | 
					
						
							|  |  |  | PyObject *PyObject_GetAttrStringArgs(PyObject *o, Py_ssize_t n, ...) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Py_ssize_t i; | 
					
						
							|  |  |  | 	PyObject *item= o; | 
					
						
							|  |  |  | 	char *attr; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	va_list vargs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	va_start(vargs, n); | 
					
						
							|  |  |  | 	for (i=0; i<n; i++) { | 
					
						
							|  |  |  | 		attr = va_arg(vargs, char *); | 
					
						
							|  |  |  | 		item = PyObject_GetAttrString(item, attr); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (item)  | 
					
						
							|  |  |  | 			Py_DECREF(item); | 
					
						
							|  |  |  | 		else /* python will set the error value here */ | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	va_end(vargs); | 
					
						
							| 
									
										
										
										
											2009-02-28 13:27:45 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-03-21 06:55:30 +00:00
										 |  |  | 	Py_XINCREF(item); /* final value has is increfed, to match PyObject_GetAttrString */ | 
					
						
							| 
									
										
										
										
											2009-03-18 22:22:58 +00:00
										 |  |  | 	return item; | 
					
						
							| 
									
										
										
										
											2009-02-28 13:27:45 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int BPY_class_validate(const char *class_type, PyObject *class, PyObject *base_class, BPY_class_attr_check* class_attrs, PyObject **py_class_attrs) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-04-08 18:45:41 +00:00
										 |  |  | 	PyObject *item, *fitem; | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | 	PyObject *py_arg_count; | 
					
						
							|  |  |  | 	int i, arg_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (base_class) { | 
					
						
							|  |  |  | 		if (!PyObject_IsSubclass(class, base_class)) { | 
					
						
							|  |  |  | 			PyObject *name= PyObject_GetAttrString(base_class, "__name__"); | 
					
						
							|  |  |  | 			PyErr_Format( PyExc_AttributeError, "expected %s subclass of class \"%s\"", class_type, name ? _PyUnicode_AsString(name):"<UNKNOWN>"); | 
					
						
							|  |  |  | 			Py_XDECREF(name); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	for(i= 0;class_attrs->name; class_attrs++, i++) { | 
					
						
							|  |  |  | 		item = PyObject_GetAttrString(class, class_attrs->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (py_class_attrs) | 
					
						
							|  |  |  | 			py_class_attrs[i]= item; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (item==NULL) { | 
					
						
							|  |  |  | 			if ((class_attrs->flag & BPY_CLASS_ATTR_OPTIONAL)==0) { | 
					
						
							|  |  |  | 				PyErr_Format( PyExc_AttributeError, "expected %s class to have an \"%s\" attribute", class_type, class_attrs->name); | 
					
						
							|  |  |  | 				return -1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			PyErr_Clear(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			Py_DECREF(item); /* no need to keep a ref, the class owns it */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if((item==Py_None) && (class_attrs->flag & BPY_CLASS_ATTR_NONE_OK)) { | 
					
						
							|  |  |  | 				/* dont do anything, this is ok, dont bother checking other types */ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				switch(class_attrs->type) { | 
					
						
							|  |  |  | 				case 's': | 
					
						
							|  |  |  | 					if (PyUnicode_Check(item)==0) { | 
					
						
							|  |  |  | 						PyErr_Format( PyExc_AttributeError, "expected %s class \"%s\" attribute to be a string", class_type, class_attrs->name); | 
					
						
							|  |  |  | 						return -1; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-19 13:32:02 +00:00
										 |  |  | 					if(class_attrs->len != -1 && class_attrs->len < PyUnicode_GetSize(item)) { | 
					
						
							|  |  |  | 						PyErr_Format( PyExc_AttributeError, "expected %s class \"%s\" attribute string to be shorter then %d", class_type, class_attrs->name, class_attrs->len); | 
					
						
							|  |  |  | 						return -1; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case 'l': | 
					
						
							|  |  |  | 					if (PyList_Check(item)==0) { | 
					
						
							|  |  |  | 						PyErr_Format( PyExc_AttributeError, "expected %s class \"%s\" attribute to be a list", class_type, class_attrs->name); | 
					
						
							|  |  |  | 						return -1; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-07-19 13:32:02 +00:00
										 |  |  | 					if(class_attrs->len != -1 && class_attrs->len < PyList_GET_SIZE(item)) { | 
					
						
							|  |  |  | 						PyErr_Format( PyExc_AttributeError, "expected %s class \"%s\" attribute list to be shorter then %d", class_type, class_attrs->name, class_attrs->len); | 
					
						
							|  |  |  | 						return -1; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case 'f': | 
					
						
							| 
									
										
										
										
											2009-04-08 18:45:41 +00:00
										 |  |  | 					if (PyMethod_Check(item)) | 
					
						
							|  |  |  | 						fitem= PyMethod_Function(item); /* py 2.x */ | 
					
						
							|  |  |  | 					else | 
					
						
							|  |  |  | 						fitem= item; /* py 3.x */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if (PyFunction_Check(fitem)==0) { | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | 						PyErr_Format( PyExc_AttributeError, "expected %s class \"%s\" attribute to be a function", class_type, class_attrs->name); | 
					
						
							|  |  |  | 						return -1; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if (class_attrs->arg_count >= 0) { /* -1 if we dont care*/ | 
					
						
							| 
									
										
										
										
											2009-04-08 18:45:41 +00:00
										 |  |  | 						py_arg_count = PyObject_GetAttrString(PyFunction_GET_CODE(fitem), "co_argcount"); | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | 						arg_count = PyLong_AsSsize_t(py_arg_count); | 
					
						
							|  |  |  | 						Py_DECREF(py_arg_count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if (arg_count != class_attrs->arg_count) { | 
					
						
							|  |  |  | 							PyErr_Format( PyExc_AttributeError, "expected %s class \"%s\" function to have %d args", class_type, class_attrs->name, class_attrs->arg_count); | 
					
						
							|  |  |  | 							return -1; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* returns the exception string as a new PyUnicode object, depends on external StringIO module */ | 
					
						
							|  |  |  | PyObject *BPY_exception_buffer(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	PyObject *stdout_backup = PySys_GetObject("stdout"); /* borrowed */ | 
					
						
							|  |  |  | 	PyObject *stderr_backup = PySys_GetObject("stderr"); /* borrowed */ | 
					
						
							|  |  |  | 	PyObject *string_io = NULL; | 
					
						
							|  |  |  | 	PyObject *string_io_buf = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-11 13:57:56 +00:00
										 |  |  | 	PyObject *string_io_mod= NULL; | 
					
						
							|  |  |  | 	PyObject *string_io_getvalue= NULL; | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	PyObject *error_type, *error_value, *error_traceback; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (!PyErr_Occurred()) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	PyErr_Fetch(&error_type, &error_value, &error_traceback); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	PyErr_Clear(); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2010-03-16 17:19:42 +00:00
										 |  |  | 	/* import io
 | 
					
						
							|  |  |  | 	 * string_io = io.StringIO() | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	 */ | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(! (string_io_mod= PyImport_ImportModule("io")) ) { | 
					
						
							| 
									
										
										
										
											2009-07-11 13:57:56 +00:00
										 |  |  | 		goto error_cleanup; | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	} else if (! (string_io = PyObject_CallMethod(string_io_mod, "StringIO", NULL))) { | 
					
						
							| 
									
										
										
										
											2009-07-11 13:57:56 +00:00
										 |  |  | 		goto error_cleanup; | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	} else if (! (string_io_getvalue= PyObject_GetAttrString(string_io, "getvalue"))) { | 
					
						
							| 
									
										
										
										
											2009-07-11 13:57:56 +00:00
										 |  |  | 		goto error_cleanup; | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	Py_INCREF(stdout_backup); // since these were borrowed we dont want them freed when replaced.
 | 
					
						
							|  |  |  | 	Py_INCREF(stderr_backup); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	PySys_SetObject("stdout", string_io); // both of these are free'd when restoring
 | 
					
						
							|  |  |  | 	PySys_SetObject("stderr", string_io); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	PyErr_Restore(error_type, error_value, error_traceback); | 
					
						
							|  |  |  | 	PyErr_Print(); /* print the error */ | 
					
						
							|  |  |  | 	PyErr_Clear(); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	string_io_buf = PyObject_CallObject(string_io_getvalue, NULL); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	PySys_SetObject("stdout", stdout_backup); | 
					
						
							|  |  |  | 	PySys_SetObject("stderr", stderr_backup); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	Py_DECREF(stdout_backup); /* now sys owns the ref again */ | 
					
						
							|  |  |  | 	Py_DECREF(stderr_backup); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	Py_DECREF(string_io_mod); | 
					
						
							|  |  |  | 	Py_DECREF(string_io_getvalue); | 
					
						
							|  |  |  | 	Py_DECREF(string_io); /* free the original reference */ | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	PyErr_Clear(); | 
					
						
							|  |  |  | 	return string_io_buf; | 
					
						
							| 
									
										
										
										
											2009-07-11 13:57:56 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | error_cleanup: | 
					
						
							|  |  |  | 	/* could not import the module so print the error and close */ | 
					
						
							|  |  |  | 	Py_XDECREF(string_io_mod); | 
					
						
							|  |  |  | 	Py_XDECREF(string_io); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	PyErr_Restore(error_type, error_value, error_traceback); | 
					
						
							|  |  |  | 	PyErr_Print(); /* print the error */ | 
					
						
							|  |  |  | 	PyErr_Clear(); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | char *BPy_enum_as_string(EnumPropertyItem *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	DynStr *dynstr= BLI_dynstr_new(); | 
					
						
							|  |  |  | 	EnumPropertyItem *e; | 
					
						
							|  |  |  | 	char *cstring; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (e= item; item->identifier; item++) { | 
					
						
							| 
									
										
										
											
												RNA
* Enums can now be dynamically created in the _itemf callback,
  using RNA_enum_item(s)_add, RNA_enum_item_end. All places asking
  for enum items now need to potentially free the items.
* This callback now also gets context, this was added specifically
  for operators. This doesn't fit design well at all, needed to do
  some ugly hacks, but can't find a good solution at the moment.
* All enums must have a default list of items too, even with an
  _itemf callback, for docs and fallback in case there is no context.
* Used by MESH_OT_merge, MESH_OT_select_similar, TFM_OT_select_orientation.
* Also changes some operator properties that were enums to booleas
  (unselected, deselect), to make them consistent with other ops.
											
										 
											2009-07-10 19:56:13 +00:00
										 |  |  | 		if(item->identifier[0]) | 
					
						
							|  |  |  | 			BLI_dynstr_appendf(dynstr, (e==item)?"'%s'":", '%s'", item->identifier); | 
					
						
							| 
									
										
										
										
											2009-04-01 12:43:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cstring = BLI_dynstr_get_cstring(dynstr); | 
					
						
							|  |  |  | 	BLI_dynstr_free(dynstr); | 
					
						
							|  |  |  | 	return cstring; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-04-19 13:37:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int BPy_reports_to_error(ReportList *reports) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *report_str; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	report_str= BKE_reports_string(reports, RPT_ERROR); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(report_str) { | 
					
						
							|  |  |  | 		PyErr_SetString(PyExc_SystemError, report_str); | 
					
						
							|  |  |  | 		MEM_freeN(report_str); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (report_str != NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int BPy_errors_to_report(ReportList *reports) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	PyObject *pystring; | 
					
						
							| 
									
										
										
										
											2009-11-23 17:36:44 +00:00
										 |  |  | 	PyObject *pystring_format= NULL; // workaround, see below
 | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	char *cstring; | 
					
						
							| 
									
										
										
										
											2009-11-04 22:36:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	char *filename; | 
					
						
							|  |  |  | 	int lineno; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	if (!PyErr_Occurred()) | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* less hassle if we allow NULL */ | 
					
						
							|  |  |  | 	if(reports==NULL) { | 
					
						
							|  |  |  | 		PyErr_Print(); | 
					
						
							|  |  |  | 		PyErr_Clear(); | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	pystring= BPY_exception_buffer(); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(pystring==NULL) { | 
					
						
							|  |  |  | 		BKE_report(reports, RPT_ERROR, "unknown py-exception, could not convert"); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-11-04 22:36:46 +00:00
										 |  |  | 	BPY_getFileAndNum(&filename, &lineno); | 
					
						
							| 
									
										
										
										
											2009-11-23 17:36:44 +00:00
										 |  |  | 	if(filename==NULL) | 
					
						
							|  |  |  | 		filename= "<unknown location>"; | 
					
						
							| 
									
										
										
										
											2009-11-04 22:36:46 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	cstring= _PyUnicode_AsString(pystring); | 
					
						
							| 
									
										
										
										
											2009-11-23 17:36:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 0 // ARG!. workaround for a bug in blenders use of vsnprintf
 | 
					
						
							| 
									
										
										
										
											2009-11-04 22:36:46 +00:00
										 |  |  | 	BKE_reportf(reports, RPT_ERROR, "%s\nlocation:%s:%d\n", cstring, filename, lineno); | 
					
						
							| 
									
										
										
										
											2009-11-23 17:36:44 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 	pystring_format= PyUnicode_FromFormat("%s\nlocation:%s:%d\n", cstring, filename, lineno); | 
					
						
							|  |  |  | 	cstring= _PyUnicode_AsString(pystring_format); | 
					
						
							|  |  |  | 	BKE_report(reports, RPT_ERROR, cstring); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-11-04 22:36:46 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	fprintf(stderr, "%s\nlocation:%s:%d\n", cstring, filename, lineno); // not exactly needed. just for testing
 | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	Py_DECREF(pystring); | 
					
						
							| 
									
										
										
										
											2009-11-23 17:36:44 +00:00
										 |  |  | 	Py_DECREF(pystring_format); // workaround
 | 
					
						
							| 
									
										
										
										
											2009-06-14 12:53:47 +00:00
										 |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-11-04 22:36:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-01 10:51:34 +00:00
										 |  |  | /* array utility function */ | 
					
						
							|  |  |  | int BPyAsPrimitiveArray(void *array, PyObject *value, int length, PyTypeObject *type, char *error_prefix) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	PyObject *value_fast; | 
					
						
							|  |  |  | 	int value_len; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(!(value_fast=PySequence_Fast(value, error_prefix))) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	value_len= PySequence_Fast_GET_SIZE(value_fast); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(value_len != length) { | 
					
						
							|  |  |  | 		Py_DECREF(value); | 
					
						
							|  |  |  | 		PyErr_Format(PyExc_TypeError, "%s: invalid sequence length. expected %d, got %d.", error_prefix, length, value_len); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* for each type */ | 
					
						
							|  |  |  | 	if(type == &PyFloat_Type) { | 
					
						
							|  |  |  | 		float *array_float= array; | 
					
						
							|  |  |  | 		for(i=0; i<length; i++) { | 
					
						
							|  |  |  | 			array_float[i] = PyFloat_AsDouble(PySequence_Fast_GET_ITEM(value_fast, i)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(type == &PyLong_Type) { | 
					
						
							|  |  |  | 		int *array_int= array; | 
					
						
							|  |  |  | 		for(i=0; i<length; i++) { | 
					
						
							|  |  |  | 			array_int[i] = PyLong_AsSsize_t(PySequence_Fast_GET_ITEM(value_fast, i)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(type == &PyBool_Type) { | 
					
						
							|  |  |  | 		int *array_bool= array; | 
					
						
							|  |  |  | 		for(i=0; i<length; i++) { | 
					
						
							|  |  |  | 			array_bool[i] = (PyLong_AsSsize_t(PySequence_Fast_GET_ITEM(value_fast, i)) != 0); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		Py_DECREF(value_fast); | 
					
						
							|  |  |  | 		PyErr_Format(PyExc_TypeError, "%s: internal error %s is invalid.", error_prefix, type->tp_name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Py_DECREF(value_fast); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(PyErr_Occurred()) { | 
					
						
							|  |  |  | 		PyErr_Format(PyExc_TypeError, "%s: one or more items could not be used as a %s.", error_prefix, type->tp_name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } |