enchant.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.0 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available at through the world-wide-web at |
  10. | http://www.php.net/license/3_0.txt. |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Author: Pierre-Alain Joye <paj@pearfr.org> |
  16. | Ilia Alshanetsky <ilia@prohost.org> |
  17. +----------------------------------------------------------------------+
  18. $Id$
  19. */
  20. #ifdef HAVE_CONFIG_H
  21. #include "config.h"
  22. #endif
  23. #include "php.h"
  24. #include "php_ini.h"
  25. #include "ext/standard/info.h"
  26. #include <enchant.h>
  27. #include "php_enchant.h"
  28. typedef EnchantBroker * EnchantBrokerPtr;
  29. typedef struct _broker_struct enchant_broker;
  30. typedef struct _dict_struct enchant_dict;
  31. typedef enchant_broker * enchant_brokerPtr;
  32. typedef enchant_dict * enchant_dictPtr;
  33. typedef struct _broker_struct {
  34. EnchantBroker *pbroker;
  35. enchant_dict **dict;
  36. unsigned int dictcnt;
  37. long rsrc_id;
  38. } _enchant_broker;
  39. typedef struct _dict_struct {
  40. unsigned int id;
  41. EnchantDict *pdict;
  42. enchant_broker *pbroker;
  43. long rsrc_id;
  44. enchant_dict *next;
  45. enchant_dict *prev;
  46. } _enchant_dict;
  47. /* True global resources - no need for thread safety here */
  48. static int le_enchant_broker;
  49. static int le_enchant_dict;
  50. /* If you declare any globals in php_enchant.h uncomment this:*/
  51. /*ZEND_DECLARE_MODULE_GLOBALS(enchant)*/
  52. #define PHP_ENCHANT_MYSPELL 1
  53. #define PHP_ENCHANT_ISPELL 2
  54. /* {{{ arginfo */
  55. ZEND_BEGIN_ARG_INFO(arginfo_enchant_broker_init, 0)
  56. ZEND_END_ARG_INFO()
  57. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_broker_free, 0, 0, 1)
  58. ZEND_ARG_INFO(0, broker)
  59. ZEND_END_ARG_INFO()
  60. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_broker_set_dict_path, 0, 0, 3)
  61. ZEND_ARG_INFO(0, broker)
  62. ZEND_ARG_INFO(0, name)
  63. ZEND_ARG_INFO(0, value)
  64. ZEND_END_ARG_INFO()
  65. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_broker_get_dict_path, 0, 0, 2)
  66. ZEND_ARG_INFO(0, broker)
  67. ZEND_ARG_INFO(0, name)
  68. ZEND_END_ARG_INFO()
  69. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_broker_request_dict, 0, 0, 2)
  70. ZEND_ARG_INFO(0, broker)
  71. ZEND_ARG_INFO(0, tag)
  72. ZEND_END_ARG_INFO()
  73. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_broker_request_pwl_dict, 0, 0, 2)
  74. ZEND_ARG_INFO(0, broker)
  75. ZEND_ARG_INFO(0, filename)
  76. ZEND_END_ARG_INFO()
  77. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_broker_free_dict, 0, 0, 1)
  78. ZEND_ARG_INFO(0, dict)
  79. ZEND_END_ARG_INFO()
  80. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_broker_set_ordering, 0, 0, 3)
  81. ZEND_ARG_INFO(0, broker)
  82. ZEND_ARG_INFO(0, tag)
  83. ZEND_ARG_INFO(0, ordering)
  84. ZEND_END_ARG_INFO()
  85. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_dict_quick_check, 0, 0, 2)
  86. ZEND_ARG_INFO(0, dict)
  87. ZEND_ARG_INFO(0, word)
  88. ZEND_ARG_INFO(1, suggestions)
  89. ZEND_END_ARG_INFO()
  90. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_dict_check, 0, 0, 2)
  91. ZEND_ARG_INFO(0, dict)
  92. ZEND_ARG_INFO(0, word)
  93. ZEND_END_ARG_INFO()
  94. ZEND_BEGIN_ARG_INFO_EX(arginfo_enchant_dict_store_replacement, 0, 0, 3)
  95. ZEND_ARG_INFO(0, dict)
  96. ZEND_ARG_INFO(0, mis)
  97. ZEND_ARG_INFO(0, cor)
  98. ZEND_END_ARG_INFO()
  99. /* }}} */
  100. /* {{{ enchant_functions[]
  101. *
  102. * Every user visible function must have an entry in enchant_functions[].
  103. */
  104. zend_function_entry enchant_functions[] = {
  105. PHP_FE(enchant_broker_init, arginfo_enchant_broker_init)
  106. PHP_FE(enchant_broker_free, arginfo_enchant_broker_free)
  107. PHP_FE(enchant_broker_get_error, arginfo_enchant_broker_free)
  108. PHP_FE(enchant_broker_set_dict_path, arginfo_enchant_broker_set_dict_path)
  109. PHP_FE(enchant_broker_get_dict_path, arginfo_enchant_broker_get_dict_path)
  110. PHP_FE(enchant_broker_list_dicts, arginfo_enchant_broker_free)
  111. PHP_FE(enchant_broker_request_dict, arginfo_enchant_broker_request_dict)
  112. PHP_FE(enchant_broker_request_pwl_dict, arginfo_enchant_broker_request_pwl_dict)
  113. PHP_FE(enchant_broker_free_dict, arginfo_enchant_broker_free_dict)
  114. PHP_FE(enchant_broker_dict_exists, arginfo_enchant_broker_request_dict)
  115. PHP_FE(enchant_broker_set_ordering, arginfo_enchant_broker_set_ordering)
  116. PHP_FE(enchant_broker_describe, arginfo_enchant_broker_free)
  117. PHP_FE(enchant_dict_check, arginfo_enchant_dict_check)
  118. PHP_FE(enchant_dict_suggest, arginfo_enchant_dict_check)
  119. PHP_FE(enchant_dict_add_to_personal, arginfo_enchant_dict_check)
  120. PHP_FE(enchant_dict_add_to_session, arginfo_enchant_dict_check)
  121. PHP_FE(enchant_dict_is_in_session, arginfo_enchant_dict_check)
  122. PHP_FE(enchant_dict_store_replacement, arginfo_enchant_dict_store_replacement)
  123. PHP_FE(enchant_dict_get_error, arginfo_enchant_broker_free_dict)
  124. PHP_FE(enchant_dict_describe, arginfo_enchant_broker_free_dict)
  125. PHP_FE(enchant_dict_quick_check, arginfo_enchant_dict_quick_check)
  126. PHP_FE_END
  127. };
  128. /* }}} */
  129. /* {{{ enchant_module_entry
  130. */
  131. zend_module_entry enchant_module_entry = {
  132. #if ZEND_MODULE_API_NO >= 20010901
  133. STANDARD_MODULE_HEADER,
  134. #endif
  135. "enchant",
  136. enchant_functions,
  137. PHP_MINIT(enchant),
  138. PHP_MSHUTDOWN(enchant),
  139. NULL, /* Replace with NULL if there's nothing to do at request start */
  140. NULL, /* Replace with NULL if there's nothing to do at request end */
  141. PHP_MINFO(enchant),
  142. #if ZEND_MODULE_API_NO >= 20010901
  143. PHP_ENCHANT_VERSION,
  144. #endif
  145. STANDARD_MODULE_PROPERTIES
  146. };
  147. /* }}} */
  148. #ifdef COMPILE_DL_ENCHANT
  149. ZEND_GET_MODULE(enchant)
  150. #endif
  151. static void
  152. enumerate_providers_fn (const char * const name,
  153. const char * const desc,
  154. const char * const file,
  155. void * ud) /* {{{ */
  156. {
  157. zval *zdesc = (zval *) ud;
  158. zval *tmp_array;
  159. MAKE_STD_ZVAL(tmp_array);
  160. array_init(tmp_array);
  161. add_assoc_string(tmp_array, "name", (char *)name, 1);
  162. add_assoc_string(tmp_array, "desc", (char *)desc, 1);
  163. add_assoc_string(tmp_array, "file", (char *)file, 1);
  164. if (Z_TYPE_P(zdesc)!=IS_ARRAY) {
  165. array_init(zdesc);
  166. }
  167. add_next_index_zval(zdesc, tmp_array);
  168. }
  169. /* }}} */
  170. static void
  171. describe_dict_fn (const char * const lang,
  172. const char * const name,
  173. const char * const desc,
  174. const char * const file,
  175. void * ud) /* {{{ */
  176. {
  177. zval *zdesc = (zval *) ud;
  178. array_init(zdesc);
  179. add_assoc_string(zdesc, "lang", (char *)lang, 1);
  180. add_assoc_string(zdesc, "name", (char *)name, 1);
  181. add_assoc_string(zdesc, "desc", (char *)desc, 1);
  182. add_assoc_string(zdesc, "file", (char *)file, 1);
  183. }
  184. /* }}} */
  185. static void php_enchant_list_dicts_fn( const char * const lang_tag,
  186. const char * const provider_name, const char * const provider_desc,
  187. const char * const provider_file, void * ud) /* {{{ */
  188. {
  189. zval *zdesc = (zval *) ud;
  190. zval *tmp_array;
  191. MAKE_STD_ZVAL(tmp_array);
  192. array_init(tmp_array);
  193. add_assoc_string(tmp_array, "lang_tag", (char *)lang_tag, 1);
  194. add_assoc_string(tmp_array, "provider_name", (char *)provider_name, 1);
  195. add_assoc_string(tmp_array, "provider_desc", (char *)provider_desc, 1);
  196. add_assoc_string(tmp_array, "provider_file", (char *)provider_file, 1);
  197. if (Z_TYPE_P(zdesc) != IS_ARRAY) {
  198. array_init(zdesc);
  199. }
  200. add_next_index_zval(zdesc, tmp_array);
  201. }
  202. /* }}} */
  203. static void php_enchant_broker_free(zend_rsrc_list_entry *rsrc TSRMLS_DC) /* {{{ */
  204. {
  205. if (rsrc->ptr) {
  206. enchant_broker *broker = (enchant_broker *)rsrc->ptr;
  207. if (broker) {
  208. if (broker->pbroker) {
  209. if (broker->dictcnt && broker->dict) {
  210. if (broker->dict) {
  211. int total;
  212. total = broker->dictcnt-1;
  213. do {
  214. zend_list_delete(broker->dict[total]->rsrc_id);
  215. efree(broker->dict[total]);
  216. total--;
  217. } while (total>=0);
  218. }
  219. efree(broker->dict);
  220. broker->dict = NULL;
  221. }
  222. enchant_broker_free(broker->pbroker);
  223. }
  224. efree(broker);
  225. }
  226. }
  227. }
  228. /* }}} */
  229. static void php_enchant_dict_free(zend_rsrc_list_entry *rsrc TSRMLS_DC) /* {{{ */
  230. {
  231. if (rsrc->ptr) {
  232. enchant_dict *pdict = (enchant_dict *)rsrc->ptr;
  233. if (pdict) {
  234. if (pdict->pdict && pdict->pbroker) {
  235. enchant_broker_free_dict(pdict->pbroker->pbroker, pdict->pdict);
  236. if (pdict->id) {
  237. pdict->pbroker->dict[pdict->id-1]->next = NULL;
  238. }
  239. zend_list_delete(pdict->pbroker->rsrc_id);
  240. }
  241. }
  242. }
  243. }
  244. /* }}} */
  245. /* {{{ PHP_MINIT_FUNCTION
  246. */
  247. PHP_MINIT_FUNCTION(enchant)
  248. {
  249. le_enchant_broker = zend_register_list_destructors_ex(php_enchant_broker_free, NULL, "enchant_broker", module_number);
  250. le_enchant_dict = zend_register_list_destructors_ex(php_enchant_dict_free, NULL, "enchant_dict", module_number);
  251. REGISTER_LONG_CONSTANT("ENCHANT_MYSPELL", PHP_ENCHANT_MYSPELL, CONST_CS | CONST_PERSISTENT);
  252. REGISTER_LONG_CONSTANT("ENCHANT_ISPELL", PHP_ENCHANT_ISPELL, CONST_CS | CONST_PERSISTENT);
  253. return SUCCESS;
  254. }
  255. /* }}} */
  256. /* {{{ PHP_MSHUTDOWN_FUNCTION
  257. */
  258. PHP_MSHUTDOWN_FUNCTION(enchant)
  259. {
  260. return SUCCESS;
  261. }
  262. /* }}} */
  263. static void __enumerate_providers_fn (const char * const name,
  264. const char * const desc,
  265. const char * const file,
  266. void * ud) /* {{{ */
  267. {
  268. php_info_print_table_row(3, name, desc, file);
  269. }
  270. /* }}} */
  271. /* {{{ PHP_MINFO_FUNCTION
  272. */
  273. PHP_MINFO_FUNCTION(enchant)
  274. {
  275. EnchantBroker *pbroker;
  276. pbroker = enchant_broker_init();
  277. php_info_print_table_start();
  278. php_info_print_table_header(2, "enchant support", "enabled");
  279. php_info_print_table_row(2, "Version", PHP_ENCHANT_VERSION);
  280. #ifdef ENCHANT_VERSION_STRING
  281. php_info_print_table_row(2, "Libenchant Version", ENCHANT_VERSION_STRING);
  282. #elif defined(HAVE_ENCHANT_BROKER_SET_PARAM)
  283. php_info_print_table_row(2, "Libenchant Version", "1.5.0 or later");
  284. #endif
  285. php_info_print_table_row(2, "Revision", "$Id$");
  286. php_info_print_table_end();
  287. php_info_print_table_start();
  288. enchant_broker_describe(pbroker, __enumerate_providers_fn, NULL);
  289. php_info_print_table_end();
  290. enchant_broker_free(pbroker);
  291. }
  292. /* }}} */
  293. #define PHP_ENCHANT_GET_BROKER \
  294. ZEND_FETCH_RESOURCE(pbroker, enchant_broker *, &broker, -1, "enchant_broker", le_enchant_broker); \
  295. if (!pbroker || !pbroker->pbroker) { \
  296. php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", "Resource broker invalid"); \
  297. RETURN_FALSE; \
  298. }
  299. #define PHP_ENCHANT_GET_DICT \
  300. ZEND_FETCH_RESOURCE(pdict, enchant_dict *, &dict, -1, "enchant_dict", le_enchant_dict); \
  301. if (!pdict || !pdict->pdict) { \
  302. php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", "Invalid dictionary resource."); \
  303. RETURN_FALSE; \
  304. }
  305. /* {{{ proto resource enchant_broker_init()
  306. create a new broker object capable of requesting */
  307. PHP_FUNCTION(enchant_broker_init)
  308. {
  309. enchant_broker *broker;
  310. EnchantBroker *pbroker;
  311. if (zend_parse_parameters_none() == FAILURE) {
  312. return;
  313. }
  314. pbroker = enchant_broker_init();
  315. if (pbroker) {
  316. broker = (enchant_broker *) emalloc(sizeof(enchant_broker));
  317. broker->pbroker = pbroker;
  318. broker->dict = NULL;
  319. broker->dictcnt = 0;
  320. broker->rsrc_id = ZEND_REGISTER_RESOURCE(return_value, broker, le_enchant_broker);
  321. } else {
  322. RETURN_FALSE;
  323. }
  324. }
  325. /* }}} */
  326. /* {{{ proto boolean enchant_broker_free(resource broker)
  327. Destroys the broker object and its dictionnaries */
  328. PHP_FUNCTION(enchant_broker_free)
  329. {
  330. zval *broker;
  331. enchant_broker *pbroker;
  332. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &broker) == FAILURE) {
  333. RETURN_FALSE;
  334. }
  335. PHP_ENCHANT_GET_BROKER;
  336. zend_list_delete(Z_RESVAL_P(broker));
  337. RETURN_TRUE;
  338. }
  339. /* }}} */
  340. /* {{{ proto string enchant_broker_get_error(resource broker)
  341. Returns the last error of the broker */
  342. PHP_FUNCTION(enchant_broker_get_error)
  343. {
  344. zval *broker;
  345. enchant_broker *pbroker;
  346. char *msg;
  347. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &broker) == FAILURE) {
  348. RETURN_FALSE;
  349. }
  350. PHP_ENCHANT_GET_BROKER;
  351. msg = enchant_broker_get_error(pbroker->pbroker);
  352. if (msg) {
  353. RETURN_STRING((char *)msg, 1);
  354. }
  355. RETURN_FALSE;
  356. }
  357. /* }}} */
  358. #if HAVE_ENCHANT_BROKER_SET_PARAM
  359. /* {{{ proto bool enchant_broker_set_dict_path(resource broker, int dict_type, string value)
  360. Set the directory path for a given backend, works with ispell and myspell */
  361. PHP_FUNCTION(enchant_broker_set_dict_path)
  362. {
  363. zval *broker;
  364. enchant_broker *pbroker;
  365. long dict_type;
  366. char *value;
  367. int value_len;
  368. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rls", &broker, &dict_type, &value, &value_len) == FAILURE) {
  369. RETURN_FALSE;
  370. }
  371. if (!value_len) {
  372. RETURN_FALSE;
  373. }
  374. PHP_ENCHANT_GET_BROKER;
  375. switch (dict_type) {
  376. case PHP_ENCHANT_MYSPELL:
  377. PHP_ENCHANT_GET_BROKER;
  378. enchant_broker_set_param(pbroker->pbroker, "enchant.myspell.dictionary.path", (const char *)value);
  379. RETURN_TRUE;
  380. break;
  381. case PHP_ENCHANT_ISPELL:
  382. PHP_ENCHANT_GET_BROKER;
  383. enchant_broker_set_param(pbroker->pbroker, "enchant.ispell.dictionary.path", (const char *)value);
  384. RETURN_TRUE;
  385. break;
  386. default:
  387. RETURN_FALSE;
  388. }
  389. }
  390. /* }}} */
  391. /* {{{ proto string enchant_broker_get_dict_path(resource broker, int dict_type)
  392. Get the directory path for a given backend, works with ispell and myspell */
  393. PHP_FUNCTION(enchant_broker_get_dict_path)
  394. {
  395. zval *broker;
  396. enchant_broker *pbroker;
  397. long dict_type;
  398. char *value;
  399. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rl", &broker, &dict_type) == FAILURE) {
  400. RETURN_FALSE;
  401. }
  402. PHP_ENCHANT_GET_BROKER;
  403. switch (dict_type) {
  404. case PHP_ENCHANT_MYSPELL:
  405. PHP_ENCHANT_GET_BROKER;
  406. value = enchant_broker_get_param(pbroker->pbroker, "enchant.myspell.dictionary.path");
  407. break;
  408. case PHP_ENCHANT_ISPELL:
  409. PHP_ENCHANT_GET_BROKER;
  410. value = enchant_broker_get_param(pbroker->pbroker, "enchant.ispell.dictionary.path");
  411. break;
  412. default:
  413. RETURN_FALSE;
  414. }
  415. RETURN_STRING(value, 1);
  416. }
  417. /* }}} */
  418. #else
  419. /* {{{ proto bool enchant_broker_set_dict_path(resource broker, int dict_type, string value)
  420. Set the directory path for a given backend, works with ispell and myspell */
  421. PHP_FUNCTION(enchant_broker_set_dict_path)
  422. {
  423. RETURN_FALSE;
  424. }
  425. /* }}} */
  426. /* {{{ proto string enchant_broker_get_dict_path(resource broker, int dict_type)
  427. Get the directory path for a given backend, works with ispell and myspell */
  428. PHP_FUNCTION(enchant_broker_get_dict_path)
  429. {
  430. RETURN_FALSE;
  431. }
  432. /* }}} */
  433. #endif
  434. /* {{{ proto string enchant_broker_list_dicts(resource broker)
  435. Lists the dictionaries available for the given broker */
  436. PHP_FUNCTION(enchant_broker_list_dicts)
  437. {
  438. zval *broker;
  439. enchant_broker *pbroker;
  440. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &broker) == FAILURE) {
  441. RETURN_FALSE;
  442. }
  443. PHP_ENCHANT_GET_BROKER;
  444. enchant_broker_list_dicts(pbroker->pbroker, php_enchant_list_dicts_fn, (void *)return_value);
  445. }
  446. /* }}} */
  447. /* {{{ proto resource enchant_broker_request_dict(resource broker, string tag)
  448. create a new dictionary using tag, the non-empty language tag you wish to request
  449. a dictionary for ("en_US", "de_DE", ...) */
  450. PHP_FUNCTION(enchant_broker_request_dict)
  451. {
  452. zval *broker;
  453. enchant_broker *pbroker;
  454. enchant_dict *dict;
  455. EnchantDict *d;
  456. char *tag;
  457. int taglen;
  458. int pos;
  459. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs", &broker, &tag, &taglen) == FAILURE) {
  460. RETURN_FALSE;
  461. }
  462. PHP_ENCHANT_GET_BROKER;
  463. if (taglen == 0) {
  464. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Tag cannot be empty");
  465. RETURN_FALSE;
  466. }
  467. d = enchant_broker_request_dict(pbroker->pbroker, (const char *)tag);
  468. if (d) {
  469. pos = pbroker->dictcnt++;
  470. if (pbroker->dictcnt) {
  471. pbroker->dict = (enchant_dict **)erealloc(pbroker->dict, sizeof(enchant_dict *) * pbroker->dictcnt);
  472. } else {
  473. pbroker->dict = (enchant_dict **)emalloc(sizeof(enchant_dict *));
  474. pos = 0;
  475. }
  476. dict = pbroker->dict[pos] = (enchant_dict *)emalloc(sizeof(enchant_dict));
  477. dict->id = pos;
  478. dict->pbroker = pbroker;
  479. dict->pdict = d;
  480. dict->prev = pos ? pbroker->dict[pos-1] : NULL;
  481. dict->next = NULL;
  482. pbroker->dict[pos] = dict;
  483. if (pos) {
  484. pbroker->dict[pos-1]->next = dict;
  485. }
  486. dict->rsrc_id = ZEND_REGISTER_RESOURCE(return_value, dict, le_enchant_dict);
  487. zend_list_addref(pbroker->rsrc_id);
  488. } else {
  489. RETURN_FALSE;
  490. }
  491. }
  492. /* }}} */
  493. /* {{{ proto resource enchant_broker_request_pwl_dict(resource broker, string filename)
  494. creates a dictionary using a PWL file. A PWL file is personal word file one word per line. It must exist before the call.*/
  495. PHP_FUNCTION(enchant_broker_request_pwl_dict)
  496. {
  497. zval *broker;
  498. enchant_broker *pbroker;
  499. enchant_dict *dict;
  500. EnchantDict *d;
  501. char *pwl;
  502. int pwllen;
  503. int pos;
  504. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rp", &broker, &pwl, &pwllen) == FAILURE) {
  505. RETURN_FALSE;
  506. }
  507. #if PHP_API_VERSION < 20100412
  508. if ((PG(safe_mode) && (!php_checkuid(pwl, NULL, CHECKUID_CHECK_FILE_AND_DIR))) || php_check_open_basedir(pwl TSRMLS_CC)) {
  509. #else
  510. if (php_check_open_basedir(pwl TSRMLS_CC)) {
  511. #endif
  512. RETURN_FALSE;
  513. }
  514. PHP_ENCHANT_GET_BROKER;
  515. d = enchant_broker_request_pwl_dict(pbroker->pbroker, (const char *)pwl);
  516. if (d) {
  517. pos = pbroker->dictcnt++;
  518. if (pbroker->dictcnt) {
  519. pbroker->dict = (enchant_dict **)erealloc(pbroker->dict, sizeof(enchant_dict *) * pbroker->dictcnt);
  520. } else {
  521. pbroker->dict = (enchant_dict **)emalloc(sizeof(enchant_dict *));
  522. pos = 0;
  523. }
  524. dict = pbroker->dict[pos] = (enchant_dict *)emalloc(sizeof(enchant_dict));
  525. dict->id = pos;
  526. dict->pbroker = pbroker;
  527. dict->pdict = d;
  528. dict->prev = pos?pbroker->dict[pos-1]:NULL;
  529. dict->next = NULL;
  530. pbroker->dict[pos] = dict;
  531. if (pos) {
  532. pbroker->dict[pos-1]->next = dict;
  533. }
  534. dict->rsrc_id = ZEND_REGISTER_RESOURCE(return_value, dict, le_enchant_dict);
  535. } else {
  536. RETURN_FALSE;
  537. }
  538. }
  539. /* }}} */
  540. /* {{{ proto resource enchant_broker_free_dict(resource dict)
  541. Free the dictionary resource */
  542. PHP_FUNCTION(enchant_broker_free_dict)
  543. {
  544. zval *dict;
  545. enchant_dict *pdict;
  546. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &dict) == FAILURE) {
  547. RETURN_FALSE;
  548. }
  549. PHP_ENCHANT_GET_DICT;
  550. zend_list_delete(Z_RESVAL_P(dict));
  551. RETURN_TRUE;
  552. }
  553. /* }}} */
  554. /* {{{ proto bool enchant_broker_dict_exists(resource broker, string tag)
  555. Whether a dictionary exists or not. Using non-empty tag */
  556. PHP_FUNCTION(enchant_broker_dict_exists)
  557. {
  558. zval *broker;
  559. char *tag;
  560. int taglen;
  561. enchant_broker * pbroker;
  562. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs", &broker, &tag, &taglen) == FAILURE) {
  563. RETURN_FALSE;
  564. }
  565. PHP_ENCHANT_GET_BROKER;
  566. RETURN_BOOL(enchant_broker_dict_exists(pbroker->pbroker, tag));
  567. }
  568. /* }}} */
  569. /* {{{ proto bool enchant_broker_set_ordering(resource broker, string tag, string ordering)
  570. Declares a preference of dictionaries to use for the language
  571. described/referred to by 'tag'. The ordering is a comma delimited
  572. list of provider names. As a special exception, the "*" tag can
  573. be used as a language tag to declare a default ordering for any
  574. language that does not explicitly declare an ordering. */
  575. PHP_FUNCTION(enchant_broker_set_ordering)
  576. {
  577. zval *broker;
  578. char *pordering;
  579. int porderinglen;
  580. char *ptag;
  581. int ptaglen;
  582. enchant_broker * pbroker;
  583. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rss", &broker, &ptag, &ptaglen, &pordering, &porderinglen) == FAILURE) {
  584. RETURN_FALSE;
  585. }
  586. PHP_ENCHANT_GET_BROKER;
  587. enchant_broker_set_ordering(pbroker->pbroker, ptag, pordering);
  588. RETURN_TRUE;
  589. }
  590. /* }}} */
  591. /* {{{ proto array enchant_broker_describe(resource broker)
  592. Enumerates the Enchant providers and tells you some rudimentary information about them. The same info is provided through phpinfo() */
  593. PHP_FUNCTION(enchant_broker_describe)
  594. {
  595. EnchantBrokerDescribeFn describetozval = enumerate_providers_fn;
  596. zval *broker;
  597. enchant_broker * pbroker;
  598. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &broker) == FAILURE) {
  599. RETURN_FALSE;
  600. }
  601. PHP_ENCHANT_GET_BROKER;
  602. enchant_broker_describe(pbroker->pbroker, describetozval, (void *)return_value);
  603. }
  604. /* }}} */
  605. /* {{{ proto bool enchant_dict_quick_check(resource dict, string word [, array &suggestions])
  606. If the word is correctly spelled return true, otherwise return false, if suggestions variable
  607. is provided, fill it with spelling alternatives. */
  608. PHP_FUNCTION(enchant_dict_quick_check)
  609. {
  610. zval *dict, *sugg = NULL;
  611. char *word;
  612. int wordlen;
  613. enchant_dict *pdict;
  614. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs|z/", &dict, &word, &wordlen, &sugg) == FAILURE) {
  615. RETURN_FALSE;
  616. }
  617. if (sugg) {
  618. zval_dtor(sugg);
  619. array_init(sugg);
  620. }
  621. PHP_ENCHANT_GET_DICT;
  622. if (enchant_dict_check(pdict->pdict, word, wordlen) > 0) {
  623. int n_sugg;
  624. size_t n_sugg_st;
  625. char **suggs;
  626. if (!sugg && ZEND_NUM_ARGS() == 2) {
  627. RETURN_FALSE;
  628. }
  629. suggs = enchant_dict_suggest(pdict->pdict, word, wordlen, &n_sugg_st);
  630. memcpy(&n_sugg, &n_sugg_st, sizeof(n_sugg));
  631. if (suggs && n_sugg) {
  632. int i;
  633. for (i = 0; i < n_sugg; i++) {
  634. add_next_index_string(sugg, suggs[i], 1);
  635. }
  636. enchant_dict_free_suggestions(pdict->pdict, suggs);
  637. }
  638. RETURN_FALSE;
  639. }
  640. RETURN_TRUE;
  641. }
  642. /* }}} */
  643. /* {{{ proto bool enchant_dict_check(resource dict, string word)
  644. If the word is correctly spelled return true, otherwise return false */
  645. PHP_FUNCTION(enchant_dict_check)
  646. {
  647. zval *dict;
  648. char *word;
  649. int wordlen;
  650. enchant_dict *pdict;
  651. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs", &dict, &word, &wordlen) == FAILURE) {
  652. RETURN_FALSE;
  653. }
  654. PHP_ENCHANT_GET_DICT;
  655. RETURN_BOOL(!enchant_dict_check(pdict->pdict, word, wordlen));
  656. }
  657. /* }}} */
  658. /* {{{ proto array enchant_dict_suggest(resource dict, string word)
  659. Will return a list of values if any of those pre-conditions are not met.*/
  660. PHP_FUNCTION(enchant_dict_suggest)
  661. {
  662. zval *dict;
  663. char *word;
  664. int wordlen;
  665. char **suggs;
  666. enchant_dict *pdict;
  667. int n_sugg;
  668. size_t n_sugg_st;
  669. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs", &dict, &word, &wordlen) == FAILURE) {
  670. RETURN_FALSE;
  671. }
  672. PHP_ENCHANT_GET_DICT;
  673. suggs = enchant_dict_suggest(pdict->pdict, word, wordlen, &n_sugg_st);
  674. memcpy(&n_sugg, &n_sugg_st, sizeof(n_sugg));
  675. if (suggs && n_sugg) {
  676. int i;
  677. array_init(return_value);
  678. for (i = 0; i < n_sugg; i++) {
  679. add_next_index_string(return_value, suggs[i], 1);
  680. }
  681. enchant_dict_free_suggestions(pdict->pdict, suggs);
  682. }
  683. }
  684. /* }}} */
  685. /* {{{ proto void enchant_dict_add_to_personal(resource dict, string word)
  686. add 'word' to personal word list */
  687. PHP_FUNCTION(enchant_dict_add_to_personal)
  688. {
  689. zval *dict;
  690. char *word;
  691. int wordlen;
  692. enchant_dict *pdict;
  693. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs", &dict, &word, &wordlen) == FAILURE) {
  694. RETURN_FALSE;
  695. }
  696. PHP_ENCHANT_GET_DICT;
  697. enchant_dict_add_to_personal(pdict->pdict, word, wordlen);
  698. }
  699. /* }}} */
  700. /* {{{ proto void enchant_dict_add_to_session(resource dict, string word)
  701. add 'word' to this spell-checking session */
  702. PHP_FUNCTION(enchant_dict_add_to_session)
  703. {
  704. zval *dict;
  705. char *word;
  706. int wordlen;
  707. enchant_dict *pdict;
  708. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs", &dict, &word, &wordlen) == FAILURE) {
  709. RETURN_FALSE;
  710. }
  711. PHP_ENCHANT_GET_DICT;
  712. enchant_dict_add_to_session(pdict->pdict, word, wordlen);
  713. }
  714. /* }}} */
  715. /* {{{ proto bool enchant_dict_is_in_session(resource dict, string word)
  716. whether or not 'word' exists in this spelling-session */
  717. PHP_FUNCTION(enchant_dict_is_in_session)
  718. {
  719. zval *dict;
  720. char *word;
  721. int wordlen;
  722. enchant_dict *pdict;
  723. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs", &dict, &word, &wordlen) == FAILURE) {
  724. RETURN_FALSE;
  725. }
  726. PHP_ENCHANT_GET_DICT;
  727. RETURN_BOOL(enchant_dict_is_in_session(pdict->pdict, word, wordlen));
  728. }
  729. /* }}} */
  730. /* {{{ proto void enchant_dict_store_replacement(resource dict, string mis, string cor)
  731. add a correction for 'mis' using 'cor'.
  732. Notes that you replaced @mis with @cor, so it's possibly more likely
  733. that future occurrences of @mis will be replaced with @cor. So it might
  734. bump @cor up in the suggestion list.*/
  735. PHP_FUNCTION(enchant_dict_store_replacement)
  736. {
  737. zval *dict;
  738. char *mis, *cor;
  739. int mislen, corlen;
  740. enchant_dict *pdict;
  741. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rss", &dict, &mis, &mislen, &cor, &corlen) == FAILURE) {
  742. RETURN_FALSE;
  743. }
  744. PHP_ENCHANT_GET_DICT;
  745. enchant_dict_store_replacement(pdict->pdict, mis, mislen, cor, corlen);
  746. }
  747. /* }}} */
  748. /* {{{ proto string enchant_dict_get_error(resource dict)
  749. Returns the last error of the current spelling-session */
  750. PHP_FUNCTION(enchant_dict_get_error)
  751. {
  752. zval *dict;
  753. enchant_dict *pdict;
  754. char *msg;
  755. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &dict) == FAILURE) {
  756. RETURN_FALSE;
  757. }
  758. PHP_ENCHANT_GET_DICT;
  759. msg = enchant_dict_get_error(pdict->pdict);
  760. if (msg) {
  761. RETURN_STRING((char *)msg, 1);
  762. }
  763. RETURN_FALSE;
  764. }
  765. /* }}} */
  766. /* {{{ proto array enchant_dict_describe(resource dict)
  767. Describes an individual dictionary 'dict' */
  768. PHP_FUNCTION(enchant_dict_describe)
  769. {
  770. zval *dict;
  771. enchant_dict *pdict;
  772. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &dict) == FAILURE) {
  773. RETURN_FALSE;
  774. }
  775. PHP_ENCHANT_GET_DICT;
  776. enchant_dict_describe(pdict->pdict, describe_dict_fn, (void *)return_value);
  777. }
  778. /* }}} */
  779. /*
  780. * Local variables:
  781. * tab-width: 4
  782. * c-basic-offset: 4
  783. * End:
  784. * vim600: noet sw=4 ts=4 fdm=marker
  785. * vim<600: noet sw=4 ts=4
  786. */