/* guile-xapian --- Guile bindings for Xapian * Copyright © 2020, 2023–2024 Arun Isaac <arunisaac@systemreboot.net> * Copyright © 2021, 2022 Bob131 <bob@bob131.so> * * This file is part of guile-xapian. * * guile-xapian is free software: you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation, either version 2 of the * License, or (at your option) any later version. * * guile-xapian is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with guile-xapian. If not, see * <https://www.gnu.org/licenses/>. */ %module wrap %{ %} %scheme %{ ;; Try loading the shared library by absolute path. If that fails, try ;; searching for it using LD_LIBRARY_PATH and other mechanisms. We do this ;; so that guile-xapian finds libguilexapian both before and after installation. (catch #t (lambda () (load-extension "@libdir@/libguilexapian-@GUILE_EFFECTIVE_VERSION@" "SWIG_init")) (lambda _ (load-extension "libguilexapian-@GUILE_EFFECTIVE_VERSION@" "SWIG_init"))) %} %{ #include <string> // We introduce a dummy typedef for std::string without declaring it // to SWIG so we can establish a new SWIG typemap applying only to // certain redeclarations of existing C++ functions taking or // returning strings. typedef std::string SWIG_bytevector; static inline void typemap_in_SWIG_bytevector (SWIG_bytevector& out, SCM in) { out.assign ((char*) SCM_BYTEVECTOR_CONTENTS (in), SCM_BYTEVECTOR_LENGTH (in)); } static inline SCM typemap_out_SWIG_bytevector (const SWIG_bytevector& in) { SCM result = scm_c_make_bytevector (in.size ()); memcpy (SCM_BYTEVECTOR_CONTENTS (result), in.data (), in.size ()); return result; } %} %typemap(in) SWIG_bytevector { if (scm_is_bytevector ($input)) { $1 = std::string (); typemap_in_SWIG_bytevector ($1, $input); } else { %argument_fail (SWIG_TypeError, "$type", $symname, $argnum); } } %typemap(in) SWIG_bytevector&, SWIG_bytevector* { if (scm_is_bytevector ($input)) { $1 = new std::string (); typemap_in_SWIG_bytevector (*$1, $input); } else { %argument_fail (SWIG_TypeError, "$type", $symname, $argnum); } } %typemap(out) SWIG_bytevector { $result = typemap_out_SWIG_bytevector ($1); } %typemap(out) SWIG_bytevector&, SWIG_bytevector* { $result = typemap_out_SWIG_bytevector (*$1); } %typecheck(SWIG_TYPECHECK_SWIGOBJECT) SWIG_bytevector, SWIG_bytevector&, SWIG_bytevector* { $1 = scm_is_bytevector ($input); } %include xapian-head.i %include except.i %include xapian-headers.i %extend Xapian::ValueIterator { SWIG_bytevector get_value_bytes () { return **$self; } } %extend Xapian::Document { SWIG_bytevector get_value_bytes (Xapian::valueno valueno) { return $self->get_value (valueno); } void add_value_bytes (Xapian::valueno valueno, const SWIG_bytevector& bytes) { $self->add_value (valueno, bytes); } SWIG_bytevector get_data_bytes () { return $self->get_data (); } void set_data_bytes (const SWIG_bytevector& bytes) { $self->set_data (bytes); } } %extend Xapian::Database { SWIG_bytevector get_metadata_bytes (const std::string& key) { return $self->get_metadata (key); } } %extend Xapian::WritableDatabase { void set_metadata_bytes (const std::string& key, const SWIG_bytevector& bytes) { $self->set_metadata (key, bytes); } } %extend Xapian::Query { Query (op op_, Xapian::valueno slot, const SWIG_bytevector& range_limit) { return new Xapian::Query (op_, slot, range_limit); } Query (op op_, Xapian::valueno slot, const SWIG_bytevector& range_lower, const SWIG_bytevector& range_upper) { return new Xapian::Query (op_, slot, range_lower, range_upper); } } // Child class of Xapian::RangeProcessor that calls back to a // user-specified Scheme procedure to process fields. %{ class GuileXapianRangeProcessorWrapper : public Xapian::RangeProcessor { private: SCM proc; public: GuileXapianRangeProcessorWrapper(Xapian::valueno slot, const std::string &str, unsigned flags, SCM _proc) : Xapian::RangeProcessor(slot, str, flags) { proc = _proc; } Xapian::Query operator()(const std::string &begin, const std::string &end) { void *ptr; int res = SWIG_ConvertPtr(scm_call_2(proc, begin.empty() ? SCM_BOOL_F : scm_from_utf8_string(begin.c_str()), end.empty() ? SCM_BOOL_F : scm_from_utf8_string(end.c_str())), &ptr, SWIGTYPE_p_Xapian__Query, 0); return *((Xapian::Query*)ptr); } }; %} class GuileXapianRangeProcessorWrapper : public Xapian::RangeProcessor { public: GuileXapianRangeProcessorWrapper(Xapian::valueno, std::string const&, unsigned, SCM); ~GuileXapianRangeProcessorWrapper(); Xapian::Query operator()(std::string const&, std::string const&); }; // Child class of Xapian::FieldProcessor that calls back to a // user-specified Scheme procedure to process fields. %{ class GuileXapianFieldProcessorWrapper : public Xapian::FieldProcessor { private: SCM proc; public: GuileXapianFieldProcessorWrapper(SCM _proc) { proc = _proc; } Xapian::Query operator()(const std::string &str) { void *ptr; int res = SWIG_ConvertPtr(scm_call_1(proc, scm_from_utf8_string(str.c_str())), &ptr, SWIGTYPE_p_Xapian__Query, 0); return *((Xapian::Query*)ptr); } }; %} class GuileXapianFieldProcessorWrapper : public Xapian::FieldProcessor { public: GuileXapianFieldProcessorWrapper(SCM); ~GuileXapianFieldProcessorWrapper(); Xapian::Query operator()(std::string const&); };