Extended the result set system. Added support for filtering/limits.
[idzebra-moved-to-github.git] / rset / rsisamb.c
1 /* $Id: rsisamb.c,v 1.33 2005-05-03 09:11:36 adam Exp $
2    Copyright (C) 1995-2005
3    Index Data ApS
4
5 This file is part of the Zebra server.
6
7 Zebra is free software; you can redistribute it and/or modify it under
8 the terms of the GNU General Public License as published by the Free
9 Software Foundation; either version 2, or (at your option) any later
10 version.
11
12 Zebra is distributed in the hope that it will be useful, but WITHOUT ANY
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
15 for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with Zebra; see the file LICENSE.zebra.  If not, write to the
19 Free Software Foundation, 59 Temple Place - Suite 330, Boston, MA
20 02111-1307, USA.
21 */
22
23 #include <stdio.h>
24 #include <assert.h>
25 #include <idzebra/util.h>
26 #include <rset.h>
27 #include <string.h>
28
29 static RSFD r_open(RSET ct, int flag);
30 static void r_close(RSFD rfd);
31 static void r_delete(RSET ct);
32 static int r_forward(RSFD rfd, void *buf, TERMID *term, const void *untilbuf);
33 static void r_pos(RSFD rfd, double *current, double *total);
34 static int r_read(RSFD rfd, void *buf, TERMID *term);
35 static int r_read_filter(RSFD rfd, void *buf, TERMID *term);
36 static int r_write(RSFD rfd, const void *buf);
37
38 static const struct rset_control control = 
39 {
40     "isamb",
41     r_delete,
42     rset_get_one_term,
43     r_open,
44     r_close,
45     r_forward, 
46     r_pos,
47     r_read,
48     r_write,
49 };
50
51 static const struct rset_control control_filter = 
52 {
53     "isamb",
54     r_delete,
55     rset_get_one_term,
56     r_open,
57     r_close,
58     r_forward, 
59     r_pos,
60     r_read_filter,
61     r_write,
62 };
63
64 struct rfd_private {
65     ISAMB_PP pt;
66     void *buf;
67 };
68
69 struct rset_private {
70     ISAMB   is;
71     ISAM_P pos;
72 };
73
74 static int log_level = 0;
75 static int log_level_initialized = 0;
76
77 RSET rsisamb_create(NMEM nmem, struct rset_key_control *kcontrol,
78                     int scope,
79                     ISAMB is, ISAM_P pos, TERMID term)
80 {
81     RSET rnew = rset_create_base(
82         kcontrol->filter_func ? &control_filter : &control,
83         nmem, kcontrol, scope, term);
84     struct rset_private *info;
85     if (!log_level_initialized)
86     {
87         log_level = yaz_log_module_level("rsisamb");
88         log_level_initialized = 1;
89     }
90     info = (struct rset_private *) nmem_malloc(rnew->nmem, sizeof(*info));
91     info->is = is;
92     info->pos = pos;
93     rnew->priv = info;
94     yaz_log(log_level, "rsisamb_create");
95     return rnew;
96 }
97
98 static void r_delete(RSET ct)
99 {
100     yaz_log(log_level, "rsisamb_delete");
101 }
102
103 RSFD r_open(RSET ct, int flag)
104 {
105     struct rset_private *info = (struct rset_private *) ct->priv;
106     RSFD rfd;
107     struct rfd_private *ptinfo;
108
109     if (flag & RSETF_WRITE)
110     {
111         yaz_log(YLOG_FATAL, "ISAMB set type is read-only");
112         return NULL;
113     }
114     rfd = rfd_create_base(ct);
115     if (rfd->priv)
116         ptinfo = (struct rfd_private *) (rfd->priv);
117     else {
118         ptinfo = (struct rfd_private *) nmem_malloc(ct->nmem,sizeof(*ptinfo));
119         ptinfo->buf = nmem_malloc(ct->nmem,ct->keycontrol->key_size);
120         rfd->priv = ptinfo;
121     }
122     ptinfo->pt = isamb_pp_open(info->is, info->pos, ct->scope );
123     yaz_log(log_level, "rsisamb_open");
124     return rfd;
125 }
126
127 static void r_close(RSFD rfd)
128 {
129     struct rfd_private *ptinfo = (struct rfd_private *)(rfd->priv);
130     isamb_pp_close (ptinfo->pt);
131     rfd_delete_base(rfd);
132     yaz_log(log_level, "rsisamb_close");
133 }
134
135
136 static int r_forward(RSFD rfd, void *buf, TERMID *term, const void *untilbuf)
137 {
138     struct rfd_private *pinfo = (struct rfd_private *)(rfd->priv);
139     int rc;
140     rc = isamb_pp_forward(pinfo->pt, buf, untilbuf);
141     if (rc && term)
142         *term = rfd->rset->term;
143     yaz_log(log_level, "rsisamb_forward");
144     return rc; 
145 }
146
147 static void r_pos(RSFD rfd, double *current, double *total)
148 {
149     struct rfd_private *pinfo = (struct rfd_private *)(rfd->priv);
150     assert(rfd);
151     isamb_pp_pos(pinfo->pt, current, total);
152     yaz_log(log_level, "isamb.r_pos returning %0.1f/%0.1f",
153               *current, *total);
154 }
155
156 static int r_read(RSFD rfd, void *buf, TERMID *term)
157 {
158     struct rfd_private *pinfo = (struct rfd_private *)(rfd->priv);
159     int rc;
160     rc = isamb_pp_read(pinfo->pt, buf);
161     if (rc && term)
162         *term = rfd->rset->term;
163     yaz_log(log_level, "isamb.r_read ");
164     return rc;
165 }
166
167 static int r_read_filter(RSFD rfd, void *buf, TERMID *term)
168 {
169     struct rfd_private *pinfo = (struct rfd_private *)rfd->priv;
170     const struct rset_key_control *kctrl = rfd->rset->keycontrol;
171     int rc;
172     while((rc = isamb_pp_read(pinfo->pt, buf)))
173     {
174         int incl = (*kctrl->filter_func)(buf, kctrl->filter_data);
175         if (incl)
176             break;
177     }
178     if (rc && term)
179         *term = rfd->rset->term;
180     yaz_log(log_level, "isamb.r_read_filter");
181     return rc;
182 }
183
184 static int r_write(RSFD rfd, const void *buf)
185 {
186     yaz_log(YLOG_FATAL, "ISAMB set type is read-only");
187     return -1;
188 }