resourcebundle_class.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | http://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Authors: Hans-Peter Oeri (University of St.Gallen) <hp@oeri.ch> |
  14. +----------------------------------------------------------------------+
  15. */
  16. #include <stdlib.h>
  17. #include <unicode/ures.h>
  18. #include <unicode/uenum.h>
  19. #include <zend.h>
  20. #include <Zend/zend_exceptions.h>
  21. #include <Zend/zend_interfaces.h>
  22. #include <php.h>
  23. #include "php_intl.h"
  24. #include "intl_data.h"
  25. #include "intl_common.h"
  26. #include "resourcebundle/resourcebundle.h"
  27. #include "resourcebundle/resourcebundle_iterator.h"
  28. #include "resourcebundle/resourcebundle_class.h"
  29. zend_class_entry *ResourceBundle_ce_ptr = NULL;
  30. static zend_object_handlers ResourceBundle_object_handlers;
  31. /* {{{ ResourceBundle_object_free */
  32. static void ResourceBundle_object_free( zend_object *object )
  33. {
  34. ResourceBundle_object *rb = php_intl_resourcebundle_fetch_object(object);
  35. // only free local errors
  36. intl_error_reset( INTL_DATA_ERROR_P(rb) );
  37. if (rb->me) {
  38. ures_close( rb->me );
  39. }
  40. if (rb->child) {
  41. ures_close( rb->child );
  42. }
  43. zend_object_std_dtor( &rb->zend );
  44. }
  45. /* }}} */
  46. /* {{{ ResourceBundle_object_create */
  47. static zend_object *ResourceBundle_object_create( zend_class_entry *ce )
  48. {
  49. ResourceBundle_object *rb;
  50. rb = zend_object_alloc(sizeof(ResourceBundle_object), ce);
  51. zend_object_std_init( &rb->zend, ce );
  52. object_properties_init( &rb->zend, ce);
  53. intl_error_init( INTL_DATA_ERROR_P(rb) );
  54. rb->me = NULL;
  55. rb->child = NULL;
  56. rb->zend.handlers = &ResourceBundle_object_handlers;
  57. return &rb->zend;
  58. }
  59. /* }}} */
  60. /* {{{ ResourceBundle_ctor */
  61. static int resourcebundle_ctor(INTERNAL_FUNCTION_PARAMETERS, zend_bool is_constructor)
  62. {
  63. const char *bundlename;
  64. size_t bundlename_len = 0;
  65. const char *locale;
  66. size_t locale_len = 0;
  67. zend_bool fallback = 1;
  68. int zpp_flags = is_constructor ? ZEND_PARSE_PARAMS_THROW : 0;
  69. zval *object = return_value;
  70. ResourceBundle_object *rb = Z_INTL_RESOURCEBUNDLE_P( object );
  71. intl_error_reset( NULL );
  72. if( zend_parse_parameters_ex( zpp_flags, ZEND_NUM_ARGS(), "s!s!|b",
  73. &locale, &locale_len, &bundlename, &bundlename_len, &fallback ) == FAILURE )
  74. {
  75. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  76. "resourcebundle_ctor: unable to parse input parameters", 0 );
  77. return FAILURE;
  78. }
  79. INTL_CHECK_LOCALE_LEN_OR_FAILURE(locale_len);
  80. if (locale == NULL) {
  81. locale = intl_locale_get_default();
  82. }
  83. if (bundlename_len >= MAXPATHLEN) {
  84. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR, "Bundle name too long", 0 );
  85. zval_ptr_dtor(return_value);
  86. ZVAL_NULL(return_value);
  87. return FAILURE;
  88. }
  89. if (fallback) {
  90. rb->me = ures_open(bundlename, locale, &INTL_DATA_ERROR_CODE(rb));
  91. } else {
  92. rb->me = ures_openDirect(bundlename, locale, &INTL_DATA_ERROR_CODE(rb));
  93. }
  94. INTL_CTOR_CHECK_STATUS(rb, "resourcebundle_ctor: Cannot load libICU resource bundle");
  95. if (!fallback && (INTL_DATA_ERROR_CODE(rb) == U_USING_FALLBACK_WARNING ||
  96. INTL_DATA_ERROR_CODE(rb) == U_USING_DEFAULT_WARNING)) {
  97. char *pbuf;
  98. intl_errors_set_code(NULL, INTL_DATA_ERROR_CODE(rb));
  99. spprintf(&pbuf, 0, "resourcebundle_ctor: Cannot load libICU resource "
  100. "'%s' without fallback from %s to %s",
  101. bundlename ? bundlename : "(default data)", locale,
  102. ures_getLocaleByType(
  103. rb->me, ULOC_ACTUAL_LOCALE, &INTL_DATA_ERROR_CODE(rb)));
  104. intl_errors_set_custom_msg(INTL_DATA_ERROR_P(rb), pbuf, 1);
  105. efree(pbuf);
  106. return FAILURE;
  107. }
  108. return SUCCESS;
  109. }
  110. /* }}} */
  111. /* {{{ arginfo_resourcebundle__construct */
  112. ZEND_BEGIN_ARG_INFO_EX( arginfo_resourcebundle___construct, 0, 0, 2 )
  113. ZEND_ARG_INFO( 0, locale )
  114. ZEND_ARG_INFO( 0, bundlename )
  115. ZEND_ARG_INFO( 0, fallback )
  116. ZEND_END_ARG_INFO()
  117. /* }}} */
  118. /* {{{ proto ResourceBundle::__construct( string $locale [, string $bundlename [, bool $fallback = true ]] )
  119. * ResourceBundle object constructor
  120. */
  121. PHP_METHOD( ResourceBundle, __construct )
  122. {
  123. zend_error_handling error_handling;
  124. zend_replace_error_handling(EH_THROW, IntlException_ce_ptr, &error_handling);
  125. return_value = getThis();
  126. if (resourcebundle_ctor(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1) == FAILURE) {
  127. if (!EG(exception)) {
  128. zend_throw_exception(IntlException_ce_ptr, "Constructor failed", 0);
  129. }
  130. }
  131. zend_restore_error_handling(&error_handling);
  132. }
  133. /* }}} */
  134. /* {{{ proto ResourceBundle ResourceBundle::create( string $locale [, string $bundlename [, bool $fallback = true ]] )
  135. proto ResourceBundle resourcebundle_create( string $locale [, string $bundlename [, bool $fallback = true ]] )
  136. */
  137. PHP_FUNCTION( resourcebundle_create )
  138. {
  139. object_init_ex( return_value, ResourceBundle_ce_ptr );
  140. if (resourcebundle_ctor(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0) == FAILURE) {
  141. zval_ptr_dtor(return_value);
  142. RETURN_NULL();
  143. }
  144. }
  145. /* }}} */
  146. /* {{{ resourcebundle_array_fetch */
  147. static void resourcebundle_array_fetch(zval *object, zval *offset, zval *return_value, int fallback)
  148. {
  149. int32_t meindex = 0;
  150. char * mekey = NULL;
  151. zend_bool is_numeric = 0;
  152. char *pbuf;
  153. ResourceBundle_object *rb;
  154. intl_error_reset( NULL );
  155. RESOURCEBUNDLE_METHOD_FETCH_OBJECT;
  156. if(Z_TYPE_P(offset) == IS_LONG) {
  157. is_numeric = 1;
  158. meindex = (int32_t)Z_LVAL_P(offset);
  159. rb->child = ures_getByIndex( rb->me, meindex, rb->child, &INTL_DATA_ERROR_CODE(rb) );
  160. } else if(Z_TYPE_P(offset) == IS_STRING) {
  161. mekey = Z_STRVAL_P(offset);
  162. rb->child = ures_getByKey(rb->me, mekey, rb->child, &INTL_DATA_ERROR_CODE(rb) );
  163. } else {
  164. intl_errors_set(INTL_DATA_ERROR_P(rb), U_ILLEGAL_ARGUMENT_ERROR,
  165. "resourcebundle_get: index should be integer or string", 0);
  166. RETURN_NULL();
  167. }
  168. intl_error_set_code( NULL, INTL_DATA_ERROR_CODE(rb) );
  169. if (U_FAILURE(INTL_DATA_ERROR_CODE(rb))) {
  170. if (is_numeric) {
  171. spprintf( &pbuf, 0, "Cannot load resource element %d", meindex );
  172. } else {
  173. spprintf( &pbuf, 0, "Cannot load resource element '%s'", mekey );
  174. }
  175. intl_errors_set_custom_msg( INTL_DATA_ERROR_P(rb), pbuf, 1 );
  176. efree(pbuf);
  177. RETURN_NULL();
  178. }
  179. if (!fallback && (INTL_DATA_ERROR_CODE(rb) == U_USING_FALLBACK_WARNING || INTL_DATA_ERROR_CODE(rb) == U_USING_DEFAULT_WARNING)) {
  180. UErrorCode icuerror;
  181. const char * locale = ures_getLocaleByType( rb->me, ULOC_ACTUAL_LOCALE, &icuerror );
  182. if (is_numeric) {
  183. spprintf( &pbuf, 0, "Cannot load element %d without fallback from to %s", meindex, locale );
  184. } else {
  185. spprintf( &pbuf, 0, "Cannot load element '%s' without fallback from to %s", mekey, locale );
  186. }
  187. intl_errors_set_custom_msg( INTL_DATA_ERROR_P(rb), pbuf, 1 );
  188. efree(pbuf);
  189. RETURN_NULL();
  190. }
  191. resourcebundle_extract_value( return_value, rb );
  192. }
  193. /* }}} */
  194. /* {{{ resourcebundle_array_get */
  195. zval *resourcebundle_array_get(zval *object, zval *offset, int type, zval *rv)
  196. {
  197. if(offset == NULL) {
  198. php_error( E_ERROR, "Cannot apply [] to ResourceBundle object" );
  199. }
  200. ZVAL_NULL(rv);
  201. resourcebundle_array_fetch(object, offset, rv, 1);
  202. return rv;
  203. }
  204. /* }}} */
  205. /* {{{ arginfo_resourcebundle_get */
  206. ZEND_BEGIN_ARG_INFO_EX( arginfo_resourcebundle_get, 0, 0, 1 )
  207. ZEND_ARG_INFO( 0, index )
  208. ZEND_ARG_INFO( 0, fallback )
  209. ZEND_END_ARG_INFO()
  210. /* }}} */
  211. /* {{{ proto mixed ResourceBundle::get( int|string $resindex [, bool $fallback = true ] )
  212. * proto mixed resourcebundle_get( ResourceBundle $rb, int|string $resindex [, bool $fallback = true ] )
  213. * Get resource identified by numerical index or key name.
  214. */
  215. PHP_FUNCTION( resourcebundle_get )
  216. {
  217. zend_bool fallback = 1;
  218. zval * offset;
  219. zval * object;
  220. if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(), "Oz|b", &object, ResourceBundle_ce_ptr, &offset, &fallback ) == FAILURE) {
  221. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  222. "resourcebundle_get: unable to parse input params", 0);
  223. RETURN_FALSE;
  224. }
  225. resourcebundle_array_fetch(object, offset, return_value, fallback);
  226. }
  227. /* }}} */
  228. /* {{{ resourcebundle_array_count */
  229. int resourcebundle_array_count(zval *object, zend_long *count)
  230. {
  231. ResourceBundle_object *rb;
  232. RESOURCEBUNDLE_METHOD_FETCH_OBJECT_NO_CHECK;
  233. if (rb->me == NULL) {
  234. intl_errors_set(&rb->error, U_ILLEGAL_ARGUMENT_ERROR,
  235. "Found unconstructed ResourceBundle", 0);
  236. return 0;
  237. }
  238. *count = ures_getSize( rb->me );
  239. return SUCCESS;
  240. }
  241. /* }}} */
  242. /* {{{ arginfo_resourcebundle_count */
  243. ZEND_BEGIN_ARG_INFO_EX( arginfo_resourcebundle_count, 0, 0, 0 )
  244. ZEND_END_ARG_INFO()
  245. /* }}} */
  246. /* {{{ proto int ResourceBundle::count()
  247. * proto int resourcebundle_count( ResourceBundle $bundle )
  248. * Get resources count
  249. */
  250. PHP_FUNCTION( resourcebundle_count )
  251. {
  252. int32_t len;
  253. RESOURCEBUNDLE_METHOD_INIT_VARS;
  254. if( zend_parse_method_parameters( ZEND_NUM_ARGS(), getThis(), "O", &object, ResourceBundle_ce_ptr ) == FAILURE ) {
  255. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  256. "resourcebundle_count: unable to parse input params", 0);
  257. RETURN_FALSE;
  258. }
  259. RESOURCEBUNDLE_METHOD_FETCH_OBJECT;
  260. len = ures_getSize( rb->me );
  261. RETURN_LONG( len );
  262. }
  263. /* {{{ arginfo_resourcebundle_getlocales */
  264. ZEND_BEGIN_ARG_INFO_EX( arginfo_resourcebundle_getlocales, 0, 0, 1 )
  265. ZEND_ARG_INFO( 0, bundlename )
  266. ZEND_END_ARG_INFO()
  267. /* }}} */
  268. /* {{{ proto array ResourceBundle::getLocales( string $bundlename )
  269. * proto array resourcebundle_locales( string $bundlename )
  270. * Get available locales from ResourceBundle name
  271. */
  272. PHP_FUNCTION( resourcebundle_locales )
  273. {
  274. char * bundlename;
  275. size_t bundlename_len = 0;
  276. const char * entry;
  277. int entry_len;
  278. UEnumeration *icuenum;
  279. UErrorCode icuerror = U_ZERO_ERROR;
  280. intl_errors_reset( NULL );
  281. if( zend_parse_parameters(ZEND_NUM_ARGS(), "s", &bundlename, &bundlename_len ) == FAILURE )
  282. {
  283. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  284. "resourcebundle_locales: unable to parse input params", 0);
  285. RETURN_FALSE;
  286. }
  287. if (bundlename_len >= MAXPATHLEN) {
  288. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR, "resourcebundle_locales: bundle name too long", 0 );
  289. RETURN_FALSE;
  290. }
  291. if(bundlename_len == 0) {
  292. // fetch default locales list
  293. bundlename = NULL;
  294. }
  295. icuenum = ures_openAvailableLocales( bundlename, &icuerror );
  296. INTL_CHECK_STATUS(icuerror, "Cannot fetch locales list");
  297. uenum_reset( icuenum, &icuerror );
  298. INTL_CHECK_STATUS(icuerror, "Cannot iterate locales list");
  299. array_init( return_value );
  300. while ((entry = uenum_next( icuenum, &entry_len, &icuerror ))) {
  301. add_next_index_stringl( return_value, (char *) entry, entry_len);
  302. }
  303. uenum_close( icuenum );
  304. }
  305. /* }}} */
  306. /* {{{ arginfo_resourcebundle_get_error_code */
  307. ZEND_BEGIN_ARG_INFO_EX( arginfo_resourcebundle_get_error_code, 0, 0, 0 )
  308. ZEND_END_ARG_INFO()
  309. /* }}} */
  310. /* {{{ proto string ResourceBundle::getErrorCode( )
  311. * proto string resourcebundle_get_error_code( ResourceBundle $bundle )
  312. * Get text description for ResourceBundle's last error code.
  313. */
  314. PHP_FUNCTION( resourcebundle_get_error_code )
  315. {
  316. RESOURCEBUNDLE_METHOD_INIT_VARS;
  317. if( zend_parse_method_parameters( ZEND_NUM_ARGS(), getThis(), "O",
  318. &object, ResourceBundle_ce_ptr ) == FAILURE )
  319. {
  320. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  321. "resourcebundle_get_error_code: unable to parse input params", 0 );
  322. RETURN_FALSE;
  323. }
  324. rb = Z_INTL_RESOURCEBUNDLE_P( object );
  325. RETURN_LONG(INTL_DATA_ERROR_CODE(rb));
  326. }
  327. /* }}} */
  328. /* {{{ arginfo_resourcebundle_get_error_message */
  329. ZEND_BEGIN_ARG_INFO_EX( arginfo_resourcebundle_get_error_message, 0, 0, 0 )
  330. ZEND_END_ARG_INFO()
  331. /* }}} */
  332. /* {{{ proto string ResourceBundle::getErrorMessage( )
  333. * proto string resourcebundle_get_error_message( ResourceBundle $bundle )
  334. * Get text description for ResourceBundle's last error.
  335. */
  336. PHP_FUNCTION( resourcebundle_get_error_message )
  337. {
  338. zend_string* message = NULL;
  339. RESOURCEBUNDLE_METHOD_INIT_VARS;
  340. if( zend_parse_method_parameters( ZEND_NUM_ARGS(), getThis(), "O",
  341. &object, ResourceBundle_ce_ptr ) == FAILURE )
  342. {
  343. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  344. "resourcebundle_get_error_message: unable to parse input params", 0 );
  345. RETURN_FALSE;
  346. }
  347. rb = Z_INTL_RESOURCEBUNDLE_P( object );
  348. message = intl_error_get_message(INTL_DATA_ERROR_P(rb));
  349. RETURN_STR(message);
  350. }
  351. /* }}} */
  352. /* {{{ ResourceBundle_class_functions
  353. * Every 'ResourceBundle' class method has an entry in this table
  354. */
  355. static const zend_function_entry ResourceBundle_class_functions[] = {
  356. PHP_ME( ResourceBundle, __construct, arginfo_resourcebundle___construct, ZEND_ACC_PUBLIC|ZEND_ACC_CTOR )
  357. ZEND_NAMED_ME( create, ZEND_FN( resourcebundle_create ), arginfo_resourcebundle___construct, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC )
  358. ZEND_NAMED_ME( get, ZEND_FN(resourcebundle_get), arginfo_resourcebundle_get, ZEND_ACC_PUBLIC )
  359. ZEND_NAMED_ME( count, ZEND_FN(resourcebundle_count), arginfo_resourcebundle_count, ZEND_ACC_PUBLIC )
  360. ZEND_NAMED_ME( getLocales, ZEND_FN(resourcebundle_locales), arginfo_resourcebundle_getlocales, ZEND_ACC_PUBLIC | ZEND_ACC_STATIC )
  361. ZEND_NAMED_ME( getErrorCode, ZEND_FN(resourcebundle_get_error_code), arginfo_resourcebundle_get_error_code, ZEND_ACC_PUBLIC )
  362. ZEND_NAMED_ME( getErrorMessage, ZEND_FN(resourcebundle_get_error_message), arginfo_resourcebundle_get_error_message, ZEND_ACC_PUBLIC )
  363. PHP_FE_END
  364. };
  365. /* }}} */
  366. /* {{{ resourcebundle_register_class
  367. * Initialize 'ResourceBundle' class
  368. */
  369. void resourcebundle_register_class( void )
  370. {
  371. zend_class_entry ce;
  372. INIT_CLASS_ENTRY( ce, "ResourceBundle", ResourceBundle_class_functions );
  373. ce.create_object = ResourceBundle_object_create;
  374. ce.get_iterator = resourcebundle_get_iterator;
  375. ResourceBundle_ce_ptr = zend_register_internal_class( &ce );
  376. ResourceBundle_object_handlers = std_object_handlers;
  377. ResourceBundle_object_handlers.offset = XtOffsetOf(ResourceBundle_object, zend);
  378. ResourceBundle_object_handlers.clone_obj = NULL; /* ICU ResourceBundle has no clone implementation */
  379. ResourceBundle_object_handlers.free_obj = ResourceBundle_object_free;
  380. ResourceBundle_object_handlers.read_dimension = resourcebundle_array_get;
  381. ResourceBundle_object_handlers.count_elements = resourcebundle_array_count;
  382. zend_class_implements(ResourceBundle_ce_ptr, 1, zend_ce_traversable);
  383. }
  384. /* }}} */
  385. /*
  386. * Local variables:
  387. * tab-width: 4
  388. * c-basic-offset: 4
  389. * End:
  390. * vim600: noet sw=4 ts=4 fdm=marker
  391. * vim<600: noet sw=4 ts=4
  392. */