dba.c 33 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.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. | Authors: Sascha Schumann <sascha@schumann.cx> |
  16. | Marcus Boerger <helly@php.net> |
  17. +----------------------------------------------------------------------+
  18. */
  19. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #if HAVE_DBA
  24. #include "php_ini.h"
  25. #include <stdio.h>
  26. #include <fcntl.h>
  27. #ifdef HAVE_SYS_FILE_H
  28. #include <sys/file.h>
  29. #endif
  30. #include "php_dba.h"
  31. #include "ext/standard/info.h"
  32. #include "ext/standard/php_string.h"
  33. #include "ext/standard/flock_compat.h"
  34. #include "php_gdbm.h"
  35. #include "php_ndbm.h"
  36. #include "php_dbm.h"
  37. #include "php_cdb.h"
  38. #include "php_db1.h"
  39. #include "php_db2.h"
  40. #include "php_db3.h"
  41. #include "php_db4.h"
  42. #include "php_flatfile.h"
  43. #include "php_inifile.h"
  44. #include "php_qdbm.h"
  45. #include "php_tcadb.h"
  46. #include "php_lmdb.h"
  47. /* {{{ arginfo */
  48. ZEND_BEGIN_ARG_INFO_EX(arginfo_dba_popen, 0, 0, 2)
  49. ZEND_ARG_INFO(0, path)
  50. ZEND_ARG_INFO(0, mode)
  51. ZEND_ARG_INFO(0, handlername)
  52. ZEND_ARG_VARIADIC_INFO(0, handler_parameters)
  53. ZEND_END_ARG_INFO()
  54. ZEND_BEGIN_ARG_INFO_EX(arginfo_dba_open, 0, 0, 2)
  55. ZEND_ARG_INFO(0, path)
  56. ZEND_ARG_INFO(0, mode)
  57. ZEND_ARG_INFO(0, handlername)
  58. ZEND_ARG_VARIADIC_INFO(0, handler_parameters)
  59. ZEND_END_ARG_INFO()
  60. ZEND_BEGIN_ARG_INFO(arginfo_dba_close, 0)
  61. ZEND_ARG_INFO(0, handle)
  62. ZEND_END_ARG_INFO()
  63. ZEND_BEGIN_ARG_INFO(arginfo_dba_exists, 0)
  64. ZEND_ARG_INFO(0, key)
  65. ZEND_ARG_INFO(0, handle)
  66. ZEND_END_ARG_INFO()
  67. ZEND_BEGIN_ARG_INFO_EX(arginfo_dba_fetch, 0, 0, 2)
  68. ZEND_ARG_INFO(0, key)
  69. ZEND_ARG_INFO(0, skip)
  70. ZEND_ARG_INFO(0, handle)
  71. ZEND_END_ARG_INFO()
  72. ZEND_BEGIN_ARG_INFO(arginfo_dba_key_split, 0)
  73. ZEND_ARG_INFO(0, key)
  74. ZEND_END_ARG_INFO()
  75. ZEND_BEGIN_ARG_INFO(arginfo_dba_firstkey, 0)
  76. ZEND_ARG_INFO(0, handle)
  77. ZEND_END_ARG_INFO()
  78. ZEND_BEGIN_ARG_INFO(arginfo_dba_nextkey, 0)
  79. ZEND_ARG_INFO(0, handle)
  80. ZEND_END_ARG_INFO()
  81. ZEND_BEGIN_ARG_INFO(arginfo_dba_delete, 0)
  82. ZEND_ARG_INFO(0, key)
  83. ZEND_ARG_INFO(0, handle)
  84. ZEND_END_ARG_INFO()
  85. ZEND_BEGIN_ARG_INFO(arginfo_dba_insert, 0)
  86. ZEND_ARG_INFO(0, key)
  87. ZEND_ARG_INFO(0, value)
  88. ZEND_ARG_INFO(0, handle)
  89. ZEND_END_ARG_INFO()
  90. ZEND_BEGIN_ARG_INFO(arginfo_dba_replace, 0)
  91. ZEND_ARG_INFO(0, key)
  92. ZEND_ARG_INFO(0, value)
  93. ZEND_ARG_INFO(0, handle)
  94. ZEND_END_ARG_INFO()
  95. ZEND_BEGIN_ARG_INFO(arginfo_dba_optimize, 0)
  96. ZEND_ARG_INFO(0, handle)
  97. ZEND_END_ARG_INFO()
  98. ZEND_BEGIN_ARG_INFO(arginfo_dba_sync, 0)
  99. ZEND_ARG_INFO(0, handle)
  100. ZEND_END_ARG_INFO()
  101. ZEND_BEGIN_ARG_INFO_EX(arginfo_dba_handlers, 0, 0, 0)
  102. ZEND_ARG_INFO(0, full_info)
  103. ZEND_END_ARG_INFO()
  104. ZEND_BEGIN_ARG_INFO(arginfo_dba_list, 0)
  105. ZEND_END_ARG_INFO()
  106. /* }}} */
  107. /* {{{ dba_functions[]
  108. */
  109. static const zend_function_entry dba_functions[] = {
  110. PHP_FE(dba_open, arginfo_dba_open)
  111. PHP_FE(dba_popen, arginfo_dba_popen)
  112. PHP_FE(dba_close, arginfo_dba_close)
  113. PHP_FE(dba_delete, arginfo_dba_delete)
  114. PHP_FE(dba_exists, arginfo_dba_exists)
  115. PHP_FE(dba_fetch, arginfo_dba_fetch)
  116. PHP_FE(dba_insert, arginfo_dba_insert)
  117. PHP_FE(dba_replace, arginfo_dba_replace)
  118. PHP_FE(dba_firstkey, arginfo_dba_firstkey)
  119. PHP_FE(dba_nextkey, arginfo_dba_nextkey)
  120. PHP_FE(dba_optimize, arginfo_dba_optimize)
  121. PHP_FE(dba_sync, arginfo_dba_sync)
  122. PHP_FE(dba_handlers, arginfo_dba_handlers)
  123. PHP_FE(dba_list, arginfo_dba_list)
  124. PHP_FE(dba_key_split, arginfo_dba_key_split)
  125. PHP_FE_END
  126. };
  127. /* }}} */
  128. PHP_MINIT_FUNCTION(dba);
  129. PHP_MSHUTDOWN_FUNCTION(dba);
  130. PHP_MINFO_FUNCTION(dba);
  131. ZEND_BEGIN_MODULE_GLOBALS(dba)
  132. char *default_handler;
  133. dba_handler *default_hptr;
  134. ZEND_END_MODULE_GLOBALS(dba)
  135. ZEND_DECLARE_MODULE_GLOBALS(dba)
  136. #ifdef ZTS
  137. #define DBA_G(v) TSRMG(dba_globals_id, zend_dba_globals *, v)
  138. #else
  139. #define DBA_G(v) (dba_globals.v)
  140. #endif
  141. static PHP_GINIT_FUNCTION(dba);
  142. zend_module_entry dba_module_entry = {
  143. STANDARD_MODULE_HEADER,
  144. "dba",
  145. dba_functions,
  146. PHP_MINIT(dba),
  147. PHP_MSHUTDOWN(dba),
  148. NULL,
  149. NULL,
  150. PHP_MINFO(dba),
  151. PHP_DBA_VERSION,
  152. PHP_MODULE_GLOBALS(dba),
  153. PHP_GINIT(dba),
  154. NULL,
  155. NULL,
  156. STANDARD_MODULE_PROPERTIES_EX
  157. };
  158. #ifdef COMPILE_DL_DBA
  159. ZEND_GET_MODULE(dba)
  160. #endif
  161. /* {{{ macromania */
  162. #define DBA_ID_PARS \
  163. zval *id; \
  164. dba_info *info = NULL; \
  165. int ac = ZEND_NUM_ARGS()
  166. /* these are used to get the standard arguments */
  167. /* {{{ php_dba_myke_key */
  168. static size_t php_dba_make_key(zval *key, char **key_str, char **key_free)
  169. {
  170. if (Z_TYPE_P(key) == IS_ARRAY) {
  171. zval *group, *name;
  172. HashPosition pos;
  173. size_t len;
  174. if (zend_hash_num_elements(Z_ARRVAL_P(key)) != 2) {
  175. zend_throw_error(NULL, "Key does not have exactly two elements: (key, name)");
  176. return 0;
  177. }
  178. zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(key), &pos);
  179. group = zend_hash_get_current_data_ex(Z_ARRVAL_P(key), &pos);
  180. zend_hash_move_forward_ex(Z_ARRVAL_P(key), &pos);
  181. name = zend_hash_get_current_data_ex(Z_ARRVAL_P(key), &pos);
  182. convert_to_string_ex(group);
  183. convert_to_string_ex(name);
  184. if (Z_STRLEN_P(group) == 0) {
  185. *key_str = Z_STRVAL_P(name);
  186. *key_free = NULL;
  187. return Z_STRLEN_P(name);
  188. }
  189. len = spprintf(key_str, 0, "[%s]%s", Z_STRVAL_P(group), Z_STRVAL_P(name));
  190. *key_free = *key_str;
  191. return len;
  192. } else {
  193. zval tmp;
  194. size_t len;
  195. ZVAL_COPY(&tmp, key);
  196. convert_to_string(&tmp);
  197. len = Z_STRLEN(tmp);
  198. if (len) {
  199. *key_free = *key_str = estrndup(Z_STRVAL(tmp), Z_STRLEN(tmp));
  200. }
  201. zval_ptr_dtor(&tmp);
  202. return len;
  203. }
  204. }
  205. /* }}} */
  206. #define DBA_GET2 \
  207. zval *key; \
  208. char *key_str, *key_free; \
  209. size_t key_len; \
  210. if (zend_parse_parameters(ac, "zr", &key, &id) == FAILURE) { \
  211. return; \
  212. } \
  213. if ((key_len = php_dba_make_key(key, &key_str, &key_free)) == 0) {\
  214. RETURN_FALSE; \
  215. }
  216. #define DBA_GET2_3 \
  217. zval *key; \
  218. char *key_str, *key_free; \
  219. size_t key_len; \
  220. zend_long skip = 0; \
  221. switch(ac) { \
  222. case 2: \
  223. if (zend_parse_parameters(ac, "zr", &key, &id) == FAILURE) { \
  224. return; \
  225. } \
  226. break; \
  227. case 3: \
  228. if (zend_parse_parameters(ac, "zlr", &key, &skip, &id) == FAILURE) { \
  229. return; \
  230. } \
  231. break; \
  232. default: \
  233. WRONG_PARAM_COUNT; \
  234. } \
  235. if ((key_len = php_dba_make_key(key, &key_str, &key_free)) == 0) {\
  236. RETURN_FALSE; \
  237. }
  238. #define DBA_FETCH_RESOURCE(info, id) \
  239. if ((info = (dba_info *)zend_fetch_resource2(Z_RES_P(id), "DBA identifier", le_db, le_pdb)) == NULL) { \
  240. RETURN_FALSE; \
  241. }
  242. #define DBA_FETCH_RESOURCE_WITH_ID(info, id) \
  243. if ((info = (dba_info *)zend_fetch_resource2(Z_RES_P(id), "DBA identifier", le_db, le_pdb)) == NULL) { \
  244. DBA_ID_DONE; \
  245. RETURN_FALSE; \
  246. }
  247. #define DBA_ID_GET2 DBA_ID_PARS; DBA_GET2; DBA_FETCH_RESOURCE_WITH_ID(info, id)
  248. #define DBA_ID_GET2_3 DBA_ID_PARS; DBA_GET2_3; DBA_FETCH_RESOURCE_WITH_ID(info, id)
  249. #define DBA_ID_DONE \
  250. if (key_free) efree(key_free)
  251. /* a DBA handler must have specific routines */
  252. #define DBA_NAMED_HND(alias, name, flags) \
  253. {\
  254. #alias, flags, dba_open_##name, dba_close_##name, dba_fetch_##name, dba_update_##name, \
  255. dba_exists_##name, dba_delete_##name, dba_firstkey_##name, dba_nextkey_##name, \
  256. dba_optimize_##name, dba_sync_##name, dba_info_##name \
  257. },
  258. #define DBA_HND(name, flags) DBA_NAMED_HND(name, name, flags)
  259. /* check whether the user has write access */
  260. #define DBA_WRITE_CHECK \
  261. if(info->mode != DBA_WRITER && info->mode != DBA_TRUNC && info->mode != DBA_CREAT) { \
  262. php_error_docref(NULL, E_WARNING, "You cannot perform a modification to a database without proper access"); \
  263. RETURN_FALSE; \
  264. }
  265. /* the same check, but with a call to DBA_ID_DONE before returning */
  266. #define DBA_WRITE_CHECK_WITH_ID \
  267. if(info->mode != DBA_WRITER && info->mode != DBA_TRUNC && info->mode != DBA_CREAT) { \
  268. php_error_docref(NULL, E_WARNING, "You cannot perform a modification to a database without proper access"); \
  269. DBA_ID_DONE; \
  270. RETURN_FALSE; \
  271. }
  272. /* }}} */
  273. /* {{{ globals */
  274. static dba_handler handler[] = {
  275. #if DBA_GDBM
  276. DBA_HND(gdbm, DBA_LOCK_EXT) /* Locking done in library if set */
  277. #endif
  278. #if DBA_DBM
  279. DBA_HND(dbm, DBA_LOCK_ALL) /* No lock in lib */
  280. #endif
  281. #if DBA_NDBM
  282. DBA_HND(ndbm, DBA_LOCK_ALL) /* Could be done in library: filemode = 0644 + S_ENFMT */
  283. #endif
  284. #if DBA_CDB
  285. DBA_HND(cdb, DBA_STREAM_OPEN|DBA_LOCK_ALL) /* No lock in lib */
  286. #endif
  287. #if DBA_CDB_BUILTIN
  288. DBA_NAMED_HND(cdb_make, cdb, DBA_STREAM_OPEN|DBA_LOCK_ALL) /* No lock in lib */
  289. #endif
  290. #if DBA_DB1
  291. DBA_HND(db1, DBA_LOCK_ALL) /* No lock in lib */
  292. #endif
  293. #if DBA_DB2
  294. DBA_HND(db2, DBA_LOCK_ALL) /* No lock in lib */
  295. #endif
  296. #if DBA_DB3
  297. DBA_HND(db3, DBA_LOCK_ALL) /* No lock in lib */
  298. #endif
  299. #if DBA_DB4
  300. DBA_HND(db4, DBA_LOCK_ALL) /* No lock in lib */
  301. #endif
  302. #if DBA_INIFILE
  303. DBA_HND(inifile, DBA_STREAM_OPEN|DBA_LOCK_ALL|DBA_CAST_AS_FD) /* No lock in lib */
  304. #endif
  305. #if DBA_FLATFILE
  306. DBA_HND(flatfile, DBA_STREAM_OPEN|DBA_LOCK_ALL|DBA_NO_APPEND) /* No lock in lib */
  307. #endif
  308. #if DBA_QDBM
  309. DBA_HND(qdbm, DBA_LOCK_EXT)
  310. #endif
  311. #if DBA_TCADB
  312. DBA_HND(tcadb, DBA_LOCK_ALL)
  313. #endif
  314. #if DBA_LMDB
  315. DBA_HND(lmdb, DBA_LOCK_EXT)
  316. #endif
  317. { NULL, 0, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL }
  318. };
  319. #if DBA_FLATFILE
  320. #define DBA_DEFAULT "flatfile"
  321. #elif DBA_DB4
  322. #define DBA_DEFAULT "db4"
  323. #elif DBA_DB3
  324. #define DBA_DEFAULT "db3"
  325. #elif DBA_DB2
  326. #define DBA_DEFAULT "db2"
  327. #elif DBA_DB1
  328. #define DBA_DEFAULT "db1"
  329. #elif DBA_GDBM
  330. #define DBA_DEFAULT "gdbm"
  331. #elif DBA_NBBM
  332. #define DBA_DEFAULT "ndbm"
  333. #elif DBA_DBM
  334. #define DBA_DEFAULT "dbm"
  335. #elif DBA_QDBM
  336. #define DBA_DEFAULT "qdbm"
  337. #elif DBA_TCADB
  338. #define DBA_DEFAULT "tcadb"
  339. #elif DBA_LMDB
  340. #define DBA_DEFAULT "lmdb"
  341. #else
  342. #define DBA_DEFAULT ""
  343. #endif
  344. /* cdb/cdb_make and ini are no option here */
  345. static int le_db;
  346. static int le_pdb;
  347. /* }}} */
  348. /* {{{ dba_fetch_resource
  349. PHPAPI void dba_fetch_resource(dba_info **pinfo, zval **id)
  350. {
  351. dba_info *info;
  352. DBA_ID_FETCH
  353. *pinfo = info;
  354. }
  355. */
  356. /* }}} */
  357. /* {{{ dba_get_handler
  358. PHPAPI dba_handler *dba_get_handler(const char* handler_name)
  359. {
  360. dba_handler *hptr;
  361. for (hptr = handler; hptr->name && strcasecmp(hptr->name, handler_name); hptr++);
  362. return hptr;
  363. }
  364. */
  365. /* }}} */
  366. /* {{{ dba_close
  367. */
  368. static void dba_close(dba_info *info)
  369. {
  370. if (info->hnd) {
  371. info->hnd->close(info);
  372. }
  373. if (info->path) {
  374. pefree(info->path, info->flags&DBA_PERSISTENT);
  375. }
  376. if (info->fp && info->fp != info->lock.fp) {
  377. if (info->flags & DBA_PERSISTENT) {
  378. php_stream_pclose(info->fp);
  379. } else {
  380. php_stream_close(info->fp);
  381. }
  382. }
  383. if (info->lock.fp) {
  384. if (info->flags & DBA_PERSISTENT) {
  385. php_stream_pclose(info->lock.fp);
  386. } else {
  387. php_stream_close(info->lock.fp);
  388. }
  389. }
  390. if (info->lock.name) {
  391. pefree(info->lock.name, info->flags&DBA_PERSISTENT);
  392. }
  393. pefree(info, info->flags&DBA_PERSISTENT);
  394. }
  395. /* }}} */
  396. /* {{{ dba_close_rsrc
  397. */
  398. static void dba_close_rsrc(zend_resource *rsrc)
  399. {
  400. dba_info *info = (dba_info *)rsrc->ptr;
  401. dba_close(info);
  402. }
  403. /* }}} */
  404. /* {{{ dba_close_pe_rsrc_deleter */
  405. int dba_close_pe_rsrc_deleter(zval *el, void *pDba)
  406. {
  407. if (Z_RES_P(el)->ptr == pDba) {
  408. if (Z_DELREF_P(el) == 0) {
  409. return ZEND_HASH_APPLY_REMOVE;
  410. } else {
  411. return ZEND_HASH_APPLY_KEEP | ZEND_HASH_APPLY_STOP;
  412. }
  413. } else {
  414. return ZEND_HASH_APPLY_KEEP;
  415. }
  416. }
  417. /* }}} */
  418. /* {{{ dba_close_pe_rsrc */
  419. static void dba_close_pe_rsrc(zend_resource *rsrc)
  420. {
  421. dba_info *info = (dba_info *)rsrc->ptr;
  422. /* closes the resource by calling dba_close_rsrc() */
  423. zend_hash_apply_with_argument(&EG(persistent_list), dba_close_pe_rsrc_deleter, info);
  424. }
  425. /* }}} */
  426. /* {{{ PHP_INI
  427. */
  428. ZEND_INI_MH(OnUpdateDefaultHandler)
  429. {
  430. dba_handler *hptr;
  431. if (!ZSTR_LEN(new_value)) {
  432. DBA_G(default_hptr) = NULL;
  433. return OnUpdateString(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
  434. }
  435. for (hptr = handler; hptr->name && strcasecmp(hptr->name, ZSTR_VAL(new_value)); hptr++);
  436. if (!hptr->name) {
  437. php_error_docref(NULL, E_WARNING, "No such handler: %s", ZSTR_VAL(new_value));
  438. return FAILURE;
  439. }
  440. DBA_G(default_hptr) = hptr;
  441. return OnUpdateString(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
  442. }
  443. PHP_INI_BEGIN()
  444. STD_PHP_INI_ENTRY("dba.default_handler", DBA_DEFAULT, PHP_INI_ALL, OnUpdateDefaultHandler, default_handler, zend_dba_globals, dba_globals)
  445. PHP_INI_END()
  446. /* }}} */
  447. /* {{{ PHP_GINIT_FUNCTION
  448. */
  449. static PHP_GINIT_FUNCTION(dba)
  450. {
  451. dba_globals->default_handler = "";
  452. dba_globals->default_hptr = NULL;
  453. }
  454. /* }}} */
  455. /* {{{ PHP_MINIT_FUNCTION
  456. */
  457. PHP_MINIT_FUNCTION(dba)
  458. {
  459. REGISTER_INI_ENTRIES();
  460. le_db = zend_register_list_destructors_ex(dba_close_rsrc, NULL, "dba", module_number);
  461. le_pdb = zend_register_list_destructors_ex(dba_close_pe_rsrc, dba_close_rsrc, "dba persistent", module_number);
  462. return SUCCESS;
  463. }
  464. /* }}} */
  465. /* {{{ PHP_MSHUTDOWN_FUNCTION
  466. */
  467. PHP_MSHUTDOWN_FUNCTION(dba)
  468. {
  469. UNREGISTER_INI_ENTRIES();
  470. return SUCCESS;
  471. }
  472. /* }}} */
  473. #include "zend_smart_str.h"
  474. /* {{{ PHP_MINFO_FUNCTION
  475. */
  476. PHP_MINFO_FUNCTION(dba)
  477. {
  478. dba_handler *hptr;
  479. smart_str handlers = {0};
  480. for(hptr = handler; hptr->name; hptr++) {
  481. smart_str_appends(&handlers, hptr->name);
  482. smart_str_appendc(&handlers, ' ');
  483. }
  484. php_info_print_table_start();
  485. php_info_print_table_row(2, "DBA support", "enabled");
  486. if (handlers.s) {
  487. smart_str_0(&handlers);
  488. php_info_print_table_row(2, "Supported handlers", ZSTR_VAL(handlers.s));
  489. smart_str_free(&handlers);
  490. } else {
  491. php_info_print_table_row(2, "Supported handlers", "none");
  492. }
  493. php_info_print_table_end();
  494. DISPLAY_INI_ENTRIES();
  495. }
  496. /* }}} */
  497. /* {{{ php_dba_update
  498. */
  499. static void php_dba_update(INTERNAL_FUNCTION_PARAMETERS, int mode)
  500. {
  501. size_t val_len;
  502. zval *id;
  503. dba_info *info = NULL;
  504. int ac = ZEND_NUM_ARGS();
  505. zval *key;
  506. char *val;
  507. char *key_str, *key_free;
  508. size_t key_len;
  509. if (zend_parse_parameters(ac, "zsr", &key, &val, &val_len, &id) == FAILURE) {
  510. return;
  511. }
  512. if ((key_len = php_dba_make_key(key, &key_str, &key_free)) == 0) {
  513. RETURN_FALSE;
  514. }
  515. DBA_FETCH_RESOURCE_WITH_ID(info, id);
  516. DBA_WRITE_CHECK_WITH_ID;
  517. if (info->hnd->update(info, key_str, key_len, val, val_len, mode) == SUCCESS) {
  518. DBA_ID_DONE;
  519. RETURN_TRUE;
  520. }
  521. DBA_ID_DONE;
  522. RETURN_FALSE;
  523. }
  524. /* }}} */
  525. #define FREENOW if(args) {int i; for (i=0; i<ac; i++) { zval_ptr_dtor(&args[i]); } efree(args);} if(key) efree(key)
  526. /* {{{ php_find_dbm
  527. */
  528. dba_info *php_dba_find(const char* path)
  529. {
  530. zend_resource *le;
  531. dba_info *info;
  532. zend_long numitems, i;
  533. numitems = zend_hash_next_free_element(&EG(regular_list));
  534. for (i=1; i<numitems; i++) {
  535. if ((le = zend_hash_index_find_ptr(&EG(regular_list), i)) == NULL) {
  536. continue;
  537. }
  538. if (le->type == le_db || le->type == le_pdb) {
  539. info = (dba_info *)(le->ptr);
  540. if (!strcmp(info->path, path)) {
  541. return (dba_info *)(le->ptr);
  542. }
  543. }
  544. }
  545. return NULL;
  546. }
  547. /* }}} */
  548. /* {{{ php_dba_open
  549. */
  550. static void php_dba_open(INTERNAL_FUNCTION_PARAMETERS, int persistent)
  551. {
  552. zval *args = NULL;
  553. int ac = ZEND_NUM_ARGS();
  554. dba_mode_t modenr;
  555. dba_info *info, *other;
  556. dba_handler *hptr;
  557. char *key = NULL, *error = NULL;
  558. size_t keylen = 0;
  559. int i;
  560. int lock_mode, lock_flag, lock_dbf = 0;
  561. char *file_mode;
  562. char mode[4], *pmode, *lock_file_mode = NULL;
  563. int persistent_flag = persistent ? STREAM_OPEN_PERSISTENT : 0;
  564. zend_string *opened_path = NULL;
  565. char *lock_name;
  566. #ifdef PHP_WIN32
  567. zend_bool restarted = 0;
  568. zend_bool need_creation = 0;
  569. #endif
  570. if (ac < 2) {
  571. WRONG_PARAM_COUNT;
  572. }
  573. /* we pass additional args to the respective handler */
  574. args = safe_emalloc(ac, sizeof(zval), 0);
  575. if (zend_get_parameters_array_ex(ac, args) != SUCCESS) {
  576. efree(args);
  577. WRONG_PARAM_COUNT;
  578. }
  579. /* we only take string arguments */
  580. for (i = 0; i < ac; i++) {
  581. ZVAL_STR(&args[i], zval_get_string(&args[i]));
  582. keylen += Z_STRLEN(args[i]);
  583. }
  584. if (persistent) {
  585. zend_resource *le;
  586. /* calculate hash */
  587. key = safe_emalloc(keylen, 1, 1);
  588. key[keylen] = '\0';
  589. keylen = 0;
  590. for(i = 0; i < ac; i++) {
  591. memcpy(key+keylen, Z_STRVAL(args[i]), Z_STRLEN(args[i]));
  592. keylen += Z_STRLEN(args[i]);
  593. }
  594. /* try to find if we already have this link in our persistent list */
  595. if ((le = zend_hash_str_find_ptr(&EG(persistent_list), key, keylen)) != NULL) {
  596. FREENOW;
  597. if (le->type != le_pdb) {
  598. RETURN_FALSE;
  599. }
  600. info = (dba_info *)le->ptr;
  601. GC_ADDREF(le);
  602. RETURN_RES(zend_register_resource(info, le_pdb));
  603. return;
  604. }
  605. }
  606. if (ac==2) {
  607. hptr = DBA_G(default_hptr);
  608. if (!hptr) {
  609. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "No default handler selected");
  610. FREENOW;
  611. RETURN_FALSE;
  612. }
  613. } else {
  614. for (hptr = handler; hptr->name && strcasecmp(hptr->name, Z_STRVAL(args[2])); hptr++);
  615. }
  616. if (!hptr->name) {
  617. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "No such handler: %s", Z_STRVAL(args[2]));
  618. FREENOW;
  619. RETURN_FALSE;
  620. }
  621. /* Check mode: [rwnc][fl]?t?
  622. * r: Read
  623. * w: Write
  624. * n: Create/Truncate
  625. * c: Create
  626. *
  627. * d: force lock on database file
  628. * l: force lock on lck file
  629. * -: ignore locking
  630. *
  631. * t: test open database, warning if locked
  632. */
  633. strlcpy(mode, Z_STRVAL(args[1]), sizeof(mode));
  634. pmode = &mode[0];
  635. if (pmode[0] && (pmode[1]=='d' || pmode[1]=='l' || pmode[1]=='-')) { /* force lock on db file or lck file or disable locking */
  636. switch (pmode[1]) {
  637. case 'd':
  638. lock_dbf = 1;
  639. if ((hptr->flags & DBA_LOCK_ALL) == 0) {
  640. lock_flag = (hptr->flags & DBA_LOCK_ALL);
  641. break;
  642. }
  643. /* no break */
  644. case 'l':
  645. lock_flag = DBA_LOCK_ALL;
  646. if ((hptr->flags & DBA_LOCK_ALL) == 0) {
  647. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_NOTICE, "Handler %s does locking internally", hptr->name);
  648. }
  649. break;
  650. default:
  651. case '-':
  652. if ((hptr->flags & DBA_LOCK_ALL) == 0) {
  653. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "Locking cannot be disabled for handler %s", hptr->name);
  654. FREENOW;
  655. RETURN_FALSE;
  656. }
  657. lock_flag = 0;
  658. break;
  659. }
  660. } else {
  661. lock_flag = (hptr->flags&DBA_LOCK_ALL);
  662. lock_dbf = 1;
  663. }
  664. switch (*pmode++) {
  665. case 'r':
  666. modenr = DBA_READER;
  667. lock_mode = (lock_flag & DBA_LOCK_READER) ? LOCK_SH : 0;
  668. file_mode = "r";
  669. break;
  670. case 'w':
  671. modenr = DBA_WRITER;
  672. lock_mode = (lock_flag & DBA_LOCK_WRITER) ? LOCK_EX : 0;
  673. file_mode = "r+b";
  674. break;
  675. case 'c': {
  676. #ifdef PHP_WIN32
  677. if (hptr->flags & (DBA_NO_APPEND|DBA_CAST_AS_FD)) {
  678. php_stream_statbuf ssb;
  679. need_creation = (SUCCESS != php_stream_stat_path(Z_STRVAL(args[0]), &ssb));
  680. }
  681. #endif
  682. modenr = DBA_CREAT;
  683. lock_mode = (lock_flag & DBA_LOCK_CREAT) ? LOCK_EX : 0;
  684. if (lock_mode) {
  685. if (lock_dbf) {
  686. /* the create/append check will be done on the lock
  687. * when the lib opens the file it is already created
  688. */
  689. file_mode = "r+b"; /* read & write, seek 0 */
  690. #ifdef PHP_WIN32
  691. if (!need_creation) {
  692. lock_file_mode = "r+b";
  693. } else
  694. #endif
  695. lock_file_mode = "a+b"; /* append */
  696. } else {
  697. #ifdef PHP_WIN32
  698. if (!need_creation) {
  699. file_mode = "r+b";
  700. } else
  701. #endif
  702. file_mode = "a+b"; /* append */
  703. lock_file_mode = "w+b"; /* create/truncate */
  704. }
  705. } else {
  706. #ifdef PHP_WIN32
  707. if (!need_creation) {
  708. file_mode = "r+b";
  709. } else
  710. #endif
  711. file_mode = "a+b";
  712. }
  713. /* In case of the 'a+b' append mode, the handler is responsible
  714. * to handle any rewind problems (see flatfile handler).
  715. */
  716. break;
  717. }
  718. case 'n':
  719. modenr = DBA_TRUNC;
  720. lock_mode = (lock_flag & DBA_LOCK_TRUNC) ? LOCK_EX : 0;
  721. file_mode = "w+b";
  722. break;
  723. default:
  724. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "Illegal DBA mode");
  725. FREENOW;
  726. RETURN_FALSE;
  727. }
  728. if (!lock_file_mode) {
  729. lock_file_mode = file_mode;
  730. }
  731. if (*pmode=='d' || *pmode=='l' || *pmode=='-') {
  732. pmode++; /* done already - skip here */
  733. }
  734. if (*pmode=='t') {
  735. pmode++;
  736. if (!lock_flag) {
  737. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "You cannot combine modifiers - (no lock) and t (test lock)");
  738. FREENOW;
  739. RETURN_FALSE;
  740. }
  741. if (!lock_mode) {
  742. if ((hptr->flags & DBA_LOCK_ALL) == 0) {
  743. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "Handler %s uses its own locking which doesn't support mode modifier t (test lock)", hptr->name);
  744. FREENOW;
  745. RETURN_FALSE;
  746. } else {
  747. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "Handler %s doesn't uses locking for this mode which makes modifier t (test lock) obsolete", hptr->name);
  748. FREENOW;
  749. RETURN_FALSE;
  750. }
  751. } else {
  752. lock_mode |= LOCK_NB; /* test =: non blocking */
  753. }
  754. }
  755. if (*pmode) {
  756. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "Illegal DBA mode");
  757. FREENOW;
  758. RETURN_FALSE;
  759. }
  760. info = pemalloc(sizeof(dba_info), persistent);
  761. memset(info, 0, sizeof(dba_info));
  762. info->path = pestrdup(Z_STRVAL(args[0]), persistent);
  763. info->mode = modenr;
  764. info->argc = ac - 3;
  765. info->argv = args + 3;
  766. info->flags = (hptr->flags & ~DBA_LOCK_ALL) | (lock_flag & DBA_LOCK_ALL) | (persistent ? DBA_PERSISTENT : 0);
  767. info->lock.mode = lock_mode;
  768. /* if any open call is a locking call:
  769. * check if we already habe a locking call open that should block this call
  770. * the problem is some systems would allow read during write
  771. */
  772. if (hptr->flags & DBA_LOCK_ALL) {
  773. if ((other = php_dba_find(info->path)) != NULL) {
  774. if ( ( (lock_mode&LOCK_EX) && (other->lock.mode&(LOCK_EX|LOCK_SH)) )
  775. || ( (other->lock.mode&LOCK_EX) && (lock_mode&(LOCK_EX|LOCK_SH)) )
  776. ) {
  777. error = "Unable to establish lock (database file already open)"; /* force failure exit */
  778. }
  779. }
  780. }
  781. #ifdef PHP_WIN32
  782. restart:
  783. #endif
  784. if (!error && lock_mode) {
  785. if (lock_dbf) {
  786. lock_name = Z_STRVAL(args[0]);
  787. } else {
  788. spprintf(&lock_name, 0, "%s.lck", info->path);
  789. if (!strcmp(file_mode, "r")) {
  790. /* when in read only mode try to use existing .lck file first */
  791. /* do not log errors for .lck file while in read ony mode on .lck file */
  792. lock_file_mode = "rb";
  793. info->lock.fp = php_stream_open_wrapper(lock_name, lock_file_mode, STREAM_MUST_SEEK|IGNORE_PATH|persistent_flag, &opened_path);
  794. }
  795. if (!info->lock.fp) {
  796. /* when not in read mode or failed to open .lck file read only. now try again in create(write) mode and log errors */
  797. lock_file_mode = "a+b";
  798. } else {
  799. if (opened_path) {
  800. info->lock.name = pestrndup(ZSTR_VAL(opened_path), ZSTR_LEN(opened_path), persistent);
  801. zend_string_release_ex(opened_path, 0);
  802. }
  803. }
  804. }
  805. if (!info->lock.fp) {
  806. info->lock.fp = php_stream_open_wrapper(lock_name, lock_file_mode, STREAM_MUST_SEEK|REPORT_ERRORS|IGNORE_PATH|persistent_flag, &opened_path);
  807. if (info->lock.fp) {
  808. if (lock_dbf) {
  809. /* replace the path info with the real path of the opened file */
  810. pefree(info->path, persistent);
  811. info->path = pestrndup(ZSTR_VAL(opened_path), ZSTR_LEN(opened_path), persistent);
  812. }
  813. /* now store the name of the lock */
  814. info->lock.name = pestrndup(ZSTR_VAL(opened_path), ZSTR_LEN(opened_path), persistent);
  815. zend_string_release_ex(opened_path, 0);
  816. }
  817. }
  818. if (!lock_dbf) {
  819. efree(lock_name);
  820. }
  821. if (!info->lock.fp) {
  822. dba_close(info);
  823. /* stream operation already wrote an error message */
  824. FREENOW;
  825. RETURN_FALSE;
  826. }
  827. if (!php_stream_supports_lock(info->lock.fp)) {
  828. error = "Stream does not support locking";
  829. }
  830. if (php_stream_lock(info->lock.fp, lock_mode)) {
  831. error = "Unable to establish lock"; /* force failure exit */
  832. }
  833. }
  834. /* centralised open stream for builtin */
  835. if (!error && (hptr->flags&DBA_STREAM_OPEN)==DBA_STREAM_OPEN) {
  836. if (info->lock.fp && lock_dbf) {
  837. info->fp = info->lock.fp; /* use the same stream for locking and database access */
  838. } else {
  839. info->fp = php_stream_open_wrapper(info->path, file_mode, STREAM_MUST_SEEK|REPORT_ERRORS|IGNORE_PATH|persistent_flag, NULL);
  840. }
  841. if (!info->fp) {
  842. dba_close(info);
  843. /* stream operation already wrote an error message */
  844. FREENOW;
  845. RETURN_FALSE;
  846. }
  847. if (hptr->flags & (DBA_NO_APPEND|DBA_CAST_AS_FD)) {
  848. /* Needed because some systems do not allow to write to the original
  849. * file contents with O_APPEND being set.
  850. */
  851. if (SUCCESS != php_stream_cast(info->fp, PHP_STREAM_AS_FD, (void*)&info->fd, 1)) {
  852. php_error_docref(NULL, E_WARNING, "Could not cast stream");
  853. dba_close(info);
  854. FREENOW;
  855. RETURN_FALSE;
  856. #ifdef F_SETFL
  857. } else if (modenr == DBA_CREAT) {
  858. int flags = fcntl(info->fd, F_GETFL);
  859. fcntl(info->fd, F_SETFL, flags & ~O_APPEND);
  860. #elif defined(PHP_WIN32)
  861. } else if (modenr == DBA_CREAT && need_creation && !restarted) {
  862. zend_bool close_both;
  863. close_both = (info->fp != info->lock.fp);
  864. php_stream_close(info->lock.fp);
  865. if (close_both) {
  866. php_stream_close(info->fp);
  867. }
  868. info->fp = NULL;
  869. info->lock.fp = NULL;
  870. info->fd = -1;
  871. pefree(info->lock.name, persistent);
  872. lock_file_mode = "r+b";
  873. restarted = 1;
  874. goto restart;
  875. #endif
  876. }
  877. }
  878. }
  879. if (error || hptr->open(info, &error) != SUCCESS) {
  880. dba_close(info);
  881. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "Driver initialization failed for handler: %s%s%s", hptr->name, error?": ":"", error?error:"");
  882. FREENOW;
  883. RETURN_FALSE;
  884. }
  885. info->hnd = hptr;
  886. info->argc = 0;
  887. info->argv = NULL;
  888. if (persistent) {
  889. if (zend_register_persistent_resource(key, keylen, info, le_pdb) == NULL) {
  890. dba_close(info);
  891. php_error_docref2(NULL, Z_STRVAL(args[0]), Z_STRVAL(args[1]), E_WARNING, "Could not register persistent resource");
  892. FREENOW;
  893. RETURN_FALSE;
  894. }
  895. }
  896. RETVAL_RES(zend_register_resource(info, (persistent ? le_pdb : le_db)));
  897. FREENOW;
  898. }
  899. /* }}} */
  900. #undef FREENOW
  901. /* {{{ proto resource dba_popen(string path, string mode [, string handlername, string ...])
  902. Opens path using the specified handler in mode persistently */
  903. PHP_FUNCTION(dba_popen)
  904. {
  905. php_dba_open(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
  906. }
  907. /* }}} */
  908. /* {{{ proto resource dba_open(string path, string mode [, string handlername, string ...])
  909. Opens path using the specified handler in mode*/
  910. PHP_FUNCTION(dba_open)
  911. {
  912. php_dba_open(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
  913. }
  914. /* }}} */
  915. /* {{{ proto void dba_close(resource handle)
  916. Closes database */
  917. PHP_FUNCTION(dba_close)
  918. {
  919. zval *id;
  920. dba_info *info = NULL;
  921. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &id) == FAILURE) {
  922. return;
  923. }
  924. DBA_FETCH_RESOURCE(info, id);
  925. zend_list_close(Z_RES_P(id));
  926. }
  927. /* }}} */
  928. /* {{{ proto bool dba_exists(string key, resource handle)
  929. Checks, if the specified key exists */
  930. PHP_FUNCTION(dba_exists)
  931. {
  932. DBA_ID_GET2;
  933. if(info->hnd->exists(info, key_str, key_len) == SUCCESS) {
  934. DBA_ID_DONE;
  935. RETURN_TRUE;
  936. }
  937. DBA_ID_DONE;
  938. RETURN_FALSE;
  939. }
  940. /* }}} */
  941. /* {{{ proto string dba_fetch(string key, [int skip ,] resource handle)
  942. Fetches the data associated with key */
  943. PHP_FUNCTION(dba_fetch)
  944. {
  945. char *val;
  946. size_t len = 0;
  947. DBA_ID_GET2_3;
  948. if (ac==3) {
  949. if (!strcmp(info->hnd->name, "cdb")) {
  950. if (skip < 0) {
  951. php_error_docref(NULL, E_NOTICE, "Handler %s accepts only skip values greater than or equal to zero, using skip=0", info->hnd->name);
  952. skip = 0;
  953. }
  954. } else if (!strcmp(info->hnd->name, "inifile")) {
  955. /* "-1" is compareable to 0 but allows a non restrictive
  956. * access which is fater. For example 'inifile' uses this
  957. * to allow faster access when the key was already found
  958. * using firstkey/nextkey. However explicitly setting the
  959. * value to 0 ensures the first value.
  960. */
  961. if (skip < -1) {
  962. php_error_docref(NULL, E_NOTICE, "Handler %s accepts only skip value -1 and greater, using skip=0", info->hnd->name);
  963. skip = 0;
  964. }
  965. } else {
  966. php_error_docref(NULL, E_NOTICE, "Handler %s does not support optional skip parameter, the value will be ignored", info->hnd->name);
  967. skip = 0;
  968. }
  969. } else {
  970. skip = 0;
  971. }
  972. if((val = info->hnd->fetch(info, key_str, key_len, skip, &len)) != NULL) {
  973. DBA_ID_DONE;
  974. RETVAL_STRINGL(val, len);
  975. efree(val);
  976. return;
  977. }
  978. DBA_ID_DONE;
  979. RETURN_FALSE;
  980. }
  981. /* }}} */
  982. /* {{{ proto array|false dba_key_split(string key)
  983. Splits an inifile key into an array of the form array(0=>group,1=>value_name) but returns false if input is false or null */
  984. PHP_FUNCTION(dba_key_split)
  985. {
  986. zval *zkey;
  987. char *key, *name;
  988. size_t key_len;
  989. if (ZEND_NUM_ARGS() != 1) {
  990. WRONG_PARAM_COUNT;
  991. }
  992. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "z", &zkey) == SUCCESS) {
  993. if (Z_TYPE_P(zkey) == IS_NULL || (Z_TYPE_P(zkey) == IS_FALSE)) {
  994. RETURN_BOOL(0);
  995. }
  996. }
  997. if (zend_parse_parameters(ZEND_NUM_ARGS(), "s", &key, &key_len) == FAILURE) {
  998. RETURN_BOOL(0);
  999. }
  1000. array_init(return_value);
  1001. if (key[0] == '[' && (name = strchr(key, ']')) != NULL) {
  1002. add_next_index_stringl(return_value, key+1, name - (key + 1));
  1003. add_next_index_stringl(return_value, name+1, key_len - (name - key + 1));
  1004. } else {
  1005. add_next_index_stringl(return_value, "", 0);
  1006. add_next_index_stringl(return_value, key, key_len);
  1007. }
  1008. }
  1009. /* }}} */
  1010. /* {{{ proto string dba_firstkey(resource handle)
  1011. Resets the internal key pointer and returns the first key */
  1012. PHP_FUNCTION(dba_firstkey)
  1013. {
  1014. char *fkey;
  1015. size_t len;
  1016. zval *id;
  1017. dba_info *info = NULL;
  1018. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &id) == FAILURE) {
  1019. return;
  1020. }
  1021. DBA_FETCH_RESOURCE(info, id);
  1022. fkey = info->hnd->firstkey(info, &len);
  1023. if (fkey) {
  1024. RETVAL_STRINGL(fkey, len);
  1025. efree(fkey);
  1026. return;
  1027. }
  1028. RETURN_FALSE;
  1029. }
  1030. /* }}} */
  1031. /* {{{ proto string dba_nextkey(resource handle)
  1032. Returns the next key */
  1033. PHP_FUNCTION(dba_nextkey)
  1034. {
  1035. char *nkey;
  1036. size_t len;
  1037. zval *id;
  1038. dba_info *info = NULL;
  1039. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &id) == FAILURE) {
  1040. return;
  1041. }
  1042. DBA_FETCH_RESOURCE(info, id);
  1043. nkey = info->hnd->nextkey(info, &len);
  1044. if (nkey) {
  1045. RETVAL_STRINGL(nkey, len);
  1046. efree(nkey);
  1047. return;
  1048. }
  1049. RETURN_FALSE;
  1050. }
  1051. /* }}} */
  1052. /* {{{ proto bool dba_delete(string key, resource handle)
  1053. Deletes the entry associated with key
  1054. If inifile: remove all other key lines */
  1055. PHP_FUNCTION(dba_delete)
  1056. {
  1057. DBA_ID_GET2;
  1058. DBA_WRITE_CHECK_WITH_ID;
  1059. if(info->hnd->delete(info, key_str, key_len) == SUCCESS)
  1060. {
  1061. DBA_ID_DONE;
  1062. RETURN_TRUE;
  1063. }
  1064. DBA_ID_DONE;
  1065. RETURN_FALSE;
  1066. }
  1067. /* }}} */
  1068. /* {{{ proto bool dba_insert(string key, string value, resource handle)
  1069. If not inifile: Insert value as key, return false, if key exists already
  1070. If inifile: Add vakue as key (next instance of key) */
  1071. PHP_FUNCTION(dba_insert)
  1072. {
  1073. php_dba_update(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
  1074. }
  1075. /* }}} */
  1076. /* {{{ proto bool dba_replace(string key, string value, resource handle)
  1077. Inserts value as key, replaces key, if key exists already
  1078. If inifile: remove all other key lines */
  1079. PHP_FUNCTION(dba_replace)
  1080. {
  1081. php_dba_update(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
  1082. }
  1083. /* }}} */
  1084. /* {{{ proto bool dba_optimize(resource handle)
  1085. Optimizes (e.g. clean up, vacuum) database */
  1086. PHP_FUNCTION(dba_optimize)
  1087. {
  1088. zval *id;
  1089. dba_info *info = NULL;
  1090. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &id) == FAILURE) {
  1091. return;
  1092. }
  1093. DBA_FETCH_RESOURCE(info, id);
  1094. DBA_WRITE_CHECK;
  1095. if (info->hnd->optimize(info) == SUCCESS) {
  1096. RETURN_TRUE;
  1097. }
  1098. RETURN_FALSE;
  1099. }
  1100. /* }}} */
  1101. /* {{{ proto bool dba_sync(resource handle)
  1102. Synchronizes database */
  1103. PHP_FUNCTION(dba_sync)
  1104. {
  1105. zval *id;
  1106. dba_info *info = NULL;
  1107. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &id) == FAILURE) {
  1108. return;
  1109. }
  1110. DBA_FETCH_RESOURCE(info, id);
  1111. if (info->hnd->sync(info) == SUCCESS) {
  1112. RETURN_TRUE;
  1113. }
  1114. RETURN_FALSE;
  1115. }
  1116. /* }}} */
  1117. /* {{{ proto array dba_handlers([bool full_info])
  1118. List configured database handlers */
  1119. PHP_FUNCTION(dba_handlers)
  1120. {
  1121. dba_handler *hptr;
  1122. zend_bool full_info = 0;
  1123. if (zend_parse_parameters(ZEND_NUM_ARGS(), "|b", &full_info) == FAILURE) {
  1124. RETURN_FALSE;
  1125. }
  1126. array_init(return_value);
  1127. for(hptr = handler; hptr->name; hptr++) {
  1128. if (full_info) {
  1129. // TODO: avoid reallocation ???
  1130. char *str = hptr->info(hptr, NULL);
  1131. add_assoc_string(return_value, hptr->name, str);
  1132. efree(str);
  1133. } else {
  1134. add_next_index_string(return_value, hptr->name);
  1135. }
  1136. }
  1137. }
  1138. /* }}} */
  1139. /* {{{ proto array dba_list()
  1140. List opened databases */
  1141. PHP_FUNCTION(dba_list)
  1142. {
  1143. zend_ulong numitems, i;
  1144. zend_resource *le;
  1145. dba_info *info;
  1146. if (zend_parse_parameters_none() == FAILURE) {
  1147. RETURN_FALSE;
  1148. }
  1149. array_init(return_value);
  1150. numitems = zend_hash_next_free_element(&EG(regular_list));
  1151. for (i=1; i<numitems; i++) {
  1152. if ((le = zend_hash_index_find_ptr(&EG(regular_list), i)) == NULL) {
  1153. continue;
  1154. }
  1155. if (le->type == le_db || le->type == le_pdb) {
  1156. info = (dba_info *)(le->ptr);
  1157. add_index_string(return_value, i, info->path);
  1158. }
  1159. }
  1160. }
  1161. /* }}} */
  1162. #endif /* HAVE_DBA */
  1163. /*
  1164. * Local variables:
  1165. * tab-width: 4
  1166. * c-basic-offset: 4
  1167. * End:
  1168. * vim600: sw=4 ts=4 fdm=marker
  1169. * vim<600: sw=4 ts=4
  1170. */