xapian-core  1.4.26
remoteserver.cc
Go to the documentation of this file.
1 
4 /* Copyright (C) 2006-2023 Olly Betts
5  * Copyright (C) 2006,2007,2009,2010 Lemur Consulting Ltd
6  *
7  * This program is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 2 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20  */
21 
22 #include <config.h>
23 #include "remoteserver.h"
24 
25 #include "xapian/constants.h"
26 #include "xapian/database.h"
27 #include "xapian/enquire.h"
28 #include "xapian/error.h"
29 #include "xapian/matchspy.h"
30 #include "xapian/query.h"
31 #include "xapian/valueiterator.h"
32 
33 #include <signal.h>
34 #include <cerrno>
35 #include <cstdlib>
36 
37 #include "autoptr.h"
38 #include "length.h"
39 #include "matcher/multimatch.h"
40 #include "noreturn.h"
41 #include "omassert.h"
42 #include "realtime.h"
43 #include "serialise.h"
44 #include "serialise-double.h"
45 #include "serialise-error.h"
46 #include "str.h"
47 #include "stringutils.h"
48 #include "weight/weightinternal.h"
49 
50 XAPIAN_NORETURN(static void throw_read_only());
51 static void
53 {
54  throw Xapian::InvalidOperationError("Server is read-only");
55 }
56 
58 struct ConnectionClosed { };
59 
60 RemoteServer::RemoteServer(const std::vector<std::string> &dbpaths,
61  int fdin_, int fdout_,
62  double active_timeout_, double idle_timeout_,
63  bool writable_)
64  : RemoteConnection(fdin_, fdout_, std::string()),
65  db(NULL), wdb(NULL), writable(writable_),
66  active_timeout(active_timeout_), idle_timeout(idle_timeout_)
67 {
68  // Catch errors opening the database and propagate them to the client.
69  try {
70  Assert(!dbpaths.empty());
71  // We always open the database read-only to start with. If we're
72  // writable, the client can ask to be upgraded to write access once
73  // connected if it wants it.
74  db = new Xapian::Database(dbpaths[0]);
75  // Build a better description than Database::get_description() gives
76  // in the variable context. FIXME: improve Database::get_description()
77  // and then just use that instead.
78  context = dbpaths[0];
79 
80  vector<std::string>::const_iterator i(dbpaths.begin());
81  for (++i; i != dbpaths.end(); ++i) {
82  db->add_database(Xapian::Database(*i));
83  context += ' ';
84  context += *i;
85  }
86  } catch (const Xapian::Error &err) {
87  // Propagate the exception to the client.
89  // And rethrow it so our caller can log it and close the connection.
90  throw;
91  }
92 
93 #ifndef __WIN32__
94  // It's simplest to just ignore SIGPIPE. We'll still know if the
95  // connection dies because we'll get EPIPE back from write().
96  //
97  // This is OK because RemoteServer subclasses are only used in
98  // specialised programs - if we expose any of them as API classes
99  // then we should use SO_NOSIGPIPE/MSG_NOSIGNAL instead like we do
100  // on the client side.
101  if (signal(SIGPIPE, SIG_IGN) == SIG_ERR)
102  throw Xapian::NetworkError("Couldn't set SIGPIPE to SIG_IGN", errno);
103 #endif
104 
105  // Send greeting message.
106  msg_update(string());
107 }
108 
110 {
111  delete db;
112  // wdb is either NULL or equal to db, so we shouldn't delete it too!
113 }
114 
116 RemoteServer::get_message(double timeout, string & result,
117  message_type required_type)
118 {
119  double end_time = RealTime::end_time(timeout);
120  int type = RemoteConnection::get_message(result, end_time);
121 
122  // Handle "shutdown connection" message here. Treat EOF here for a read-only
123  // database the same way since a read-only client just closes the
124  // connection when done.
125  if (type == MSG_SHUTDOWN || (type < 0 && wdb == NULL))
126  throw ConnectionClosed();
127  if (type < 0)
128  throw Xapian::NetworkError("Connection closed unexpectedly");
129  if (type >= MSG_MAX) {
130  string errmsg("Invalid message type ");
131  errmsg += str(type);
132  throw Xapian::NetworkError(errmsg);
133  }
134  if (required_type != MSG_MAX && type != int(required_type)) {
135  string errmsg("Expecting message type ");
136  errmsg += str(int(required_type));
137  errmsg += ", got ";
138  errmsg += str(type);
139  throw Xapian::NetworkError(errmsg);
140  }
141  return static_cast<message_type>(type);
142 }
143 
144 void
145 RemoteServer::send_message(reply_type type, const string &message)
146 {
148  unsigned char type_as_char = static_cast<unsigned char>(type);
149  RemoteConnection::send_message(type_as_char, message, end_time);
150 }
151 
152 typedef void (RemoteServer::* dispatch_func)(const string &);
153 
154 void
156 {
157  while (true) {
158  try {
159  /* This list needs to be kept in the same order as the list of
160  * message types in "remoteprotocol.h". Note that messages at the
161  * end of the list in "remoteprotocol.h" can be omitted if they
162  * don't correspond to dispatch actions.
163  */
164  static const dispatch_func dispatch[] = {
191  0, // MSG_GETMSET - used during a conversation.
192  0, // MSG_SHUTDOWN - handled by get_message().
201  };
202 
203  string message;
204  size_t type = get_message(idle_timeout, message);
205  if (type >= sizeof(dispatch) / sizeof(dispatch[0]) || !dispatch[type]) {
206  string errmsg("Unexpected message type ");
207  errmsg += str(type);
208  throw Xapian::InvalidArgumentError(errmsg);
209  }
210  (this->*(dispatch[type]))(message);
211  } catch (const Xapian::NetworkTimeoutError & e) {
212  try {
213  // We've had a timeout, so the client may not be listening, so
214  // set the end_time to 1 and if we can't send the message right
215  // away, just exit and the client will cope.
217  } catch (...) {
218  }
219  // And rethrow it so our caller can log it and close the
220  // connection.
221  throw;
222  } catch (const Xapian::NetworkError &) {
223  // All other network errors mean we are fatally confused and are
224  // unlikely to be able to communicate further across this
225  // connection. So we don't try to propagate the error to the
226  // client, but instead just rethrow the exception so our caller can
227  // log it and close the connection.
228  throw;
229  } catch (const Xapian::Error &e) {
230  // Propagate the exception to the client, then return to the main
231  // message handling loop.
233  } catch (ConnectionClosed &) {
234  return;
235  } catch (...) {
236  // Propagate an unknown exception to the client.
237  send_message(REPLY_EXCEPTION, string());
238  // And rethrow it so our caller can log it and close the
239  // connection.
240  throw;
241  }
242  }
243 }
244 
245 void
246 RemoteServer::msg_allterms(const string &message)
247 {
248  string prev = message;
249  string reply;
250 
251  const string & prefix = message;
252  const Xapian::TermIterator end = db->allterms_end(prefix);
253  for (Xapian::TermIterator t = db->allterms_begin(prefix); t != end; ++t) {
254  const string & v = *t;
255  size_t reuse = common_prefix_length(prev, v, 255);
256  reply = encode_length(t.get_termfreq());
257  reply.append(1, char(reuse));
258  reply.append(v, reuse, string::npos);
260  prev = v;
261  }
262 
263  send_message(REPLY_DONE, string());
264 }
265 
266 void
267 RemoteServer::msg_termlist(const string &message)
268 {
269  const char *p = message.data();
270  const char *p_end = p + message.size();
271  Xapian::docid did;
272  decode_length(&p, p_end, did);
273 
275  string prev;
276  const Xapian::TermIterator end = db->termlist_end(did);
277  for (Xapian::TermIterator t = db->termlist_begin(did); t != end; ++t) {
278  const string & v = *t;
279  size_t reuse = common_prefix_length(prev, v, 255);
280  string reply = encode_length(t.get_wdf());
281  reply += encode_length(t.get_termfreq());
282  reply.append(1, char(reuse));
283  reply.append(v, reuse, string::npos);
285  prev = v;
286  }
287 
288  send_message(REPLY_DONE, string());
289 }
290 
291 void
292 RemoteServer::msg_positionlist(const string &message)
293 {
294  const char *p = message.data();
295  const char *p_end = p + message.size();
296  Xapian::docid did;
297  decode_length(&p, p_end, did);
298  string term(p, p_end - p);
299 
300  Xapian::termpos lastpos = static_cast<Xapian::termpos>(-1);
301  const Xapian::PositionIterator end = db->positionlist_end(did, term);
302  for (Xapian::PositionIterator i = db->positionlist_begin(did, term);
303  i != end; ++i) {
304  Xapian::termpos pos = *i;
305  send_message(REPLY_POSITIONLIST, encode_length(pos - lastpos - 1));
306  lastpos = pos;
307  }
308 
309  send_message(REPLY_DONE, string());
310 }
311 
312 void
313 RemoteServer::msg_postlist(const string &message)
314 {
315  const string & term = message;
316 
317  Xapian::doccount termfreq = db->get_termfreq(term);
318  Xapian::termcount collfreq = db->get_collection_freq(term);
320 
321  Xapian::docid lastdocid = 0;
322  const Xapian::PostingIterator end = db->postlist_end(term);
323  for (Xapian::PostingIterator i = db->postlist_begin(term);
324  i != end; ++i) {
325 
326  Xapian::docid newdocid = *i;
327  string reply = encode_length(newdocid - lastdocid - 1);
328  reply += encode_length(i.get_wdf());
329 
331  lastdocid = newdocid;
332  }
333 
334  send_message(REPLY_DONE, string());
335 }
336 
337 void
338 RemoteServer::msg_writeaccess(const string & msg)
339 {
340  if (!writable)
341  throw_read_only();
342 
343  int flags = Xapian::DB_OPEN;
344  const char *p = msg.c_str();
345  const char *p_end = p + msg.size();
346  if (p != p_end) {
347  unsigned flag_bits;
348  decode_length(&p, p_end, flag_bits);
349  flags |= flag_bits &~ Xapian::DB_ACTION_MASK_;
350  if (p != p_end) {
351  throw Xapian::NetworkError("Junk at end of MSG_WRITEACCESS");
352  }
353  }
354 
355  wdb = new Xapian::WritableDatabase(context, flags);
356  delete db;
357  db = wdb;
358  msg_update(msg);
359 }
360 
361 void
362 RemoteServer::msg_reopen(const string & msg)
363 {
364  if (!db->reopen()) {
365  send_message(REPLY_DONE, string());
366  return;
367  }
368  msg_update(msg);
369 }
370 
371 void
373 {
374  static const char protocol[2] = {
377  };
378  string message(protocol, 2);
379  Xapian::doccount num_docs = db->get_doccount();
380  message += encode_length(num_docs);
381  message += encode_length(db->get_lastdocid() - num_docs);
383  message += encode_length(doclen_lb);
384  message += encode_length(db->get_doclength_upper_bound() - doclen_lb);
385  message += (db->has_positions() ? '1' : '0');
386  message += encode_length(db->get_total_length());
387  string uuid = db->get_uuid();
388  message += uuid;
389  send_message(REPLY_UPDATE, message);
390 }
391 
392 void
393 RemoteServer::msg_query(const string &message_in)
394 {
395  const char *p = message_in.c_str();
396  const char *p_end = p + message_in.size();
397 
398  // Unserialise the Query.
399  size_t len;
400  decode_length_and_check(&p, p_end, len);
402  p += len;
403 
404  // Unserialise assorted Enquire settings.
405  Xapian::termcount qlen;
406  decode_length(&p, p_end, qlen);
407 
408  Xapian::valueno collapse_max;
409  decode_length(&p, p_end, collapse_max);
410 
411  Xapian::valueno collapse_key = Xapian::BAD_VALUENO;
412  if (collapse_max)
413  decode_length(&p, p_end, collapse_key);
414 
415  if (p_end - p < 4 || *p < '0' || *p > '2') {
416  throw Xapian::NetworkError("bad message (docid_order)");
417  }
419  order = static_cast<Xapian::Enquire::docid_order>(*p++ - '0');
420 
421  Xapian::valueno sort_key;
422  decode_length(&p, p_end, sort_key);
423 
424  if (*p < '0' || *p > '3') {
425  throw Xapian::NetworkError("bad message (sort_by)");
426  }
428  sort_by = static_cast<Xapian::Enquire::Internal::sort_setting>(*p++ - '0');
429 
430  if (*p < '0' || *p > '1') {
431  throw Xapian::NetworkError("bad message (sort_value_forward)");
432  }
433  bool sort_value_forward(*p++ != '0');
434 
435  double time_limit = unserialise_double(&p, p_end);
436 
437  int percent_cutoff = *p++;
438  if (percent_cutoff < 0 || percent_cutoff > 100) {
439  throw Xapian::NetworkError("bad message (percent_cutoff)");
440  }
441 
442  double weight_cutoff = unserialise_double(&p, p_end);
443  if (weight_cutoff < 0) {
444  throw Xapian::NetworkError("bad message (weight_cutoff)");
445  }
446 
447  // Unserialise the Weight object.
448  decode_length_and_check(&p, p_end, len);
449  string wtname(p, len);
450  p += len;
451 
452  const Xapian::Weight * wttype = reg.get_weighting_scheme(wtname);
453  if (wttype == NULL) {
454  // Note: user weighting schemes should be registered by adding them to
455  // a Registry, and setting the context using
456  // RemoteServer::set_registry().
457  throw Xapian::InvalidArgumentError("Weighting scheme " +
458  wtname + " not registered");
459  }
460 
461  decode_length_and_check(&p, p_end, len);
462  AutoPtr<Xapian::Weight> wt(wttype->unserialise(string(p, len)));
463  p += len;
464 
465  // Unserialise the RSet object.
466  decode_length_and_check(&p, p_end, len);
467  Xapian::RSet rset = unserialise_rset(string(p, len));
468  p += len;
469 
470  // Unserialise any MatchSpy objects.
471  vector<Xapian::Internal::opt_intrusive_ptr<Xapian::MatchSpy>> matchspies;
472  while (p != p_end) {
473  decode_length_and_check(&p, p_end, len);
474  string spytype(p, len);
475  const Xapian::MatchSpy * spyclass = reg.get_match_spy(spytype);
476  if (spyclass == NULL) {
477  throw Xapian::InvalidArgumentError("Match spy " + spytype +
478  " not registered");
479  }
480  p += len;
481 
482  decode_length_and_check(&p, p_end, len);
483  matchspies.push_back(spyclass->unserialise(string(p, len), reg)->release());
484  p += len;
485  }
486 
487  Xapian::Weight::Internal local_stats;
488  MultiMatch match(*db, query, qlen, &rset, collapse_max, collapse_key,
489  percent_cutoff, weight_cutoff, order,
490  sort_key, sort_by, sort_value_forward, time_limit,
491  local_stats, wt.get(), matchspies, false, false);
492 
494 
495  string message;
497  p = message.c_str();
498  p_end = p + message.size();
499 
500  Xapian::termcount first;
501  decode_length(&p, p_end, first);
502  Xapian::termcount maxitems;
503  decode_length(&p, p_end, maxitems);
504 
505  Xapian::termcount check_at_least;
506  decode_length(&p, p_end, check_at_least);
507 
508  AutoPtr<Xapian::Weight::Internal> total_stats(new Xapian::Weight::Internal);
509  unserialise_stats(p, p_end, *(total_stats.get()));
510  total_stats->set_bounds_from_db(*db);
511 
512  Xapian::MSet mset;
513  match.get_mset(first, maxitems, check_at_least, mset, *(total_stats.get()), 0, 0);
514  mset.internal->stats = total_stats.release();
515 
516  message.resize(0);
517  for (auto i : matchspies) {
518  string spy_results = i->serialise_results();
519  message += encode_length(spy_results.size());
520  message += spy_results;
521  }
522  message += serialise_mset(mset);
523  send_message(REPLY_RESULTS, message);
524 }
525 
526 void
527 RemoteServer::msg_document(const string &message)
528 {
529  const char *p = message.data();
530  const char *p_end = p + message.size();
531  Xapian::docid did;
532  decode_length(&p, p_end, did);
533 
534  Xapian::Document doc = db->get_document(did);
535 
537 
539  for (i = doc.values_begin(); i != doc.values_end(); ++i) {
540  string item = encode_length(i.get_valueno());
541  item += *i;
542  send_message(REPLY_VALUE, item);
543  }
544  send_message(REPLY_DONE, string());
545 }
546 
547 void
549 {
550  // Ensure *our* database stays alive, as it may contain remote databases!
551  db->keep_alive();
552  send_message(REPLY_DONE, string());
553 }
554 
555 void
556 RemoteServer::msg_termexists(const string &term)
557 {
559 }
560 
561 void
562 RemoteServer::msg_collfreq(const string &term)
563 {
565 }
566 
567 void
568 RemoteServer::msg_termfreq(const string &term)
569 {
571 }
572 
573 void
574 RemoteServer::msg_freqs(const string &term)
575 {
576  string msg = encode_length(db->get_termfreq(term));
577  msg += encode_length(db->get_collection_freq(term));
579 }
580 
581 void
582 RemoteServer::msg_valuestats(const string & message)
583 {
584  const char *p = message.data();
585  const char *p_end = p + message.size();
586  while (p != p_end) {
587  Xapian::valueno slot;
588  decode_length(&p, p_end, slot);
589  string message_out;
590  message_out += encode_length(db->get_value_freq(slot));
591  string bound = db->get_value_lower_bound(slot);
592  message_out += encode_length(bound.size());
593  message_out += bound;
594  bound = db->get_value_upper_bound(slot);
595  message_out += encode_length(bound.size());
596  message_out += bound;
597 
598  send_message(REPLY_VALUESTATS, message_out);
599  }
600 }
601 
602 void
603 RemoteServer::msg_doclength(const string &message)
604 {
605  const char *p = message.data();
606  const char *p_end = p + message.size();
607  Xapian::docid did;
608  decode_length(&p, p_end, did);
610 }
611 
612 void
613 RemoteServer::msg_uniqueterms(const string &message)
614 {
615  const char *p = message.data();
616  const char *p_end = p + message.size();
617  Xapian::docid did;
618  decode_length(&p, p_end, did);
620 }
621 
622 void
624 {
625  if (!wdb)
626  throw_read_only();
627 
628  wdb->commit();
629 
630  send_message(REPLY_DONE, string());
631 }
632 
633 void
634 RemoteServer::msg_cancel(const string &message)
635 {
636  msg_cancel_(message);
637  send_message(REPLY_DONE, string());
638 }
639 
640 void
642 {
643  if (!wdb)
644  throw_read_only();
645 
646  // We can't call cancel since that's an internal method, but this
647  // has the same effect with minimal additional overhead.
648  wdb->begin_transaction(false);
650 }
651 
652 void
653 RemoteServer::msg_adddocument(const string & message)
654 {
655  if (!wdb)
656  throw_read_only();
657 
659 
661 }
662 
663 void
664 RemoteServer::msg_deletedocument(const string & message)
665 {
666  if (!wdb)
667  throw_read_only();
668 
669  const char *p = message.data();
670  const char *p_end = p + message.size();
671  Xapian::docid did;
672  decode_length(&p, p_end, did);
673 
674  wdb->delete_document(did);
675 
676  send_message(REPLY_DONE, string());
677 }
678 
679 void
681 {
682  msg_deletedocumentterm_(message);
683  send_message(REPLY_DONE, string());
684 }
685 
686 void
688 {
689  if (!wdb)
690  throw_read_only();
691 
692  wdb->delete_document(message);
693 }
694 
695 void
696 RemoteServer::msg_replacedocument(const string & message)
697 {
698  msg_replacedocument_(message);
699  send_message(REPLY_DONE, string());
700 }
701 
702 void
703 RemoteServer::msg_replacedocument_(const string & message)
704 {
705  if (!wdb)
706  throw_read_only();
707 
708  const char *p = message.data();
709  const char *p_end = p + message.size();
710  Xapian::docid did;
711  decode_length(&p, p_end, did);
712 
713  wdb->replace_document(did, unserialise_document(string(p, p_end)));
714 }
715 
716 void
718 {
719  if (!wdb)
720  throw_read_only();
721 
722  const char *p = message.data();
723  const char *p_end = p + message.size();
724  size_t len;
725  decode_length_and_check(&p, p_end, len);
726  string unique_term(p, len);
727  p += len;
728 
729  Xapian::docid did = wdb->replace_document(unique_term, unserialise_document(string(p, p_end)));
730 
732 }
733 
734 void
735 RemoteServer::msg_getmetadata(const string & message)
736 {
738 }
739 
740 void
742 {
743  string prev = message;
744  string reply;
745 
746  const string & prefix = message;
747  const Xapian::TermIterator end = db->metadata_keys_end(prefix);
749  for (; t != end; ++t) {
750  const string & v = *t;
751  size_t reuse = common_prefix_length(prev, v, 255);
752  reply.assign(1, char(reuse));
753  reply.append(v, reuse, string::npos);
755  prev = v;
756  }
757  send_message(REPLY_DONE, string());
758 }
759 
760 void
761 RemoteServer::msg_setmetadata(const string & message)
762 {
763  msg_setmetadata_(message);
764  send_message(REPLY_DONE, string());
765 }
766 
767 void
768 RemoteServer::msg_setmetadata_(const string & message)
769 {
770  if (!wdb)
771  throw_read_only();
772  const char *p = message.data();
773  const char *p_end = p + message.size();
774  size_t keylen;
775  decode_length_and_check(&p, p_end, keylen);
776  string key(p, keylen);
777  p += keylen;
778  string val(p, p_end - p);
779  wdb->set_metadata(key, val);
780 }
781 
782 void
783 RemoteServer::msg_addspelling(const string & message)
784 {
785  msg_addspelling_(message);
786  send_message(REPLY_DONE, string());
787 }
788 
789 void
790 RemoteServer::msg_addspelling_(const string & message)
791 {
792  if (!wdb)
793  throw_read_only();
794  const char *p = message.data();
795  const char *p_end = p + message.size();
796  Xapian::termcount freqinc;
797  decode_length(&p, p_end, freqinc);
798  wdb->add_spelling(string(p, p_end - p), freqinc);
799 }
800 
801 void
802 RemoteServer::msg_removespelling(const string & message)
803 {
804  if (!wdb)
805  throw_read_only();
806  const char *p = message.data();
807  const char *p_end = p + message.size();
808  Xapian::termcount freqdec;
809  decode_length(&p, p_end, freqdec);
810  wdb->remove_spelling(string(p, p_end - p), freqdec);
811 }
Xapian::Document get_document(Xapian::docid did) const
Get a document from the database, given its document id.
Definition: omdatabase.cc:490
Xapian::docid add_document(const Xapian::Document &document)
Add a new document to the database.
Definition: omdatabase.cc:902
#define Assert(COND)
Definition: omassert.h:122
static void throw_read_only()
Definition: remoteserver.cc:52
Define the XAPIAN_NORETURN macro.
PositionIterator positionlist_end(Xapian::docid, const std::string &) const
Corresponding end iterator to positionlist_begin().
Definition: database.h:254
~RemoteServer()
Destructor.
static const Query unserialise(const std::string &serialised, const Registry &reg=Registry())
Unserialise a string and return a Query object.
Definition: query.cc:202
A RemoteConnection object provides a bidirectional connection to another RemoteConnection object on a...
void msg_cancel_(const std::string &message)
void msg_positionlist(const std::string &message)
TermIterator termlist_begin(Xapian::docid did) const
An iterator pointing to the start of the termlist for a given document.
Definition: omdatabase.cc:198
Xapian::RSet unserialise_rset(const string &s)
Unserialise a serialised Xapian::RSet object.
Definition: serialise.cc:232
void cancel_transaction()
Abort the transaction currently in progress, discarding the pending modifications made to the databas...
Definition: omdatabase.cc:890
length encoded as a string
int get_message(std::string &result, double end_time)
Read one message from fdin.
This class is used to access a database, or a group of databases.
Definition: database.h:68
unsigned timeout
A timeout value in milliseconds.
Definition: types.h:100
class for performing a match
Xapian::WritableDatabase * wdb
The WritableDatabase we&#39;re using, or NULL if we&#39;re read-only.
Definition: remoteserver.h:50
void msg_keepalive(const std::string &message)
void msg_cancel(const std::string &message)
void remove_spelling(const std::string &word, Xapian::termcount freqdec=1) const
Remove a word from the spelling dictionary.
Definition: omdatabase.cc:1015
MatchSpy * release()
Start reference counting this object.
Definition: matchspy.h:184
InvalidOperationError indicates the API was used in an invalid way.
Definition: error.h:283
Xapian::Document unserialise_document(const string &s)
Unserialise a serialised Xapian::Document object.
Definition: serialise.cc:295
PositionIterator positionlist_begin(Xapian::docid did, const std::string &tname) const
An iterator pointing to the start of the position list for a given term in a given document...
Definition: omdatabase.cc:250
void msg_reopen(const std::string &message)
Abstract base class for match spies.
Definition: matchspy.h:49
void msg_collfreq(const std::string &message)
RemoteServer(const RemoteServer &)
Don&#39;t allow copying.
Indicates a timeout expired while communicating with a remote database.
Definition: error.h:845
Xapian::termcount get_doclength_lower_bound() const
Get a lower bound on the length of a document in this DB.
Definition: omdatabase.cc:401
bool has_positions() const
Does this database have any positional information?
Definition: omdatabase.cc:238
ValueIterator values_begin() const
Iterator for the values in this document.
Definition: omdocument.cc:210
void send_message(reply_type type, const std::string &message)
Send a message to the client.
TermIterator allterms_end(const std::string &=std::string()) const
Corresponding end iterator to allterms_begin(prefix).
Definition: database.h:269
Xapian::Internal::intrusive_ptr< Internal > internal
Definition: mset.h:52
Constants in the Xapian namespace.
Xapian::docid get_lastdocid() const
Get the highest document id which has been used in the database.
Definition: omdatabase.cc:279
void msg_openmetadatakeylist(const std::string &message)
double end_time(double timeout)
Return the end time for a timeout in timeout seconds.
Definition: realtime.h:95
void msg_replacedocumentterm(const std::string &message)
Xapian::Registry reg
The registry, which allows unserialisation of user subclasses.
Definition: remoteserver.h:70
void begin_transaction(bool flushed=true)
Begin a transaction.
Definition: omdatabase.cc:868
bool reopen()
Re-open the database.
Definition: omdatabase.cc:125
reply_type
Reply types (server -> client).
Class for iterating over document values.
Definition: valueiterator.h:40
Class representing a list of search results.
Definition: mset.h:44
STL namespace.
Convert types to std::string.
void msg_uniqueterms(const std::string &message)
void msg_termlist(const std::string &message)
string serialise_error(const Xapian::Error &e)
Serialise a Xapian::Error object to a string.
double active_timeout
Timeout for actions during a conversation.
Definition: remoteserver.h:60
void msg_replacedocument(const std::string &message)
std::string get_value_upper_bound(Xapian::valueno slot) const
Get an upper bound on the values stored in the given value slot.
Definition: omdatabase.cc:386
void replace_document(Xapian::docid did, const Xapian::Document &document)
Replace a given document in the database.
Definition: omdatabase.cc:952
void set_metadata(const std::string &key, const std::string &metadata)
Set the user-specified metadata associated with a given key.
Definition: omdatabase.cc:1064
std::string encode_length(T len)
Encode a length as a variable-length string.
Definition: length.h:36
Xapian::doccount get_doccount() const
Get the number of documents in the database.
Definition: omdatabase.cc:267
std::string get_metadata(const std::string &key) const
Get the user-specified metadata associated with a given key.
Definition: omdatabase.cc:758
Class to throw when we receive the connection closing message.
Definition: remoteserver.cc:58
Xapian::totallength get_total_length() const
Get the total length of all the documents in the database.
Definition: omdatabase.cc:312
void msg_replacedocument_(const std::string &message)
void msg_adddocument(const std::string &message)
void msg_setmetadata(const std::string &message)
std::string context
The context to report with errors.
Xapian::Query API class.
void msg_removespelling(const std::string &message)
bool writable
Do we support writing?
Definition: remoteserver.h:53
API for running queries.
Hierarchy of classes which Xapian can throw as exceptions.
Class for iterating over a list of terms.
Definition: termiterator.h:41
unsigned XAPIAN_TERMCOUNT_BASE_TYPE termcount
A counts of terms.
Definition: types.h:72
Class for iterating over a list of terms.
functions to serialise and unserialise a double
const Xapian::MatchSpy * get_match_spy(const std::string &name) const
Get a match spy given a name.
Definition: registry.cc:300
InvalidArgumentError indicates an invalid parameter value was passed to the API.
Definition: error.h:241
Xapian::termcount get_doclength_upper_bound() const
Get an upper bound on the length of a document in this DB.
Definition: omdatabase.cc:421
void run()
Repeatedly accept messages from the client and process them.
double unserialise_double(const char **p, const char *end)
Unserialise a double serialised by serialise_double.
void msg_deletedocumentterm(const std::string &message)
void msg_addspelling_(const std::string &message)
const int DB_OPEN
Open an existing database.
Definition: constants.h:50
This class provides read/write access to a database.
Definition: database.h:789
MatchSpy implementation.
Xapian::Weight::Internal class, holding database and term statistics.
void msg_postlist(const std::string &message)
void msg_commit(const std::string &message)
Class to hold statistics for a given collection.
const Xapian::Weight * get_weighting_scheme(const std::string &name) const
Get the weighting scheme given a name.
Definition: registry.cc:272
void delete_document(Xapian::docid did)
Delete a document from the database.
Definition: omdatabase.cc:925
API for working with Xapian databases.
string str(int value)
Convert int to std::string.
Definition: str.cc:90
void(RemoteServer::* dispatch_func)(const string &)
Xapian::termcount get_doclength(Xapian::docid did) const
Get the length of a document.
Definition: omdatabase.cc:461
Class for iterating over document values.
void commit()
Commit any pending modifications made to the database.
Definition: omdatabase.cc:857
Xapian remote backend server base class.
void msg_getmetadata(const std::string &message)
Class for iterating over term positions.
double idle_timeout
Timeout while waiting for a new action from the client.
Definition: remoteserver.h:67
void msg_addspelling(const std::string &message)
void msg_setmetadata_(const std::string &message)
#define XAPIAN_REMOTE_PROTOCOL_MAJOR_VERSION
ValueIterator values_end() const
Equivalent end iterator for values_begin().
Definition: document.h:281
string serialise_stats(const Xapian::Weight::Internal &stats)
Serialise a stats object.
Definition: serialise.cc:42
TermIterator allterms_begin(const std::string &prefix=std::string()) const
An iterator which runs across all terms with a given prefix.
Definition: omdatabase.cc:223
TermIterator termlist_end(Xapian::docid) const
Corresponding end iterator to termlist_begin().
Definition: database.h:240
void msg_valuestats(const std::string &message)
functions to convert classes to strings and back
bool term_exists(const std::string &tname) const
Check if a given term exists in the database.
Definition: omdatabase.cc:524
void msg_allterms(const std::string &message)
void msg_doclength(const std::string &message)
#define XAPIAN_REMOTE_PROTOCOL_MINOR_VERSION
static Xapian::Query query(Xapian::Query::op op, const string &t1=string(), const string &t2=string(), const string &t3=string(), const string &t4=string(), const string &t5=string(), const string &t6=string(), const string &t7=string(), const string &t8=string(), const string &t9=string(), const string &t10=string())
Definition: api_anydb.cc:63
void msg_query(const std::string &message)
void unserialise_stats(const char *p, const char *p_end, Xapian::Weight::Internal &stat)
Unserialise a serialised stats object.
Definition: serialise.cc:71
void decode_length_and_check(const char **p, const char *end, unsigned &out)
Decode a length encoded by encode_length.
Definition: length.cc:112
string serialise_mset(const Xapian::MSet &mset)
Serialise a Xapian::MSet object.
Definition: serialise.cc:113
unsigned XAPIAN_DOCID_BASE_TYPE doccount
A count of documents.
Definition: types.h:38
Xapian::Database * db
The database we&#39;re using.
Definition: remoteserver.h:47
void msg_writeaccess(const std::string &message)
All exceptions thrown by Xapian are subclasses of Xapian::Error.
Definition: error.h:43
Indicates a problem communicating with a remote database.
Definition: error.h:803
std::string get_value_lower_bound(Xapian::valueno slot) const
Get a lower bound on the values stored in the given value slot.
Definition: omdatabase.cc:368
std::string::size_type common_prefix_length(const std::string &a, const std::string &b)
Definition: stringutils.h:123
unsigned valueno
The number for a value slot in a document.
Definition: types.h:108
void msg_deletedocument(const std::string &message)
unsigned XAPIAN_TERMPOS_BASE_TYPE termpos
A term position within a document or query.
Definition: types.h:83
Various handy helpers which std::string really should provide.
void msg_termexists(const std::string &message)
message_type
Message types (client -> server).
void send_message(char type, const std::string &s, double end_time)
Send a message.
Xapian::TermIterator metadata_keys_begin(const std::string &prefix=std::string()) const
An iterator which returns all user-specified metadata keys.
Definition: omdatabase.cc:768
void msg_update(const std::string &message)
void keep_alive()
Send a "keep-alive" to remote databases to stop them timing out.
Definition: omdatabase.cc:538
virtual Weight * unserialise(const std::string &serialised) const
Unserialise parameters.
Definition: weight.cc:147
Remote backend server base class.
Definition: remoteserver.h:36
void msg_freqs(const std::string &message)
Various assertion macros.
Functions for handling a time or time interval in a double.
unsigned XAPIAN_DOCID_BASE_TYPE docid
A unique identifier for a document.
Definition: types.h:52
Class representing a query.
Definition: query.h:46
std::string get_data() const
Get data stored in the document.
Definition: omdocument.cc:71
virtual MatchSpy * unserialise(const std::string &serialised, const Registry &context) const
Unserialise parameters.
Definition: matchspy.cc:66
const valueno BAD_VALUENO
Reserved value to indicate "no valueno".
Definition: types.h:125
void msg_deletedocumentterm_(const std::string &message)
void msg_document(const std::string &message)
Xapian::termcount get_unique_terms(Xapian::docid did) const
Get the number of unique terms in document.
Definition: omdatabase.cc:476
PostingIterator postlist_end(const std::string &) const
Corresponding end iterator to postlist_begin().
Definition: database.h:225
functions to convert classes to strings and back
docid_order
Ordering of docids.
Definition: enquire.h:326
void decode_length(const char **p, const char *end, unsigned &out)
Decode a length encoded by encode_length.
Definition: length.cc:94
Xapian::doccount get_termfreq(const std::string &tname) const
Get the number of documents in the database indexed by a given term.
Definition: omdatabase.cc:323
Xapian::doccount get_value_freq(Xapian::valueno slot) const
Return the frequency of a given value slot.
Definition: omdatabase.cc:355
A handle representing a document in a Xapian database.
Definition: document.h:61
Wrapper around standard unique_ptr template.
void msg_termfreq(const std::string &message)
A relevance set (R-Set).
Definition: enquire.h:60
std::string get_uuid() const
Get a UUID for the database.
Definition: omdatabase.cc:776
void add_spelling(const std::string &word, Xapian::termcount freqinc=1) const
Add a word to the spelling dictionary.
Definition: omdatabase.cc:1004
PostingIterator postlist_begin(const std::string &tname) const
An iterator pointing to the start of the postlist for a given term.
Definition: omdatabase.cc:162
message_type get_message(double timeout, std::string &result, message_type required_type=MSG_MAX)
Accept a message from the client.
Abstract base class for weighting schemes.
Definition: weight.h:35
Xapian::termcount get_collection_freq(const std::string &tname) const
Return the total number of occurrences of the given term.
Definition: omdatabase.cc:339
Xapian::TermIterator metadata_keys_end(const std::string &=std::string()) const
Corresponding end iterator to metadata_keys_begin().
Definition: database.h:510