- added HttpException::__toString() which takes care about any inner exceptions
[m6w6/ext-http] / http_exception_object.c
1 /*
2 +--------------------------------------------------------------------+
3 | PECL :: http |
4 +--------------------------------------------------------------------+
5 | Redistribution and use in source and binary forms, with or without |
6 | modification, are permitted provided that the conditions mentioned |
7 | in the accompanying LICENSE file are met. |
8 +--------------------------------------------------------------------+
9 | Copyright (c) 2004-2006, Michael Wallner <mike@php.net> |
10 +--------------------------------------------------------------------+
11 */
12
13 /* $Id$ */
14
15 #include "php_http.h"
16
17 #ifdef ZEND_ENGINE_2
18
19 #include "zend_interfaces.h"
20 #include "php_http_exception_object.h"
21
22 zend_class_entry *http_exception_object_ce;
23 zend_class_entry *HTTP_EX_CE(runtime);
24 zend_class_entry *HTTP_EX_CE(header);
25 zend_class_entry *HTTP_EX_CE(malformed_headers);
26 zend_class_entry *HTTP_EX_CE(request_method);
27 zend_class_entry *HTTP_EX_CE(message_type);
28 zend_class_entry *HTTP_EX_CE(invalid_param);
29 zend_class_entry *HTTP_EX_CE(encoding);
30 zend_class_entry *HTTP_EX_CE(request);
31 zend_class_entry *HTTP_EX_CE(request_pool);
32 zend_class_entry *HTTP_EX_CE(socket);
33 zend_class_entry *HTTP_EX_CE(response);
34 zend_class_entry *HTTP_EX_CE(url);
35 zend_class_entry *HTTP_EX_CE(querystring);
36
37 #define HTTP_EMPTY_ARGS(method) HTTP_EMPTY_ARGS_EX(HttpException, method, 0)
38 #define HTTP_EXCEPTION_ME(method, visibility) PHP_ME(HttpException, method, HTTP_ARGS(HttpException, method), visibility)
39
40 HTTP_EMPTY_ARGS(__toString);
41
42 #define OBJ_PROP_CE http_deflatestream_object_ce
43 zend_function_entry http_exception_object_fe[] = {
44 HTTP_EXCEPTION_ME(__toString, ZEND_ACC_PUBLIC)
45
46 EMPTY_FUNCTION_ENTRY
47 };
48
49 PHP_MINIT_FUNCTION(http_exception_object)
50 {
51 HTTP_REGISTER_CLASS(HttpException, http_exception_object, ZEND_EXCEPTION_GET_DEFAULT(), 0);
52
53 zend_declare_property_null(HTTP_EX_DEF_CE, "innerException", lenof("innerException"), ZEND_ACC_PUBLIC TSRMLS_CC);
54
55 HTTP_REGISTER_EXCEPTION(HttpRuntimeException, HTTP_EX_CE(runtime), HTTP_EX_DEF_CE);
56 HTTP_REGISTER_EXCEPTION(HttpInvalidParamException, HTTP_EX_CE(invalid_param), HTTP_EX_DEF_CE);
57 HTTP_REGISTER_EXCEPTION(HttpHeaderException, HTTP_EX_CE(header), HTTP_EX_DEF_CE);
58 HTTP_REGISTER_EXCEPTION(HttpMalformedHeadersException, HTTP_EX_CE(malformed_headers), HTTP_EX_DEF_CE);
59 HTTP_REGISTER_EXCEPTION(HttpRequestMethodException, HTTP_EX_CE(request_method), HTTP_EX_DEF_CE);
60 HTTP_REGISTER_EXCEPTION(HttpMessageTypeException, HTTP_EX_CE(message_type), HTTP_EX_DEF_CE);
61 HTTP_REGISTER_EXCEPTION(HttpEncodingException, HTTP_EX_CE(encoding), HTTP_EX_DEF_CE);
62 HTTP_REGISTER_EXCEPTION(HttpRequestException, HTTP_EX_CE(request), HTTP_EX_DEF_CE);
63 HTTP_REGISTER_EXCEPTION(HttpRequestPoolException, HTTP_EX_CE(request_pool), HTTP_EX_DEF_CE);
64 HTTP_REGISTER_EXCEPTION(HttpSocketException, HTTP_EX_CE(socket), HTTP_EX_DEF_CE);
65 HTTP_REGISTER_EXCEPTION(HttpResponseException, HTTP_EX_CE(response), HTTP_EX_DEF_CE);
66 HTTP_REGISTER_EXCEPTION(HttpUrlException, HTTP_EX_CE(url), HTTP_EX_DEF_CE);
67 HTTP_REGISTER_EXCEPTION(HttpQueryStringException, HTTP_EX_CE(querystring), HTTP_EX_DEF_CE);
68
69 HTTP_LONG_CONSTANT("HTTP_E_RUNTIME", HTTP_E_RUNTIME);
70 HTTP_LONG_CONSTANT("HTTP_E_INVALID_PARAM", HTTP_E_INVALID_PARAM);
71 HTTP_LONG_CONSTANT("HTTP_E_HEADER", HTTP_E_HEADER);
72 HTTP_LONG_CONSTANT("HTTP_E_MALFORMED_HEADERS", HTTP_E_MALFORMED_HEADERS);
73 HTTP_LONG_CONSTANT("HTTP_E_REQUEST_METHOD", HTTP_E_REQUEST_METHOD);
74 HTTP_LONG_CONSTANT("HTTP_E_MESSAGE_TYPE", HTTP_E_MESSAGE_TYPE);
75 HTTP_LONG_CONSTANT("HTTP_E_ENCODING", HTTP_E_ENCODING);
76 HTTP_LONG_CONSTANT("HTTP_E_REQUEST", HTTP_E_REQUEST);
77 HTTP_LONG_CONSTANT("HTTP_E_REQUEST_POOL", HTTP_E_REQUEST_POOL);
78 HTTP_LONG_CONSTANT("HTTP_E_SOCKET", HTTP_E_SOCKET);
79 HTTP_LONG_CONSTANT("HTTP_E_RESPONSE", HTTP_E_RESPONSE);
80 HTTP_LONG_CONSTANT("HTTP_E_URL", HTTP_E_URL);
81 HTTP_LONG_CONSTANT("HTTP_E_QUERYSTRING", HTTP_E_QUERYSTRING);
82
83 return SUCCESS;
84 }
85
86 zend_class_entry *_http_exception_get_default()
87 {
88 return http_exception_object_ce;
89 }
90
91 zend_class_entry *_http_exception_get_for_code(long code)
92 {
93 zend_class_entry *ex = http_exception_object_ce;
94
95 switch (code) {
96 case HTTP_E_RUNTIME: ex = HTTP_EX_CE(runtime); break;
97 case HTTP_E_INVALID_PARAM: ex = HTTP_EX_CE(invalid_param); break;
98 case HTTP_E_HEADER: ex = HTTP_EX_CE(header); break;
99 case HTTP_E_MALFORMED_HEADERS: ex = HTTP_EX_CE(malformed_headers); break;
100 case HTTP_E_REQUEST_METHOD: ex = HTTP_EX_CE(request_method); break;
101 case HTTP_E_MESSAGE_TYPE: ex = HTTP_EX_CE(message_type); break;
102 case HTTP_E_ENCODING: ex = HTTP_EX_CE(encoding); break;
103 case HTTP_E_REQUEST: ex = HTTP_EX_CE(request); break;
104 case HTTP_E_REQUEST_POOL: ex = HTTP_EX_CE(request_pool); break;
105 case HTTP_E_SOCKET: ex = HTTP_EX_CE(socket); break;
106 case HTTP_E_RESPONSE: ex = HTTP_EX_CE(response); break;
107 case HTTP_E_URL: ex = HTTP_EX_CE(url); break;
108 case HTTP_E_QUERYSTRING: ex = HTTP_EX_CE(querystring); break;
109 }
110
111 return ex;
112 }
113
114 PHP_METHOD(HttpException, __toString)
115 {
116 phpstr full_str;
117 zend_class_entry *ce;
118 zval *zobj = getThis(), *retval = NULL, *m, *f, *l;
119
120 phpstr_init(&full_str);
121
122 do {
123 ce = Z_OBJCE_P(zobj);
124
125 if (zobj != getThis()) {
126 phpstr_appends(&full_str, " inner ");
127 }
128
129 m = zend_read_property(ce, zobj, "message", lenof("message"), 0 TSRMLS_CC);
130 f = zend_read_property(ce, zobj, "file", lenof("file"), 0 TSRMLS_CC);
131 l = zend_read_property(ce, zobj, "line", lenof("line"), 0 TSRMLS_CC);
132
133 if (m && f && l && Z_TYPE_P(m) == IS_STRING && Z_TYPE_P(f) == IS_STRING && Z_TYPE_P(l) == IS_LONG) {
134 phpstr_appendf(&full_str, "exception '%.*s' with message '%.*s' in %.*s:%ld" PHP_EOL,
135 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)
136 );
137 } else {
138 break;
139 }
140
141 zobj = zend_read_property(ce, zobj, "innerException", lenof("innerException"), 0 TSRMLS_CC);
142 } while (Z_TYPE_P(zobj) == IS_OBJECT);
143
144 if (zend_call_method_with_0_params(&getThis(), Z_OBJCE_P(getThis()), NULL, "gettraceasstring", &retval) && Z_TYPE_P(retval) == IS_STRING) {
145 phpstr_appends(&full_str, "Stack trace:" PHP_EOL);
146 phpstr_append(&full_str, Z_STRVAL_P(retval), Z_STRLEN_P(retval));
147 zval_ptr_dtor(&retval);
148 }
149
150 RETURN_PHPSTR_VAL(&full_str);
151 }
152 #endif
153
154 /*
155 * Local variables:
156 * tab-width: 4
157 * c-basic-offset: 4
158 * End:
159 * vim600: noet sw=4 ts=4 fdm=marker
160 * vim<600: noet sw=4 ts=4
161 */
162