/* This file is part of TALER (C) 2014 Christian Grothoff (and other contributing authors) TALER 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 3, or (at your option) any later version. TALER 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 TALER; see the file COPYING. If not, If not, see */ /** * @file taler-mint-dbinit.c * @brief Create tables for the mint database. * @author Florian Dold */ #include "platform.h" #include #include #include "mint.h" #define break_db_err(result) do { \ GNUNET_break(0); \ GNUNET_log (GNUNET_ERROR_TYPE_ERROR, "Database failure: %s\n", PQresultErrorMessage (result)); \ PQclear (result); \ } while (0) static char *mint_base_dir; static struct GNUNET_CONFIGURATION_Handle *cfg; static PGconn *db_conn; static char *TALER_MINT_db_connection_cfg_str; int TALER_MINT_init_withdraw_tables (PGconn *conn) { PGresult *result; result = PQexec (conn, "CREATE TABLE IF NOT EXISTS reserves" "(" " reserve_pub BYTEA PRIMARY KEY" ",balance_value INT4 NOT NULL" ",balance_fraction INT4 NOT NULL" ",balance_currency VARCHAR(4) NOT NULL" ",status_sig BYTEA" ",status_sign_pub BYTEA" ",expiration_date INT8 NOT NULL" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS collectable_blindcoins" "(" "blind_ev BYTEA PRIMARY KEY" ",blind_ev_sig BYTEA NOT NULL" ",denom_pub BYTEA NOT NULL" ",reserve_sig BYTEA NOT NULL" ",reserve_pub BYTEA NOT NULL REFERENCES reserves (reserve_pub)" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS known_coins " "(" " coin_pub BYTEA NOT NULL PRIMARY KEY" ",denom_pub BYTEA NOT NULL" ",denom_sig BYTEA NOT NULL" ",expended_value INT4 NOT NULL" ",expended_fraction INT4 NOT NULL" ",expended_currency VARCHAR(4) NOT NULL" ",refresh_session_pub BYTEA" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS refresh_sessions " "(" " session_pub BYTEA PRIMARY KEY CHECK (length(session_pub) = 32)" ",session_melt_sig BYTEA" ",session_commit_sig BYTEA" ",noreveal_index INT2 NOT NULL" // non-zero if all reveals were ok // and the new coin signatures are ready ",reveal_ok BOOLEAN NOT NULL DEFAULT false" ") "); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS refresh_order " "( " " session_pub BYTEA NOT NULL REFERENCES refresh_sessions (session_pub)" ",newcoin_index INT2 NOT NULL " ",denom_pub BYTEA NOT NULL " ",PRIMARY KEY (session_pub, newcoin_index)" ") "); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS refresh_commit_link" "(" " session_pub BYTEA NOT NULL REFERENCES refresh_sessions (session_pub)" ",transfer_pub BYTEA NOT NULL" ",link_secret_enc BYTEA NOT NULL" // index of the old coin in the customer's request ",oldcoin_index INT2 NOT NULL" // index for cut and choose, // ranges from 0 to kappa-1 ",cnc_index INT2 NOT NULL" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS refresh_commit_coin" "(" " session_pub BYTEA NOT NULL REFERENCES refresh_sessions (session_pub) " ",link_vector_enc BYTEA NOT NULL" // index of the new coin in the customer's request ",newcoin_index INT2 NOT NULL" // index for cut and choose, ",cnc_index INT2 NOT NULL" ",coin_ev BYTEA NOT NULL" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS refresh_melt" "(" " session_pub BYTEA NOT NULL REFERENCES refresh_sessions (session_pub) " ",coin_pub BYTEA NOT NULL REFERENCES known_coins (coin_pub) " ",denom_pub BYTEA NOT NULL " ",oldcoin_index INT2 NOT NULL" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS refresh_collectable" "(" " session_pub BYTEA NOT NULL REFERENCES refresh_sessions (session_pub) " ",ev_sig BYTEA NOT NULL" ",newcoin_index INT2 NOT NULL" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); result = PQexec (conn, "CREATE TABLE IF NOT EXISTS deposits " "( " " coin_pub BYTEA NOT NULL PRIMARY KEY CHECK (length(coin_pub)=32)" ",denom_pub BYTEA NOT NULL CHECK (length(denom_pub)=32)" ",transaction_id INT8 NOT NULL" ",amount_currency VARCHAR(4) NOT NULL" ",amount_value INT4 NOT NULL" ",amount_fraction INT4 NOT NULL" ",merchant_pub BYTEA NOT NULL" ",h_contract BYTEA NOT NULL CHECK (length(h_contract)=64)" ",h_wire BYTEA NOT NULL CHECK (length(h_wire)=64)" ",coin_sig BYTEA NOT NULL CHECK (length(coin_sig)=64)" ",wire TEXT NOT NULL" ")"); if (PGRES_COMMAND_OK != PQresultStatus(result)) { break_db_err (result); return GNUNET_SYSERR; } PQclear (result); return GNUNET_OK; } /** * The main function of the serve tool * * @param argc number of arguments from the command line * @param argv command line arguments * @return 0 ok, 1 on error */ int main (int argc, char *const *argv) { static const struct GNUNET_GETOPT_CommandLineOption options[] = { GNUNET_GETOPT_OPTION_HELP ("gnunet-mint-keyup OPTIONS"), {'d', "mint-dir", "DIR", "mint directory", 1, &GNUNET_GETOPT_set_filename, &mint_base_dir}, GNUNET_GETOPT_OPTION_END }; if (GNUNET_GETOPT_run ("taler-mint-serve", options, argc, argv) < 0) return 1; GNUNET_assert (GNUNET_OK == GNUNET_log_setup ("taler-mint-dbinit", "INFO", NULL)); if (NULL == mint_base_dir) { fprintf (stderr, "Mint base directory not given.\n"); return 1; } cfg = TALER_config_load (mint_base_dir); if (NULL == cfg) { fprintf (stderr, "Can't load mint configuration.\n"); return 1; } if (GNUNET_OK != GNUNET_CONFIGURATION_get_value_string (cfg, "mint", "db", &TALER_MINT_db_connection_cfg_str)) { fprintf (stderr, "Configuration 'mint.db' not found.\n"); return 42; } db_conn = PQconnectdb (TALER_MINT_db_connection_cfg_str); if (CONNECTION_OK != PQstatus (db_conn)) { fprintf (stderr, "Database connection failed: %s\n", PQerrorMessage (db_conn)); return 1; } if (GNUNET_OK != TALER_MINT_init_withdraw_tables (db_conn)) { fprintf (stderr, "Failed to initialize database.\n"); return 1; } return 0; }