+PHP_METHOD(HttpException, __toString)
+{
+ phpstr full_str;
+ zend_class_entry *ce;
+ zval *zobj = getThis(), *retval = NULL, *m, *f, *l;
+
+ phpstr_init(&full_str);
+
+ do {
+ ce = Z_OBJCE_P(zobj);
+
+ m = zend_read_property(ce, zobj, "message", lenof("message"), 0 TSRMLS_CC);
+ f = zend_read_property(ce, zobj, "file", lenof("file"), 0 TSRMLS_CC);
+ l = zend_read_property(ce, zobj, "line", lenof("line"), 0 TSRMLS_CC);
+
+ if (m && f && l && Z_TYPE_P(m) == IS_STRING && Z_TYPE_P(f) == IS_STRING && Z_TYPE_P(l) == IS_LONG) {
+ if (zobj != getThis()) {
+ phpstr_appends(&full_str, " inner ");
+ }
+
+ phpstr_appendf(&full_str, "exception '%.*s' with message '%.*s' in %.*s:%ld" PHP_EOL,
+ ce->name_length, ce->name, Z_STRLEN_P(m), Z_STRVAL_P(m), Z_STRLEN_P(f), Z_STRVAL_P(f), Z_LVAL_P(l)
+ );
+ } else {
+ break;
+ }
+
+ zobj = zend_read_property(ce, zobj, "innerException", lenof("innerException"), 0 TSRMLS_CC);
+ } while (Z_TYPE_P(zobj) == IS_OBJECT);
+
+ if (zend_call_method_with_0_params(&getThis(), Z_OBJCE_P(getThis()), NULL, "gettraceasstring", &retval) && Z_TYPE_P(retval) == IS_STRING) {
+ phpstr_appends(&full_str, "Stack trace:" PHP_EOL);
+ phpstr_append(&full_str, Z_STRVAL_P(retval), Z_STRLEN_P(retval));
+ zval_ptr_dtor(&retval);
+ }
+
+ RETURN_PHPSTR_VAL(&full_str);
+}
+