instance.cc 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  1. /* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
  2. *
  3. * Gearmand client and server library.
  4. *
  5. * Copyright (C) 2012 Data Differential, http://datadifferential.com/
  6. * All rights reserved.
  7. *
  8. * Redistribution and use in source and binary forms, with or without
  9. * modification, are permitted provided that the following conditions are
  10. * met:
  11. *
  12. * * Redistributions of source code must retain the above copyright
  13. * notice, this list of conditions and the following disclaimer.
  14. *
  15. * * Redistributions in binary form must reproduce the above
  16. * copyright notice, this list of conditions and the following disclaimer
  17. * in the documentation and/or other materials provided with the
  18. * distribution.
  19. *
  20. * * The names of its contributors may not be used to endorse or
  21. * promote products derived from this software without specific prior
  22. * written permission.
  23. *
  24. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  25. * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  26. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  27. * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  28. * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  29. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  30. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  31. * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  32. * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  33. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  34. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  35. *
  36. */
  37. #include <gear_config.h>
  38. #include <libgearman-server/common.h>
  39. #include "libgearman-server/plugins/base.h"
  40. #include "libgearman-server/plugins/queue/sqlite/instance.hpp"
  41. namespace gearmand {
  42. namespace queue {
  43. Instance::Instance(const std::string& schema_, const std::string& table_):
  44. _epoch_support(true),
  45. _check_replay(false),
  46. _in_trans(0),
  47. _db(NULL),
  48. delete_sth(NULL),
  49. insert_sth(NULL),
  50. replay_sth(NULL),
  51. _schema(schema_),
  52. _table(table_)
  53. {
  54. _delete_query+= "DELETE FROM ";
  55. _delete_query+= _table;
  56. _delete_query+= " WHERE unique_key=? and function_name=?";
  57. if (_epoch_support)
  58. {
  59. _insert_query+= "INSERT OR REPLACE INTO ";
  60. _insert_query+= _table;
  61. _insert_query+= " (priority, unique_key, function_name, data, when_to_run) VALUES (?,?,?,?,?)";
  62. }
  63. else
  64. {
  65. _insert_query+= "INSERT OR REPLACE INTO ";
  66. _insert_query+= _table;
  67. _insert_query+= " (priority, unique_key, function_name, data) VALUES (?,?,?,?,?)";
  68. }
  69. }
  70. Instance::~Instance()
  71. {
  72. _sqlite3_finalize(delete_sth);
  73. delete_sth= NULL;
  74. _sqlite3_finalize(insert_sth);
  75. insert_sth= NULL;
  76. _sqlite3_finalize(replay_sth);
  77. replay_sth= NULL;
  78. assert(_db);
  79. if (_db)
  80. {
  81. if (sqlite3_close(_db) != SQLITE_OK)
  82. {
  83. gearmand_error(sqlite3_errmsg(_db));
  84. }
  85. _db= NULL;
  86. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM, "sqlite shutdown database");
  87. }
  88. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM, "sqlite shutdown");
  89. }
  90. void Instance::_sqlite3_finalize(sqlite3_stmt* sth)
  91. {
  92. if (sth)
  93. {
  94. if (sqlite3_finalize(sth) != SQLITE_OK )
  95. {
  96. gearmand_log_error(GEARMAN_DEFAULT_LOG_PARAM, "finalize error: %s", sqlite3_errmsg(_db));
  97. }
  98. }
  99. }
  100. bool Instance::_sqlite_prepare(const char *query, size_t query_size, sqlite3_stmt ** sth)
  101. {
  102. reset_error();
  103. if (query_size > UINT32_MAX)
  104. {
  105. _error_string= "query size too big";
  106. return false;
  107. }
  108. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM, "sqlite query: %s", query);
  109. if (sqlite3_prepare_v2(_db, query, int(query_size), sth, NULL) != SQLITE_OK)
  110. {
  111. _error_string= sqlite3_errmsg(_db);
  112. return false;
  113. }
  114. return true;
  115. }
  116. bool Instance::_sqlite_lock()
  117. {
  118. if (_in_trans)
  119. {
  120. /* already in transaction */
  121. return true;
  122. }
  123. if (_sqlite_dispatch("BEGIN TRANSACTION") == false)
  124. {
  125. return false;
  126. }
  127. _in_trans++;
  128. return true;
  129. }
  130. static int sql_count(void * rows_, int argc, char **, char **)
  131. {
  132. int *rows= (int*)rows_;
  133. assert(argc == 1);
  134. (void)argc;
  135. assert(rows);
  136. *rows= *rows +1;
  137. return 0;
  138. }
  139. bool Instance::_sqlite_dispatch(const char* arg)
  140. {
  141. int count;
  142. return _sqlite_count(arg, count);
  143. }
  144. bool Instance::_sqlite_count(const char* arg, int& count)
  145. {
  146. reset_error();
  147. count= 0;
  148. char* error= NULL;
  149. int errcode= sqlite3_exec(_db, arg, sql_count, &count, &error);
  150. if (error != NULL or errcode != SQLITE_OK)
  151. {
  152. assert(errcode != SQLITE_OK);
  153. _error_string= error;
  154. sqlite3_free(error);
  155. return false;
  156. }
  157. return true;
  158. }
  159. bool Instance::_sqlite_count(const std::string& arg, int& count)
  160. {
  161. reset_error();
  162. count= 0;
  163. char* error= NULL;
  164. int errcode= sqlite3_exec(_db, arg.c_str(), sql_count, &count, &error);
  165. if (error != NULL or errcode != SQLITE_OK)
  166. {
  167. assert(errcode != SQLITE_OK);
  168. _error_string= error;
  169. sqlite3_free(error);
  170. return false;
  171. }
  172. return true;
  173. }
  174. bool Instance::_sqlite_dispatch(const std::string& arg)
  175. {
  176. int count;
  177. return _sqlite_count(arg, count);
  178. }
  179. bool Instance::_sqlite_commit()
  180. {
  181. if (_in_trans == 0)
  182. {
  183. /* not in transaction */
  184. return true;
  185. }
  186. if (_sqlite_dispatch("COMMIT") == false)
  187. {
  188. return false;
  189. }
  190. _in_trans= 0;
  191. return true;
  192. }
  193. gearmand_error_t Instance::init()
  194. {
  195. gearmand_info("Initializing libsqlite3 module");
  196. if (_schema.empty())
  197. {
  198. return gearmand_gerror("missing required --libsqlite3-db=<dbfile> argument", GEARMAN_QUEUE_ERROR);
  199. }
  200. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM, "sqlite open: %s", _schema.c_str());
  201. assert(_db == NULL);
  202. if (sqlite3_open_v2(_schema.c_str(), &_db, SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE, NULL) != SQLITE_OK)
  203. {
  204. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR, "sqlite3_open failed with: %s", sqlite3_errmsg(_db));
  205. }
  206. if (_db == NULL)
  207. {
  208. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR, "Unknown error while opening up sqlite file");
  209. }
  210. int rows;
  211. std::string check_table_str("SELECT 1 FROM sqlite_master WHERE type='table' AND name='");
  212. check_table_str+= _table;
  213. check_table_str+= "'";
  214. if (_sqlite_count(check_table_str, rows) == false)
  215. {
  216. return gearmand_gerror(_error_string.c_str(), GEARMAN_QUEUE_ERROR);
  217. }
  218. if (rows)
  219. {
  220. std::string query("SELECT when_to_run FROM ");
  221. query+= _table;
  222. sqlite3_stmt *select_sth;
  223. if (_sqlite_prepare(query.c_str(), _delete_query.size(), &select_sth) == false)
  224. {
  225. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM,
  226. "Error from '%s': %s",
  227. query.c_str(),
  228. _error_string.c_str());
  229. query.clear();
  230. query+= "ALTER TABLE ";
  231. query+= _table;
  232. query+= " ADD COLUMN when_to_run INTEGER";
  233. if (_sqlite_dispatch(query) == false)
  234. {
  235. gearmand_log_error(GEARMAN_DEFAULT_LOG_PARAM,
  236. "Error from '%s': %s",
  237. query.c_str(),
  238. _error_string.c_str());
  239. gearmand_info("No epoch support in sqlite queue");
  240. _epoch_support= false;
  241. }
  242. }
  243. _sqlite3_finalize(select_sth);
  244. }
  245. else
  246. {
  247. std::string query("CREATE TABLE ");
  248. query+= _table;
  249. query+= " ( unique_key TEXT, function_name TEXT, priority INTEGER, data BLOB, when_to_run INTEGER, PRIMARY KEY (unique_key, function_name))";
  250. gearmand_log_info(GEARMAN_DEFAULT_LOG_PARAM, "sqlite module creating table '%s'", _table.c_str());
  251. if (_sqlite_dispatch(query) == false)
  252. {
  253. return gearmand_gerror(_error_string.c_str(), GEARMAN_QUEUE_ERROR);
  254. }
  255. }
  256. if (_sqlite_prepare(_delete_query.c_str(), _delete_query.size(), &delete_sth) == false)
  257. {
  258. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  259. "DELETE PREPARE error: %s",
  260. _error_string.c_str());
  261. }
  262. if (_sqlite_prepare(_insert_query.c_str(), _insert_query.size(), &insert_sth) == false)
  263. {
  264. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  265. "INSERT PREPARE: %s", _error_string.c_str());
  266. }
  267. {
  268. std::string query;
  269. if (_epoch_support)
  270. {
  271. query+= "SELECT unique_key,function_name,priority,data,when_to_run FROM ";
  272. }
  273. else
  274. {
  275. query+= "SELECT unique_key,function_name,priority,data FROM ";
  276. }
  277. query+= _table;
  278. if (_sqlite_prepare(query.c_str(), query.size(), &replay_sth) == false)
  279. {
  280. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  281. "REPLAY PREPARE: %s", _error_string.c_str());
  282. }
  283. }
  284. return GEARMAN_SUCCESS;
  285. }
  286. bool Instance::_sqlite_rollback()
  287. {
  288. if (_in_trans == 0)
  289. {
  290. /* not in transaction */
  291. return true;
  292. }
  293. if (_sqlite_dispatch("ROLLBACK") == false)
  294. {
  295. return false;
  296. }
  297. _in_trans= 0;
  298. return true;
  299. }
  300. gearmand_error_t Instance::add(gearman_server_st*,
  301. const char *unique, size_t unique_size,
  302. const char *function_name,
  303. size_t function_name_size,
  304. const void *data, size_t data_size,
  305. gearman_job_priority_t priority,
  306. int64_t when)
  307. {
  308. assert(_check_replay == false);
  309. if (when and _epoch_support == false)
  310. {
  311. return gearmand_gerror("Table lacks when_to_run field", GEARMAN_QUEUE_ERROR);
  312. }
  313. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM, "sqlite add: %.*s %.*s at %ld",
  314. uint32_t(unique_size), (char *)unique,
  315. uint32_t(function_name_size), (char *)function_name,
  316. (long int)when);
  317. if (_sqlite_lock() == false)
  318. {
  319. return gearmand_gerror(_error_string.c_str(), GEARMAN_QUEUE_ERROR);
  320. }
  321. if (sqlite3_reset(insert_sth) != SQLITE_OK)
  322. {
  323. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  324. "failed to reset INSERT prep statement: %s", sqlite3_errmsg(_db));
  325. }
  326. if (sqlite3_bind_int(insert_sth, 1, priority) != SQLITE_OK)
  327. {
  328. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  329. "failed to bind priority [%d]: %s", priority, sqlite3_errmsg(_db));
  330. }
  331. if (sqlite3_bind_text(insert_sth, 2, (const char *)unique, (int)unique_size, SQLITE_TRANSIENT) != SQLITE_OK)
  332. {
  333. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  334. "failed to bind unique [%.*s]: %s", (uint32_t)unique_size, (char*)unique, sqlite3_errmsg(_db));
  335. }
  336. if (sqlite3_bind_text(insert_sth, 3, (const char *)function_name, (int)function_name_size, SQLITE_TRANSIENT) != SQLITE_OK)
  337. {
  338. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  339. "failed to bind function [%.*s]: %s", (uint32_t)function_name_size, (char*)function_name, sqlite3_errmsg(_db));
  340. }
  341. if (sqlite3_bind_blob(insert_sth, 4, data, (int)data_size, SQLITE_TRANSIENT) != SQLITE_OK)
  342. {
  343. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  344. "failed to bind data blob: %s", sqlite3_errmsg(_db));
  345. }
  346. // epoch data
  347. if (sqlite3_bind_int64(insert_sth, 5, when) != SQLITE_OK)
  348. {
  349. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  350. "failed to bind epoch int64_t(%ld): %s", (long int)when, sqlite3_errmsg(_db));
  351. }
  352. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM,
  353. "sqlite data: priority: %d, unique_key: %s, function_name: %s",
  354. priority, (char*)unique, (char*)function_name);
  355. // INSERT happens here
  356. if (sqlite3_step(insert_sth) != SQLITE_DONE)
  357. {
  358. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  359. "INSERT error: %s", sqlite3_errmsg(_db));
  360. }
  361. return GEARMAN_SUCCESS;
  362. }
  363. gearmand_error_t Instance::flush(gearman_server_st*)
  364. {
  365. gearmand_debug("sqlite flush");
  366. if (_sqlite_commit() == false)
  367. {
  368. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  369. "COMMIT called on FLUSH error: %s",
  370. _error_string.c_str());
  371. }
  372. return GEARMAN_SUCCESS;
  373. }
  374. gearmand_error_t Instance::done(gearman_server_st*,
  375. const char *unique,
  376. size_t unique_size,
  377. const char *function_name,
  378. size_t function_name_size)
  379. {
  380. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM, "sqlite done: %.*s", uint32_t(unique_size), (char *)unique);
  381. if (_sqlite_lock() == false)
  382. {
  383. return gearmand_gerror(_error_string.c_str(), GEARMAN_QUEUE_ERROR);
  384. }
  385. if (sqlite3_reset(delete_sth) != SQLITE_OK)
  386. {
  387. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  388. "failed to reset DELETE prep statement: %s", sqlite3_errmsg(_db));
  389. }
  390. if (sqlite3_bind_text(delete_sth, 1, (const char *)unique, int(unique_size), SQLITE_TRANSIENT) != SQLITE_OK)
  391. {
  392. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  393. "failed to bind unique [%.*s]: %s", uint32_t(unique_size), (char*)unique, sqlite3_errmsg(_db));
  394. }
  395. if (sqlite3_bind_text(delete_sth, 2, (const char *)function_name, int(function_name_size), SQLITE_TRANSIENT) != SQLITE_OK)
  396. {
  397. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  398. "failed to bind function [%.*s]: %s", uint32_t(function_name_size), (char*)function_name, sqlite3_errmsg(_db));
  399. }
  400. // DELETE happens here
  401. if (sqlite3_step(delete_sth) != SQLITE_DONE)
  402. {
  403. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  404. "DELETE error: %s",
  405. sqlite3_errmsg(_db));
  406. }
  407. if (_sqlite_commit() == false)
  408. {
  409. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR, "DELETE error: %s", _error_string.c_str());
  410. }
  411. return GEARMAN_SUCCESS;
  412. }
  413. gearmand_error_t Instance::replay(gearman_server_st *server)
  414. {
  415. gearmand_error_t ret;
  416. _check_replay= true;
  417. if (gearmand_failed(ret= replay_loop(server)))
  418. {
  419. if (_sqlite_rollback() == false)
  420. {
  421. gearmand_log_error(GEARMAN_DEFAULT_LOG_PARAM, "failed to rollback sqlite from failed replay error: %s", _error_string.c_str());
  422. }
  423. }
  424. _check_replay= false;
  425. return ret;
  426. }
  427. gearmand_error_t Instance::replay_loop(gearman_server_st *server)
  428. {
  429. gearmand_info("sqlite replay start");
  430. gearmand_error_t gret;
  431. while (sqlite3_step(replay_sth) == SQLITE_ROW)
  432. {
  433. const char *unique, *function_name;
  434. size_t unique_size, function_name_size;
  435. if (sqlite3_column_type(replay_sth, 0) == SQLITE_TEXT)
  436. {
  437. unique= (char *)sqlite3_column_text(replay_sth, 0);
  438. unique_size= size_t(sqlite3_column_bytes(replay_sth, 0));
  439. }
  440. else
  441. {
  442. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR, "column %d is not type TEXT: %d", 0, int(sqlite3_column_type(replay_sth, 0)));
  443. }
  444. if (sqlite3_column_type(replay_sth, 1) == SQLITE_TEXT)
  445. {
  446. function_name= (char *)sqlite3_column_text(replay_sth, 1);
  447. function_name_size= size_t(sqlite3_column_bytes(replay_sth, 1));
  448. }
  449. else
  450. {
  451. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  452. "column %d is not type TEXT", 1);
  453. }
  454. gearman_job_priority_t priority;
  455. if (sqlite3_column_type(replay_sth, 2) == SQLITE_INTEGER)
  456. {
  457. priority= (gearman_job_priority_t)sqlite3_column_int64(replay_sth, 2);
  458. }
  459. else
  460. {
  461. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  462. "column %d is not type INTEGER", 2);
  463. }
  464. if (sqlite3_column_type(replay_sth, 3) != SQLITE_BLOB)
  465. {
  466. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR, "column %d is not type TEXT", 3);
  467. }
  468. size_t data_size= (size_t)sqlite3_column_bytes(replay_sth, 3);
  469. char* data= (char*)malloc(data_size);
  470. /* need to make a copy here ... gearman_server_job_free will free it later */
  471. if (data == NULL)
  472. {
  473. return gearmand_perror("malloc");
  474. }
  475. memcpy(data, sqlite3_column_blob(replay_sth, 3), data_size);
  476. int64_t when;
  477. if (_epoch_support)
  478. {
  479. if (sqlite3_column_type(replay_sth, 4) == SQLITE_INTEGER)
  480. {
  481. when= int64_t(sqlite3_column_int64(replay_sth, 4));
  482. }
  483. else
  484. {
  485. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR, "column %d is not type INTEGER", 3);
  486. }
  487. }
  488. else
  489. {
  490. when= 0;
  491. }
  492. gearmand_log_debug(GEARMAN_DEFAULT_LOG_PARAM, "sqlite replay: %s %s", (char*)unique, (char*)function_name);
  493. gret= Instance::replay_add(server,
  494. NULL,
  495. unique, unique_size,
  496. function_name, function_name_size,
  497. data, data_size,
  498. priority, when);
  499. if (gearmand_failed(gret))
  500. {
  501. break;
  502. }
  503. }
  504. if (sqlite3_reset(replay_sth) != SQLITE_OK)
  505. {
  506. return gearmand_log_gerror(GEARMAN_DEFAULT_LOG_PARAM, GEARMAN_QUEUE_ERROR,
  507. "failed to reset REPLAY prep statement: %s", sqlite3_errmsg(_db));
  508. }
  509. return gret;
  510. }
  511. } // namespace queue
  512. } // namespace gearmand