Line data Source code
1 : /*
2 : This file is part of TALER
3 : Copyright (C) 2022 Taler Systems SA
4 :
5 : TALER is free software; you can redistribute it and/or modify it under the
6 : terms of the GNU General Public License as published by the Free Software
7 : Foundation; either version 3, or (at your option) any later version.
8 :
9 : TALER is distributed in the hope that it will be useful, but WITHOUT ANY
10 : WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
11 : A PARTICULAR PURPOSE. See the GNU General Public License for more details.
12 :
13 : You should have received a copy of the GNU General Public License along with
14 : TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
15 : */
16 : /**
17 : * @file exchangedb/pg_lookup_global_fee_by_time.c
18 : * @brief Implementation of the lookup_global_fee_by_time function for Postgres
19 : * @author Christian Grothoff
20 : */
21 : #include "taler/platform.h"
22 : #include "taler/taler_error_codes.h"
23 : #include "taler/taler_dbevents.h"
24 : #include "taler/taler_pq_lib.h"
25 : #include "pg_lookup_global_fee_by_time.h"
26 : #include "pg_helper.h"
27 :
28 : /**
29 : * Closure for #global_fee_by_time_helper()
30 : */
31 : struct GlobalFeeLookupContext
32 : {
33 :
34 : /**
35 : * Set to the wire fees. Set to invalid if fees conflict over
36 : * the given time period.
37 : */
38 : struct TALER_GlobalFeeSet *fees;
39 :
40 : /**
41 : * Set to timeout of unmerged purses
42 : */
43 : struct GNUNET_TIME_Relative *purse_timeout;
44 :
45 : /**
46 : * Set to history expiration for reserves.
47 : */
48 : struct GNUNET_TIME_Relative *history_expiration;
49 :
50 : /**
51 : * Set to number of free purses per account.
52 : */
53 : uint32_t *purse_account_limit;
54 :
55 : /**
56 : * Plugin context.
57 : */
58 : struct PostgresClosure *pg;
59 : };
60 :
61 :
62 : /**
63 : * Helper function for #TEH_PG_lookup_global_fee_by_time().
64 : * Calls the callback with each denomination key.
65 : *
66 : * @param cls a `struct GlobalFeeLookupContext`
67 : * @param result db results
68 : * @param num_results number of results in @a result
69 : */
70 : static void
71 19 : global_fee_by_time_helper (void *cls,
72 : PGresult *result,
73 : unsigned int num_results)
74 : {
75 19 : struct GlobalFeeLookupContext *wlc = cls;
76 19 : struct PostgresClosure *pg = wlc->pg;
77 :
78 19 : for (unsigned int i = 0; i<num_results; i++)
79 : {
80 : struct TALER_GlobalFeeSet fs;
81 : struct GNUNET_TIME_Relative purse_timeout;
82 : struct GNUNET_TIME_Relative history_expiration;
83 : uint32_t purse_account_limit;
84 0 : struct GNUNET_PQ_ResultSpec rs[] = {
85 0 : TALER_PQ_RESULT_SPEC_AMOUNT ("history_fee",
86 : &fs.history),
87 0 : TALER_PQ_RESULT_SPEC_AMOUNT ("account_fee",
88 : &fs.account),
89 0 : TALER_PQ_RESULT_SPEC_AMOUNT ("purse_fee",
90 : &fs.purse),
91 0 : GNUNET_PQ_result_spec_relative_time ("purse_timeout",
92 : &purse_timeout),
93 0 : GNUNET_PQ_result_spec_relative_time ("history_expiration",
94 : &history_expiration),
95 0 : GNUNET_PQ_result_spec_uint32 ("purse_account_limit",
96 : &purse_account_limit),
97 : GNUNET_PQ_result_spec_end
98 : };
99 :
100 0 : if (GNUNET_OK !=
101 0 : GNUNET_PQ_extract_result (result,
102 : rs,
103 : i))
104 : {
105 0 : GNUNET_break (0);
106 : /* invalidate */
107 0 : memset (wlc->fees,
108 : 0,
109 : sizeof (struct TALER_GlobalFeeSet));
110 0 : return;
111 : }
112 0 : if (0 == i)
113 : {
114 0 : *wlc->fees = fs;
115 0 : *wlc->purse_timeout = purse_timeout;
116 0 : *wlc->history_expiration = history_expiration;
117 0 : *wlc->purse_account_limit = purse_account_limit;
118 0 : continue;
119 : }
120 0 : if ( (0 !=
121 0 : TALER_global_fee_set_cmp (&fs,
122 0 : wlc->fees)) ||
123 0 : (purse_account_limit != *wlc->purse_account_limit) ||
124 0 : (GNUNET_TIME_relative_cmp (purse_timeout,
125 : !=,
126 0 : *wlc->purse_timeout)) ||
127 0 : (GNUNET_TIME_relative_cmp (history_expiration,
128 : !=,
129 : *wlc->history_expiration)) )
130 : {
131 : /* invalidate */
132 0 : memset (wlc->fees,
133 : 0,
134 : sizeof (struct TALER_GlobalFeeSet));
135 0 : return;
136 : }
137 : }
138 : }
139 :
140 :
141 : enum GNUNET_DB_QueryStatus
142 19 : TEH_PG_lookup_global_fee_by_time (
143 : void *cls,
144 : struct GNUNET_TIME_Timestamp start_time,
145 : struct GNUNET_TIME_Timestamp end_time,
146 : struct TALER_GlobalFeeSet *fees,
147 : struct GNUNET_TIME_Relative *purse_timeout,
148 : struct GNUNET_TIME_Relative *history_expiration,
149 : uint32_t *purse_account_limit)
150 : {
151 19 : struct PostgresClosure *pg = cls;
152 19 : struct GNUNET_PQ_QueryParam params[] = {
153 19 : GNUNET_PQ_query_param_timestamp (&start_time),
154 19 : GNUNET_PQ_query_param_timestamp (&end_time),
155 : GNUNET_PQ_query_param_end
156 : };
157 19 : struct GlobalFeeLookupContext wlc = {
158 : .fees = fees,
159 : .purse_timeout = purse_timeout,
160 : .history_expiration = history_expiration,
161 : .purse_account_limit = purse_account_limit,
162 : .pg = pg
163 : };
164 :
165 19 : PREPARE (pg,
166 : "lookup_global_fee_by_time",
167 : "SELECT"
168 : " history_fee"
169 : ",account_fee"
170 : ",purse_fee"
171 : ",purse_timeout"
172 : ",history_expiration"
173 : ",purse_account_limit"
174 : " FROM global_fee"
175 : " WHERE end_date > $1"
176 : " AND start_date < $2;");
177 19 : return GNUNET_PQ_eval_prepared_multi_select (pg->conn,
178 : "lookup_global_fee_by_time",
179 : params,
180 : &global_fee_by_time_helper,
181 : &wlc);
182 : }
|