Line data Source code
1 : /*
2 : This file is part of TALER
3 : (C) 2025 Taler Systems SA
4 :
5 : TALER is free software; you can redistribute it and/or modify
6 : it under the terms of the GNU Affero General Public License as
7 : published by the Free Software Foundation; either version 3,
8 : or (at your option) any later version.
9 :
10 : TALER is distributed in the hope that it will be useful, but
11 : WITHOUT ANY WARRANTY; without even the implied warranty of
12 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 : GNU General Public License for more details.
14 :
15 : You should have received a copy of the GNU General Public
16 : License along with TALER; see the file COPYING. If not,
17 : see <http://www.gnu.org/licenses/>
18 : */
19 :
20 : /**
21 : * @file taler-merchant-httpd_post-challenge-ID-confirm.c
22 : * @brief endpoint to solve MFA challenge
23 : * @author Christian Grothoff
24 : */
25 : #include "platform.h"
26 : #include "taler-merchant-httpd.h"
27 : #include "taler-merchant-httpd_mfa.h"
28 : #include "taler-merchant-httpd_post-challenge-ID-confirm.h"
29 :
30 :
31 : MHD_RESULT
32 0 : TMH_post_challenge_ID_confirm (const struct TMH_RequestHandler *rh,
33 : struct MHD_Connection *connection,
34 : struct TMH_HandlerContext *hc)
35 : {
36 : uint64_t challenge_serial;
37 : struct TALER_MERCHANT_MFA_BodyHash h_body;
38 : const char *tan;
39 : struct GNUNET_JSON_Specification spec[] = {
40 0 : GNUNET_JSON_spec_string ("tan",
41 : &tan),
42 0 : GNUNET_JSON_spec_end ()
43 : };
44 : enum GNUNET_DB_QueryStatus qs;
45 : bool solved;
46 : uint32_t retry_counter;
47 : enum GNUNET_GenericReturnValue ret;
48 :
49 0 : ret = TMH_mfa_parse_challenge_id (hc,
50 0 : hc->infix,
51 : &challenge_serial,
52 : &h_body);
53 0 : if (GNUNET_OK != ret)
54 0 : return (GNUNET_SYSERR == ret) ? MHD_NO : MHD_YES;
55 : {
56 : enum GNUNET_GenericReturnValue res;
57 :
58 0 : res = TALER_MHD_parse_json_data (hc->connection,
59 0 : hc->request_body,
60 : spec);
61 0 : if (GNUNET_OK != res)
62 : {
63 0 : GNUNET_break_op (0);
64 : return (GNUNET_NO == res)
65 : ? MHD_YES
66 0 : : MHD_NO;
67 : }
68 : }
69 0 : qs = TMH_db->solve_mfa_challenge (TMH_db->cls,
70 : challenge_serial,
71 : &h_body,
72 : tan,
73 : &solved,
74 : &retry_counter);
75 0 : switch (qs)
76 : {
77 0 : case GNUNET_DB_STATUS_HARD_ERROR:
78 0 : GNUNET_break (0);
79 0 : return TALER_MHD_reply_with_error (
80 : hc->connection,
81 :
82 : MHD_HTTP_INTERNAL_SERVER_ERROR,
83 : TALER_EC_GENERIC_DB_COMMIT_FAILED,
84 : "solve_mfa_challenge");
85 0 : case GNUNET_DB_STATUS_SOFT_ERROR:
86 0 : GNUNET_break (0);
87 0 : return TALER_MHD_reply_with_error (
88 : hc->connection,
89 : MHD_HTTP_INTERNAL_SERVER_ERROR,
90 : TALER_EC_GENERIC_DB_SOFT_FAILURE,
91 : "solve_mfa_challenge");
92 0 : case GNUNET_DB_STATUS_SUCCESS_NO_RESULTS:
93 0 : GNUNET_break_op (0);
94 0 : return TALER_MHD_reply_with_error (
95 : hc->connection,
96 : MHD_HTTP_NOT_FOUND,
97 : TALER_EC_MERCHANT_TAN_CHALLENGE_UNKNOWN,
98 0 : hc->infix);
99 0 : case GNUNET_DB_STATUS_SUCCESS_ONE_RESULT:
100 0 : break;
101 : }
102 0 : if (0 == retry_counter)
103 : {
104 0 : return TALER_MHD_reply_with_error (
105 : hc->connection,
106 : MHD_HTTP_TOO_MANY_REQUESTS,
107 : TALER_EC_MERCHANT_TAN_TOO_MANY_ATTEMPTS,
108 : NULL);
109 : }
110 0 : if (! solved)
111 : {
112 0 : return TALER_MHD_REPLY_JSON_PACK (
113 : hc->connection,
114 : MHD_HTTP_CONFLICT,
115 : TALER_MHD_PACK_EC (TALER_EC_MERCHANT_TAN_CHALLENGE_FAILED),
116 : GNUNET_JSON_pack_uint64 ("retry_counter",
117 : retry_counter));
118 : }
119 0 : return TALER_MHD_reply_static (
120 : hc->connection,
121 : MHD_HTTP_NO_CONTENT,
122 : NULL,
123 : NULL,
124 : 0);
125 : }
|