You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
neroshop/src/gui/backend.cpp

1498 lines
128 KiB

#include "backend.hpp"
#include <QNetworkAccessManager>
#include <QNetworkReply>
#include <QNetworkRequest>
#include <QFile>
#include <future>
#include <thread>
neroshop::Backend::Backend(QObject *parent) : QObject(parent) {}
QString neroshop::Backend::urlToLocalFile(const QUrl &url) const
{
return url.toLocalFile();
}
//----------------------------------------------------------------
void neroshop::Backend::copyTextToClipboard(const QString& text) {
QClipboard * clipboard = QGuiApplication::clipboard();
clipboard->setText(text);
std::cout << "Copied text to clipboard\n";
}
//----------------------------------------------------------------
//----------------------------------------------------------------
QStringList neroshop::Backend::getCurrencyList() const
{
QStringList currency_list;
for (const auto& [key, value] : neroshop::CurrencyMap) {
currency_list << QString::fromStdString(key);
}
return currency_list;
}
//----------------------------------------------------------------
int neroshop::Backend::getCurrencyDecimals(const QString& currency) const {
auto map_key = currency.toUpper().toStdString();
// Check if key exists in std::map
if(neroshop::CurrencyMap.count(map_key) > 0) {
auto map_value = neroshop::CurrencyMap[map_key];
int decimal_places = std::get<2>(map_value);
return decimal_places;
}
return 2;
}
//----------------------------------------------------------------
QString neroshop::Backend::getCurrencySign(const QString& currency) const {
return QString::fromStdString(neroshop::Converter::get_currency_sign(currency.toStdString()));
}
//----------------------------------------------------------------
bool neroshop::Backend::isSupportedCurrency(const QString& currency) const {
return neroshop::Converter::is_supported_currency(currency.toStdString());
}
//----------------------------------------------------------------
//----------------------------------------------------------------
void neroshop::Backend::initializeDatabase() {
std::cout << "sqlite3 v" << db::Sqlite3::get_sqlite_version() << std::endl;
db::Sqlite3 * database = neroshop::get_database();
database->execute("BEGIN;");
//-------------------------
// Todo: Make monero_address the primary key and remove id. Also, replace all foreign key references from id to monero_address
// table users
if(!database->table_exists("users")) {
database->execute("CREATE TABLE users(name TEXT, monero_address TEXT NOT NULL PRIMARY KEY"//, UNIQUE"
");");
database->execute("ALTER TABLE users ADD COLUMN public_key TEXT DEFAULT NULL;"); // encrypt_key - public_key used for encryption of messages
database->execute("ALTER TABLE users ADD COLUMN avatar BLOB DEFAULT NULL;"); // encrypt_key - public_key used for encryption of messages
// Notes: Display names are optional which means they can be an empty string but making the "name" column UNIQUE will not allow empty strings on multiple names
////database->execute("CREATE UNIQUE INDEX index_public_keys ON users (public_key);"); // This is commented out to allow multiple users to use the same public key, in the case of a user having two neroshop accounts?
}
// products (represents both items and services)
if(!database->table_exists("products")) {
database->execute("CREATE TABLE products(uuid TEXT NOT NULL PRIMARY KEY);");
database->execute("ALTER TABLE products ADD COLUMN name TEXT;");
database->execute("ALTER TABLE products ADD COLUMN description TEXT;");
//database->execute("ALTER TABLE products ADD COLUMN price REAL");// This should be the manufacturer's original price (won't be used though) // unit_price or price_per_unit
database->execute("ALTER TABLE products ADD COLUMN weight REAL;"); // kg // TODO: add weight to attributes
database->execute("ALTER TABLE products ADD COLUMN attributes TEXT;"); // attribute options format: "Color:Red,Green,Blue;Size:XS,S,M,L,XL"// Can be a number(e.g 16) or a text(l x w x h)
database->execute("ALTER TABLE products ADD COLUMN code TEXT;"); // product_code can be either upc (universal product code) or a custom sku
database->execute("ALTER TABLE products ADD COLUMN category_id INTEGER REFERENCES categories(id);");
//database->execute("ALTER TABLE products ADD COLUMN subcategory_id INTEGER REFERENCES categories(id);");
//database->execute("ALTER TABLE products ADD COLUMN ?col ?datatype;");
//database->execute("CREATE UNIQUE INDEX ?index ON products (?col);");
// the seller determines the final product price, the product condition and whether the product will have a discount or not
// Note: UPC codes can be totally different for the different variations(color, etc.) of the same product
}
// inventory // Todo: rename this to "listings"
if(!database->table_exists("listings")) {
database->execute("CREATE TABLE listings(uuid TEXT NOT NULL PRIMARY KEY);");
database->execute("ALTER TABLE listings ADD COLUMN product_id TEXT REFERENCES products(uuid);");
database->execute("ALTER TABLE listings ADD COLUMN seller_id TEXT REFERENCES users(monero_address);"); // alternative names: "store_id"
database->execute("ALTER TABLE listings ADD COLUMN quantity INTEGER;"); // stock available
database->execute("ALTER TABLE listings ADD COLUMN price REAL;"); // this is the final price of a product or list/sales price decided by the seller
database->execute("ALTER TABLE listings ADD COLUMN currency TEXT;"); // the fiat currency the seller is selling the item in
//database->execute("ALTER TABLE listings ADD COLUMN discount numeric(20,12);"); // alternative names: "seller_discount", or "discount_price"
//database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;"); // discount_times_can_use - number of times the discount can be used
//database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;"); // discounted_items_qty - number of items that the discount will apply to
//database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;"); // discount_expiry - date and time that the discount expires (will be in UTC format)
//database->execute("ALTER TABLE listings ADD COLUMN ?col ?datatype;");
database->execute("ALTER TABLE listings ADD COLUMN condition TEXT;"); // item condition
database->execute("ALTER TABLE listings ADD COLUMN location TEXT;");
//database->execute("ALTER TABLE listings ADD COLUMN last_updated ?datatype;");
database->execute("ALTER TABLE listings ADD COLUMN date TEXT DEFAULT CURRENT_TIMESTAMP;"); // date when first listed // will use ISO8601 string format as follows: YYYY-MM-DD HH:MM:SS.SSS
//database->execute("");
// For most recent listings: "SELECT * FROM listings ORDER BY date DESC;"
}
// cart
if(!database->table_exists("cart")) {
// local cart - for a single cart containing a list of product_ids
// public cart - copied to all peers' databases
database->execute("CREATE TABLE cart(uuid TEXT NOT NULL PRIMARY KEY);");
database->execute("ALTER TABLE cart ADD COLUMN user_id TEXT REFERENCES users(monero_address);");//database->execute("CREATE TABLE cart(id INTEGER NOT NULL PRIMARY KEY, user_id TEXT REFERENCES users(monero_address));");
// cart_items (public cart)
database->execute("CREATE TABLE cart_item(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
database->execute("ALTER TABLE cart_item ADD COLUMN cart_id TEXT REFERENCES cart(uuid);");
database->execute("ALTER TABLE cart_item ADD COLUMN product_id TEXT REFERENCES products(uuid);");
database->execute("ALTER TABLE cart_item ADD COLUMN quantity INTEGER;");
//database->execute("ALTER TABLE cart_item ADD COLUMN item_price numeric;"); // sales_price will be used for the final pricing rather than the unit_price
//database->execute("ALTER TABLE cart_item ADD COLUMN item_weight REAL;");//database->execute("CREATE TABLE cart_item(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, cart_id TEXT REFERENCES cart(id), product_id TEXT REFERENCES products(id), item_qty INTEGER, item_price NUMERIC, item_weight REAL);");
database->execute("CREATE UNIQUE INDEX index_cart_item ON cart_item (cart_id, product_id);"); // cart_id and product_id duo must be unqiue for each row
}
// orders (purchase_orders)
if(!database->table_exists("orders")) {
database->execute("CREATE TABLE orders(uuid TEXT NOT NULL PRIMARY KEY);");//database->execute("ALTER TABLE orders ADD COLUMN ?col ?datatype;");
database->execute("ALTER TABLE orders ADD COLUMN timestamp TEXT DEFAULT CURRENT_TIMESTAMP;"); // creation_date // to get UTC time: set to datetime('now');
//database->execute("ALTER TABLE orders ADD COLUMN number TEXT;"); // uuid
database->execute("ALTER TABLE orders ADD COLUMN status TEXT;");
database->execute("ALTER TABLE orders ADD COLUMN user_id TEXT REFERENCES users(monero_address);"); // the user that placed the order
//database->execute("ALTER TABLE orders ADD COLUMN weight REAL;"); // weight of all order items combined - not essential
database->execute("ALTER TABLE orders ADD COLUMN subtotal numeric(20, 12);");
database->execute("ALTER TABLE orders ADD COLUMN discount numeric(20, 12);");
//database->execute("ALTER TABLE orders ADD COLUMN shipping_method TEXT;"); // comment this out
database->execute("ALTER TABLE orders ADD COLUMN shipping_cost numeric(20, 12);");
database->execute("ALTER TABLE orders ADD COLUMN total numeric(20, 12);");
//database->execute("ALTER TABLE orders ADD COLUMN notes TEXT;"); // will contain sensative such as shipping address and tracking numbers that will be encrypted and can only be decrypted by the seller - this may not be necessary since buyer can contact seller privately
//database->execute("ALTER TABLE orders ADD COLUMN order_data TEXT;"); // encrypted JSON
// order_item
database->execute("CREATE TABLE order_item(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
database->execute("ALTER TABLE order_item ADD COLUMN order_id TEXT REFERENCES orders(uuid);");
database->execute("ALTER TABLE order_item ADD COLUMN product_id TEXT REFERENCES products(uuid);");
database->execute("ALTER TABLE order_item ADD COLUMN seller_id TEXT REFERENCES users(monero_address);");
database->execute("ALTER TABLE order_item ADD COLUMN quantity INTEGER;");
//database->execute("ALTER TABLE order_item ADD COLUMN item_price ?datatype;");
//database->execute("ALTER TABLE order_item ADD COLUMN ?col ?datatype;");
}
// ratings - product_ratings, seller_ratings
// maybe merge both item ratings and seller ratings together or nah?
if(!database->table_exists("seller_ratings")) {//if(!database->table_exists("user_ratings")) {
database->execute("CREATE TABLE seller_ratings(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
database->execute("ALTER TABLE seller_ratings ADD COLUMN seller_id TEXT REFERENCES users(monero_address);"); // seller_pkey or seller_pubkey//database->execute("ALTER TABLE user_ratings ADD COLUMN user_id TEXT REFERENCES users(monero_address);"); // seller_pkey or seller_pubkey
database->execute("ALTER TABLE seller_ratings ADD COLUMN score INTEGER;");
database->execute("ALTER TABLE seller_ratings ADD COLUMN user_id TEXT REFERENCES users(monero_address);"); // or rater_id // user_pkey or user_pubkey//database->execute("ALTER TABLE user_ratings ADD COLUMN rater_id TEXT REFERENCES users(monero_address);"); // user_pkey or user_pubkey
database->execute("ALTER TABLE seller_ratings ADD COLUMN comments TEXT;");
database->execute("ALTER TABLE seller_ratings ADD COLUMN signature TEXT;");
}
if(!database->table_exists("product_ratings")) {
database->execute("CREATE TABLE product_ratings(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
database->execute("ALTER TABLE product_ratings ADD COLUMN product_id TEXT REFERENCES products(uuid);");
database->execute("ALTER TABLE product_ratings ADD COLUMN stars INTEGER;");
database->execute("ALTER TABLE product_ratings ADD COLUMN user_id TEXT REFERENCES users(monero_address);");
database->execute("ALTER TABLE product_ratings ADD COLUMN comments TEXT;");
database->execute("ALTER TABLE product_ratings ADD COLUMN signature TEXT;");
}
// images
if(!database->table_exists("images")) { // TODO: rename to product_images?
database->execute("CREATE TABLE images(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
database->execute("ALTER TABLE images ADD COLUMN product_id TEXT REFERENCES products(uuid);");
database->execute("ALTER TABLE images ADD COLUMN name TEXT;");
database->execute("ALTER TABLE images ADD COLUMN data BLOB;");
//database->execute("ALTER TABLE images ADD COLUMN ?col ?datatype;");
}
// avatars - each user will have a single avatar
/*Edit: Since each user can only have one avatar, there won't be a need to store avatars in a separate table
It will be stored in a column in the users table instead.*/
/*if(!database->table_exists("avatars")) {
//database->execute("CREATE TABLE avatars(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
//database->execute("ALTER TABLE avatars ADD COLUMN user_id TEXT REFERENCES users(monero_address);");
//database->execute("ALTER TABLE avatars ADD COLUMN data BLOB;");
}*/
// favorites (wishlists)
if(!database->table_exists("favorites")) {
//database->execute("CREATE TABLE ?tbl(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
//database->execute("ALTER TABLE ?tbl ADD COLUMN user_id TEXT REFERENCES users(monero_address);");
//database->execute("ALTER TABLE ?tbl ADD COLUMN product_ids integer[];");
//database->execute("ALTER TABLE ?tbl ADD COLUMN ?col ?datatype;");
}
// product_categories, product_subcategories
// Products can fall under one category and multiple subcategories
if(!database->table_exists("categories")) { // TODO: rename to product_categories?
database->execute("CREATE TABLE categories(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
database->execute("ALTER TABLE categories ADD COLUMN name TEXT;");
database->execute("ALTER TABLE categories ADD COLUMN description TEXT;"); // alternative names
database->execute("ALTER TABLE categories ADD COLUMN thumbnail TEXT;");
}
if(!database->table_exists("subcategories")) { // TODO: rename to product_subcategories?
database->execute("CREATE TABLE subcategories(id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT);");
database->execute("ALTER TABLE subcategories ADD COLUMN name TEXT;");
database->execute("ALTER TABLE subcategories ADD COLUMN category_id INTEGER REFERENCES categories(id);");
database->execute("ALTER TABLE subcategories ADD COLUMN description TEXT;");
// categories types
int category_id = 0;
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Food & Beverages', 'Grocery', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Electronics', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Home, Furniture & Appliances', 'Domestic Goods;Furniture;Home Appliances', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Patio & Garden', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Digital Goods', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Services', 'Non-product services, Freelancing, etc.', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Books', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Movies & TV Shows', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Music & Vinyl', 'Musical instruments', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Apparel', 'Clothing, Shoes and Accessories; Jewelry and Watches; Fashion', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Pets', 'Domesticated Animals', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Toys & Games', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Baby', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Arts, Crafts, Sewing & Party Supplies', 'DIY & Handmade', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Stationery & Office Supplies', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Tools & Home Improvement', '', '') RETURNING id;"); // Hardware - Heating, cooling, flooring, paint, etc.
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Motor Vehicles, Automotive Parts & Accessories', 'Auto, Tires & Industrial', '') RETURNING id;");//category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Motor Vehicles, Automotive Parts & Accessories', 'Auto, Tires & Industrial', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Beauty & Personal Care', 'Cosmetics;Health, Beauty & Personal Care;Hygiene', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Drugs & Medications', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Sports & Outdoors', 'Sporting Equipment;Outdoors & Camping;Hiking;Hunting;Fishing;Biking', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Real Estate, Property & Housing', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Luggage & Travel', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Business, Industrial & Scientific', '', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Illegal', 'Banned and/or prohibited items', '') RETURNING id;");
category_id = database->get_integer("INSERT INTO categories (name, description, thumbnail) VALUES ('Miscellaneous', 'Others;Non-classified', '') RETURNING id;");
//category_id = database->get_integer("INSERT INTO categories (name, description) VALUES ('Collectables & Art', '') RETURNING id;");
//category_id = database->get_integer("INSERT INTO categories (name, description) VALUES ('', '') RETURNING id;");
// NOTE: Categories also act as tags to be used for filtering specific products
}
//-------------------------
database->execute("COMMIT;");
}
//----------------------------------------------------------------
std::string neroshop::Backend::getDatabaseHash() {
// Get contents from data.sqlite3 file
std::ifstream rfile (std::string("data.sqlite3").c_str(), std::ios::binary);
std::stringstream db_content;
db_content << rfile.rdbuf(); // dump file contents
rfile.close();
// Get SHA256sum of data.sqlite3 contents
std::string sha256sum;
Validator::generate_sha256_hash(db_content.str(), sha256sum);
std::cout << "sha256sum (data.sqlite3): " << sha256sum << std::endl;
return sha256sum; // database may have to be closed first in order to get the accurate hash
}
//----------------------------------------------------------------
//----------------------------------------------------------------
QVariantList neroshop::Backend::getCategoryList(bool sort_alphabetically) const {
// Do some database reading to fetch each category row (database reads do not require consensus)
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), (sort_alphabetically) ? "SELECT * FROM categories ORDER BY name ASC;" : "SELECT * FROM categories ORDER BY id ASC;", -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList category_list;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap category_object; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
//std::cout << column_value << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
if(i == 0) category_object.insert("id", QString::fromStdString(column_value).toInt());
if(i == 1) category_object.insert("name", QString::fromStdString(column_value));
if(i == 2) category_object.insert("description", QString::fromStdString(column_value));
if(i == 3) category_object.insert("thumbnail", QString::fromStdString(column_value));
//if(i == ) category_object.insert("", QString::fromStdString(column_value));
}
category_list.append(category_object);
}
sqlite3_finalize(stmt);
return category_list;
}
//----------------------------------------------------------------
int neroshop::Backend::getCategoryIdByName(const QString& category_name) const {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Execute sqlite3 statement
int category_id = database->get_integer_params("SELECT id FROM categories WHERE name = $1;", { category_name.toStdString() });
return category_id;
}
//----------------------------------------------------------------
int neroshop::Backend::getCategoryProductCount(int category_id) const {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Execute sqlite3 statement
int category_product_count = database->get_integer_params("SELECT COUNT(*) FROM products WHERE category_id = $1;", { std::to_string(category_id) });
return category_product_count;
}
//----------------------------------------------------------------
//----------------------------------------------------------------
QVariantList neroshop::Backend::registerProduct(const QString& name, const QString& description,
double weight, const QString& attributes,
const QString& product_code,
int category_id) const
{
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
QString product_uuid = QUuid::createUuid().toString();
product_uuid = product_uuid.remove("{").remove("}"); // remove brackets
std::string product_id = database->get_text_params("INSERT INTO products (uuid, name, description, weight, attributes, code, category_id) "
"VALUES ($1, $2, $3, $4, $5, $6, $7) "
"RETURNING uuid", { product_uuid.toStdString(), name.toStdString(), description.toStdString(), std::to_string(weight), attributes.toStdString(), product_code.toStdString(), std::to_string(category_id) });
if(product_id.empty()) return { false, "" };
if(product_code.isEmpty()) database->execute_params("UPDATE products SET code = NULL WHERE uuid = $1", { product_uuid.toStdString() });
if(attributes.isEmpty()) database->execute_params("UPDATE products SET attributes = NULL WHERE uuid = $1", { product_uuid.toStdString() });
return { true, QString::fromStdString(product_id) };
}
//----------------------------------------------------------------
void neroshop::Backend::uploadProductImage(const QString& product_id, const QString& filename) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
database->execute("BEGIN;");
// Read image from file and retrieve its contents
std::ifstream product_image_file(filename.toStdString(), std::ios::binary); // std::ios::binary is the same as std::ifstream::binary
if(!product_image_file.good()) {
std::cout << NEROSHOP_TAG "failed to load " << filename.toStdString() << std::endl;
database->execute("ROLLBACK;"); return;
}
product_image_file.seekg(0, std::ios::end);
size_t size = static_cast<int>(product_image_file.tellg()); // in bytes
// Limit product image size to 12582912 bytes (12 megabyte)
// Todo: Database cannot scale to billions of users if I am storing blobs so I'll have to switch to text later
if(size >= 12582912) {
neroshop::print("Product upload image cannot exceed 12 MB (twelve megabyte)", 1);
database->execute("ROLLBACK;"); return;
}
product_image_file.seekg(0);
std::vector<unsigned char> buffer(size);
if(!product_image_file.read(reinterpret_cast<char *>(&buffer[0]), size)) {
std::cout << NEROSHOP_TAG "error: only " << product_image_file.gcount() << " could be read";
database->execute("ROLLBACK;"); // abort transaction
return; // exit function
}
product_image_file.close();
// Store image in database as BLOB
std::string command = "INSERT INTO images (product_id, name, data) VALUES ($1, $2, $3);";
sqlite3_stmt * statement = nullptr;
int result = sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &statement, nullptr);
if(result != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
database->execute("ROLLBACK;"); return;
}
std::string product_uuid = product_id.toStdString();
result = sqlite3_bind_text(statement, 1, product_uuid.c_str(), product_uuid.length(), SQLITE_STATIC);
if(result != SQLITE_OK) {
neroshop::print("sqlite3_bind_text (arg: 1): " + std::string(sqlite3_errmsg(database->get_handle())), 1);
sqlite3_finalize(statement);
database->execute("ROLLBACK;"); return;
}
std::string product_image_filename = filename.toStdString();
result = sqlite3_bind_text(statement, 2, product_image_filename.c_str(), product_image_filename.length(), SQLITE_STATIC);
if(result != SQLITE_OK) {
neroshop::print("sqlite3_bind_text (arg: 2): " + std::string(sqlite3_errmsg(database->get_handle())), 1);
sqlite3_finalize(statement);
database->execute("ROLLBACK;"); return;
}
result = sqlite3_bind_blob(statement, 3, buffer.data(), size, SQLITE_STATIC);
if(result != SQLITE_OK) {
neroshop::print("sqlite3_bind_blob (arg: 3): " + std::string(sqlite3_errmsg(database->get_handle())), 1);
sqlite3_finalize(statement);
database->execute("ROLLBACK;"); return;// nullptr;
}
result = sqlite3_step(statement);
if (result != SQLITE_DONE) {
neroshop::print("sqlite3_step: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
}
sqlite3_finalize(statement);
database->execute("COMMIT;");
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getProductImages(const QString& product_id) {
// Do some database reading to fetch each category row (database reads do not require consensus)
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
std::string command = "SELECT id, name FROM images WHERE product_id = $1";////std::string command = "SELECT id, name, data FROM images WHERE product_id = $1";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Bind product_id to TEXT
std::string product_uuid = product_id.toStdString();
if(sqlite3_bind_text(stmt, 1, product_uuid.c_str(), product_uuid.length(), SQLITE_STATIC) != SQLITE_OK) {
neroshop::print("sqlite3_bind_text (arg: 1): " + std::string(sqlite3_errmsg(database->get_handle())), 1);
sqlite3_finalize(stmt);
return {};//database->execute("ROLLBACK;"); return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList product_image_list;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap product_image_object; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
if(i == 0) product_image_object.insert("image_id", QString::fromStdString(column_value).toInt());
if(i == 1) product_image_object.insert("name", QString::fromStdString(column_value));
////if(i == 2) product_image_object.insert("data", QString::fromStdString(column_value));
}
product_image_list.append(product_image_object);
}
return product_image_list;
}
//----------------------------------------------------------------
//----------------------------------------------------------------
int neroshop::Backend::getProductStarCount(const QString& product_id) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Get total number of star ratings for a specific product
unsigned int ratings_count = database->get_integer_params("SELECT COUNT(*) FROM product_ratings WHERE product_id = $1", { product_id.toStdString() });
return ratings_count;
}
//----------------------------------------------------------------
int neroshop::Backend::getProductStarCount(const QString& product_id, int star_number) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Get total number of N star ratings for a specific product
if(star_number > 5) star_number = 5;
if(star_number < 1) star_number = 1;
unsigned int star_count = database->get_integer_params("SELECT COUNT(stars) FROM product_ratings WHERE product_id = $1 AND stars = $2", { product_id.toStdString(), std::to_string(star_number) });
return star_count;
}
//----------------------------------------------------------------
float neroshop::Backend::getProductAverageStars(const QString& product_id) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Get number of star ratings for a specific product
unsigned int total_star_ratings = database->get_integer_params("SELECT COUNT(*) FROM product_ratings WHERE product_id = $1", { product_id.toStdString() });
if(total_star_ratings == 0) { /*neroshop::print("This item has no star ratings", 2);*/ return 0.0f; }
// Get number of 1, 2, 3, 4, and 5 star_ratings
int one_star_count = database->get_integer_params("SELECT COUNT(stars) FROM product_ratings WHERE product_id = $1 AND stars = $2", { product_id.toStdString(), std::to_string(1) });
int two_star_count = database->get_integer_params("SELECT COUNT(stars) FROM product_ratings WHERE product_id = $1 AND stars = $2", { product_id.toStdString(), std::to_string(2) });
int three_star_count = database->get_integer_params("SELECT COUNT(stars) FROM product_ratings WHERE product_id = $1 AND stars = $2", { product_id.toStdString(), std::to_string(3) });
int four_star_count = database->get_integer_params("SELECT COUNT(stars) FROM product_ratings WHERE product_id = $1 AND stars = $2", { product_id.toStdString(), std::to_string(4) });
int five_star_count = database->get_integer_params("SELECT COUNT(stars) FROM product_ratings WHERE product_id = $1 AND stars = $2", { product_id.toStdString(), std::to_string(5) });
// Now calculate the average stars
float average_stars = (
(1 * static_cast<float>(one_star_count)) +
(2 * static_cast<float>(two_star_count)) +
(3 * static_cast<float>(three_star_count)) +
(4 * static_cast<float>(four_star_count)) +
(5 * static_cast<float>(five_star_count))) / total_star_ratings;
return average_stars;
}
//----------------------------------------------------------------
//----------------------------------------------------------------
QString neroshop::Backend::getDisplayNameById(const QString& user_id) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
std::string display_name = database->get_text_params("SELECT name FROM users WHERE monero_address = $1", { user_id.toStdString() });
return QString::fromStdString(display_name);
}
//----------------------------------------------------------------
//----------------------------------------------------------------
int neroshop::Backend::getCartMaximumItems() {
return neroshop::Cart::get_max_items();
}
//----------------------------------------------------------------
int neroshop::Backend::getCartMaximumQuantity() {
return neroshop::Cart::get_max_quantity();
}
//----------------------------------------------------------------
int neroshop::Backend::getStockAvailable(const QString& product_id) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
int quantity = database->get_integer_params("SELECT quantity FROM listings WHERE product_id = $1 AND quantity > 0", { product_id.toStdString() });
return quantity;
}
//----------------------------------------------------------------
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListings() {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id GROUP BY images.product_id;";//WHERE stock_qty > 0;"; // image.product_id must be unique in this field to prevent duplicate listings!
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListingsByCategory(int category_id) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
//std::string command = "SELECT DISTINCT * FROM listings ORDER BY date ASC;";// LIMIT $1;";//WHERE stock_qty > 0;";
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id WHERE category_id = $1 GROUP BY images.product_id;";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Bind value to parameter arguments
if(sqlite3_bind_int(stmt, 1, category_id) != SQLITE_OK) {
neroshop::print("sqlite3_bind_int: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
sqlite3_finalize(stmt);
return {};//database->execute("ROLLBACK;"); return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListingsByMostRecent() {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
//std::string command = "SELECT DISTINCT * FROM listings ORDER BY date ASC;";// LIMIT $1;";//WHERE stock_qty > 0;";
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id GROUP BY images.product_id ORDER BY date DESC;";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListingsByMostRecentLimit(int limit) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
//std::string command = "SELECT DISTINCT * FROM listings ORDER BY date ASC;";// LIMIT $1;";//WHERE stock_qty > 0;";
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id GROUP BY images.product_id ORDER BY date DESC LIMIT $1;";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Limit the number of item results (rows)
if(sqlite3_bind_int(stmt, 1, limit) != SQLITE_OK) {
neroshop::print("sqlite3_bind_int: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
sqlite3_finalize(stmt);
return {};//database->execute("ROLLBACK;"); return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListingsByOldest() {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
//std::string command = "SELECT DISTINCT * FROM listings ORDER BY date ASC;";// LIMIT $1;";//WHERE stock_qty > 0;";
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id GROUP BY images.product_id ORDER BY date ASC;";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListingsByAlphabeticalOrder() {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
//std::string command = "SELECT DISTINCT * FROM listings ORDER BY date ASC;";// LIMIT $1;";//WHERE stock_qty > 0;";
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id GROUP BY images.product_id ORDER BY products.name COLLATE NOCASE ASC;";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListingsByPriceLowest() {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
//std::string command = "SELECT DISTINCT * FROM listings ORDER BY date ASC;";// LIMIT $1;";//WHERE stock_qty > 0;";
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id GROUP BY images.product_id ORDER BY price ASC;";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getListingsByPriceHighest() {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
//std::string command = "SELECT DISTINCT * FROM listings ORDER BY date ASC;";// LIMIT $1;";//WHERE stock_qty > 0;";
std::string command = "SELECT DISTINCT * FROM listings JOIN products ON products.uuid = listings.product_id JOIN images ON images.product_id = listings.product_id GROUP BY images.product_id ORDER BY price DESC;";
sqlite3_stmt * stmt = nullptr;
// Prepare (compile) statement
if(sqlite3_prepare_v2(database->get_handle(), command.c_str(), -1, &stmt, nullptr) != SQLITE_OK) {
neroshop::print("sqlite3_prepare_v2: " + std::string(sqlite3_errmsg(database->get_handle())), 1);
return {};
}
// Check whether the prepared statement returns no data (for example an UPDATE)
if(sqlite3_column_count(stmt) == 0) {
neroshop::print("No data found. Be sure to use an appropriate SELECT statement", 1);
return {};
}
QVariantList catalog_array;
// Get all table values row by row
while(sqlite3_step(stmt) == SQLITE_ROW) {
QVariantMap listing; // Create an object for each row
for(int i = 0; i < sqlite3_column_count(stmt); i++) {
std::string column_value = (sqlite3_column_text(stmt, i) == nullptr) ? "NULL" : reinterpret_cast<const char *>(sqlite3_column_text(stmt, i));////if(sqlite3_column_text(stmt, i) == nullptr) {throw std::runtime_error("column is NULL");}
////std::cout << column_value << " (" << i << ")" << std::endl;//std::cout << sqlite3_column_name(stmt, i) << std::endl;
// listings table
if(i == 0) listing.insert("listing_uuid", QString::fromStdString(column_value));
if(i == 1) listing.insert("product_id", QString::fromStdString(column_value));
if(i == 2) listing.insert("seller_id", QString::fromStdString(column_value));
if(i == 3) listing.insert("quantity", QString::fromStdString(column_value).toInt());
if(i == 4) listing.insert("price", QString::fromStdString(column_value).toDouble());
if(i == 5) listing.insert("currency", QString::fromStdString(column_value));
if(i == 6) listing.insert("condition", QString::fromStdString(column_value));
if(i == 7) listing.insert("location", QString::fromStdString(column_value));
if(i == 8) listing.insert("date", QString::fromStdString(column_value));
// products table
if(i == 9) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 10) listing.insert("product_name", QString::fromStdString(column_value));
if(i == 11) listing.insert("product_description", QString::fromStdString(column_value));
if(i == 12) listing.insert("weight", QString::fromStdString(column_value).toDouble());
if(i == 13) listing.insert("product_attributes", QString::fromStdString(column_value));
if(i == 14) listing.insert("product_code", QString::fromStdString(column_value));
if(i == 15) listing.insert("product_category_id", QString::fromStdString(column_value).toInt());
// images table
//if(i == 16) listing.insert("image_id", QString::fromStdString(column_value));
//if(i == 17) listing.insert("product_uuid", QString::fromStdString(column_value));
if(i == 18) listing.insert("product_image_file", QString::fromStdString(column_value));
//if(i == 19) listing.insert("product_image_data", QString::fromStdString(column_value));
}
catalog_array.append(listing);
}
sqlite3_finalize(stmt);
return catalog_array;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getNodeListDefault(const QString& coin) const {
QVariantList node_list;
std::string network_type = neroshop::Script::get_string(neroshop::lua_state, "neroshop.monero.daemon.network_type");
std::vector<std::string> node_table = neroshop::Script::get_table_string(neroshop::lua_state, "neroshop." + coin.toStdString() + ".nodes." + network_type); // Get monero nodes from settings.lua////std::cout << "lua_query: " << "neroshop." + coin.toStdString() + ".nodes." + network_type << std::endl;
for(auto strings : node_table) {
node_list << QString::fromStdString(strings);
}
return node_list;
}
//----------------------------------------------------------------
QVariantList neroshop::Backend::getNodeList(const QString& coin) const {
const QUrl url(QStringLiteral("https://monero.fail/health.json"));
QVariantList node_list;
QString coin_lower = coin.toLower(); // make coin name lowercase
QNetworkAccessManager manager;
QEventLoop loop;
QObject::connect(&manager, &QNetworkAccessManager::finished, &loop, &QEventLoop::quit);
auto reply = manager.get(QNetworkRequest(url));
loop.exec();
QJsonParseError error;
const auto json_doc = QJsonDocument::fromJson(reply->readAll(), &error);
// Use fallback monero node list if we fail to get the nodes from the url
if (error.error != QJsonParseError::NoError) {
neroshop::print("Error reading json from " + url.toString().toStdString() + "\nUsing default nodes as fallback", 2);
return getNodeListDefault(coin_lower);
}
// Get monero nodes from the JSON
QJsonObject root_obj = json_doc.object(); // {}
QJsonObject coin_obj = root_obj.value(coin_lower).toObject(); // "monero": {} // "wownero": {}
QJsonObject clearnet_obj = coin_obj.value("clear").toObject(); // "clear": {} // "onion": {}, "web_compatible": {}
// Loop through monero nodes (clearnet)
foreach(const QString& key, clearnet_obj.keys()) {//for (const auto monero_nodes : clearnet_obj) {
QJsonObject monero_node_obj = clearnet_obj.value(key).toObject();//QJsonObject monero_node_obj = monero_nodes.toObject();
QVariantMap node_object; // Create an object for each row
if(key.contains("38081") || key.contains("38089")) { // Temporarily fetch only stagenet nodes (TODO: change to mainnet port upon release)
node_object.insert("address", key);
node_object.insert("available", monero_node_obj.value("available").toBool());//std::cout << "available: " << monero_node_obj.value("available").toBool() << "\n";
////node_object.insert("", );//////std::cout << ": " << monero_node_obj.value("checks").toArray() << "\n";
node_object.insert("datetime_checked", monero_node_obj.value("datetime_checked").toString());//std::cout << "datetime_checked: " << monero_node_obj.value("datetime_checked").toString().toStdString() << "\n";
node_object.insert("datetime_entered", monero_node_obj.value("datetime_entered").toString());//std::cout << "datetime_entered: " << monero_node_obj.value("datetime_entered").toString().toStdString() << "\n";
node_object.insert("datetime_failed", monero_node_obj.value("datetime_failed").toString());//std::cout << "datetime_failed: " << monero_node_obj.value("datetime_failed").toString().toStdString() << "\n";
node_object.insert("last_height", monero_node_obj.value("last_height").toInt());//std::cout << "last_height: " << monero_node_obj.value("last_height").toInt() << "\n";
node_list.append(node_object); // Add node object to the node list
}
}
return node_list;
}
//----------------------------------------------------------------
// Todo: use QProcess to check if monero daemon is running
bool neroshop::Backend::isWalletDaemonRunning() const {
int monerod = Process::get_process_by_name("monerod");
if(monerod == -1) { return false; }
std::cout << "\033[1;90;49m" << "monerod is running (ID:" << monerod << ")\033[0m" << std::endl;
return true;
}
//----------------------------------------------------------------
//----------------------------------------------------------------
// TODO: replace function return type with enum
QVariantList neroshop::Backend::validateDisplayName(const QString& display_name) const {
// username (will appear only in lower-case letters within the app)
std::string username = display_name.toStdString();
// Empty display names are acceptable
if(display_name.isEmpty()) return { true, "" };
// If display name is set, make sure it is at least 2 characters short (min_user_length=2)
unsigned int min_user_length = 2;
if(username.length() < min_user_length) {
std::string message = "Your username must be at least " + std::to_string(min_user_length) + " characters in length";
return { false, QString::fromStdString(message) };
}
// make sure username is at least 30 characters long (max_user_length=30)
unsigned int max_user_length = 30; // what if a company has a long name? :o // also consider the textbox's max-length
if(username.length() > max_user_length) {
std::string message = "Your username must not exceed " + std::to_string(max_user_length) + " characters in length";
return { false, QString::fromStdString(message) };
}
for(int i = 0; i < username.length(); i++)
{
// make sure username does not contain any spaces //std::cout << username[i] << " - char\n";
if(isspace(username[i])) {
std::string message = "Your username cannot contain any spaces";
return { false, QString::fromStdString(message) };
}
// make sure username can only contain letters, numbers, and these specific symbols: a hyphen, underscore, and period (-,_,.) //https://stackoverflow.com/questions/39819830/what-are-the-allowed-character-in-snapchat-username#comment97381763_41959421
// symbols like @,#,$,etc. are invalid
if(!isalnum(username[i])) {
if(username[i] != '-'){
if(username[i] != '_'){
if(username[i] != '.'){
std::string message = "Your username cannot contain any symbols except (-, _, .)";// + username[i];
return { false, QString::fromStdString(message) };
}}}
}
}
// make sure username begins with a letter (username cannot start with a symbol or number)
char first_char = username.at(username.find_first_of(username));
if(!isalpha(first_char)) {
std::string message = "Your username must begin with a letter";// + first_char;
return { false, QString::fromStdString(message) };
}
// make sure username does not end with a symbol (username can end with a number, but NOT with a symbol)
char last_char = username.at(username.find_last_of(username));
if(!isalnum(last_char)) { //if(last_char != '_') { // underscore allowed at end of username? :o
std::string message = "Your username must end with a letter or number";// + last_char;
return { false, QString::fromStdString(message) };
}
// the name guest is reserved for guests only, so it cannot be used by any other user
if(neroshop::string::lower(username) == "guest") {
std::string message = "The name \"Guest\" is reserved for guests ONLY";
return { false, QString::fromStdString(message) };
}
// Check database to see if display name is available
auto name_check_result = checkDisplayName(display_name);
if(!name_check_result[0].toBool()) {////if(!Validator::validate_display_name(display_name.toStdString())) return false;
bool boolean_result = name_check_result[0].toBool();
QString message_result = name_check_result[1].toString();
return { boolean_result, message_result };
}
return { true, "" };
}
//----------------------------------------------------------------
// TODO: replace function return type with enum
QVariantList neroshop::Backend::checkDisplayName(const QString& display_name) const {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database->table_exists("users")) { return {true, ""}; }
std::string name = database->get_text_params("SELECT name FROM users WHERE name = $1 COLLATE NOCASE;", { display_name.toStdString() });
if(name.empty()) return { true, "Display name is available for use" };// Name is not taken which means that the user is good to go!
// Empty display names are acceptable
bool is_name_empty = display_name.isEmpty();
if(is_name_empty) return { true, "No display name set" };
// Note: both database and input display names are converted to lowercase strings and then compared within the app
std::string name_lowercase = QString::fromStdString(name).toLower().toStdString();//QString::fromUtf8(name.data(), name.size()).toLower();
std::string display_name_lowercase = display_name.toLower().toStdString();
if(name_lowercase == display_name_lowercase) {
return { false, "This username is already taken" };////result_list << false << QString("This username is already taken");return result_list;
}
return { true, "" };
}
//----------------------------------------------------------------
// TODO: replace function return type with enum
QVariantList neroshop::Backend::registerUser(WalletController* wallet_controller, const QString& display_name, UserController * user_controller) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Validate display name
auto name_validation_result = validateDisplayName(display_name);
if(!name_validation_result[0].toBool()) {
bool boolean_result = name_validation_result[0].toBool();
QString message_result = name_validation_result[1].toString();
return { boolean_result, message_result };
}
// Get wallet primary address and check its validity
std::string primary_address = wallet_controller->getPrimaryAddress().toStdString();//neroshop::print("Primary address: \033[1;33m" + primary_address + "\033[1;37m\n");
if(!wallet_controller->getWallet()->is_valid_address(primary_address)) {
return { false, "Invalid monero address" };
}
// Store login credentials in database
// Todo: make this command (DB entry) a client request that the server must respond to and the consensus must agree with
// Note: Multiple users cannot have the same display_name. Each display_name must be unique!
std::string user_id = database->get_text_params("INSERT INTO users(name, monero_address) VALUES($1, $2) RETURNING monero_address;", { display_name.toStdString(), primary_address });//int user_id = database->get_integer_params("INSERT INTO users(name, monero_address) VALUES($1, $2) RETURNING id;", { display_name.toStdString(), primary_address.toStdString() });
if(user_id.empty()) { return { false, "Account registration failed (due to database error)" }; }//if(user_id == 0) { return { false, "Account registration failed (due to database error)" }; }
// Create cart for user
QString cart_uuid = QUuid::createUuid().toString();
cart_uuid = cart_uuid.remove("{").remove("}"); // remove brackets
database->execute_params("INSERT INTO cart (uuid, user_id) VALUES ($1, $2)", { cart_uuid.toStdString(), user_id });
// initialize user obj (Todo: make a seperate class headers + source files for User)
std::unique_ptr<neroshop::User> seller(neroshop::Seller::on_login(*wallet_controller->getWallet()));
user_controller->_user = std::move(seller);
if (user_controller->getUser() == nullptr) {
return {false, "user is NULL"};
}
emit user_controller->userChanged();
emit user_controller->userLogged();
//user_controller->rateItem("3c20978b-a543-4c58-bc68-5f900027796f", 3, "This product is aiight");
// Display registration message
neroshop::print(((!display_name.isEmpty()) ? "Welcome to neroshop, " : "Welcome to neroshop") + display_name.toStdString(), 4);
return { true, "" };
}
//----------------------------------------------------------------
bool neroshop::Backend::loginWithWalletFile(WalletController* wallet_controller, const QString& path, const QString& password, UserController * user_controller) {
neroshop::Client * client = neroshop::Client::get_main_client();
client->write("Mr server, I am logging in with my wallet file\n");
client->write("Mr server, I am logging in with my wallet file 2\n");
client->write("Mr server, I am logging in with my wallet file 3\n");
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Open wallet file
std::packaged_task<bool(void)> open_wallet_task([wallet_controller, path, password]() -> bool {
if(!wallet_controller->open(path, password)) {
////throw std::runtime_error("Invalid password or wallet network type");
return false;
}
return true;
});
std::future<bool> future_result = open_wallet_task.get_future();
// move the task (function) to a separate thread to prevent blocking of the main thread
std::thread worker(std::move(open_wallet_task));
worker.detach(); // join may block but detach won't
bool wallet_opened = future_result.get();
if(!wallet_opened) return false;
// Get the primary address
std::string primary_address = wallet_controller->getPrimaryAddress().toStdString();
// Check database to see if user key (hash of primary address) exists
bool user_found = database->get_integer_params("SELECT EXISTS(SELECT * FROM users WHERE monero_address = $1)", { primary_address });
// If user key is not found in the database, then create one. This is like registering for an account
if(!user_found) {
// In reality, this function will return false if user key is not registered in the database
neroshop::print("user not found in database. Please try again or register", 1);
wallet_controller->close();
return false;
}
// Save user information in memory
std::string display_name = database->get_text_params("SELECT name FROM users WHERE monero_address = $1", { primary_address });
std::unique_ptr<neroshop::User> seller(neroshop::Seller::on_login(*wallet_controller->getWallet()));
user_controller->_user = std::move(seller);
if(user_controller->getUser() == nullptr) {
return false;//{false, "user is NULL"};
}
emit user_controller->userChanged();
emit user_controller->userLogged();
// Display message
neroshop::print("Welcome back, user " + ((!display_name.empty()) ? (display_name + " (id: " + primary_address + ")") : primary_address), 4);
return true;
}
//----------------------------------------------------------------
bool neroshop::Backend::loginWithMnemonic(WalletController* wallet_controller, const QString& mnemonic, UserController * user_controller) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Initialize monero wallet with existing wallet mnemonic
if(!wallet_controller->restoreFromMnemonic(mnemonic)) {
throw std::runtime_error("Invalid mnemonic or wallet network type");
return false;
}
// Get the primary address
std::string primary_address = wallet_controller->getPrimaryAddress().toStdString();
// Check database to see if user key (hash of primary address) exists
bool user_key_found = database->get_integer_params("SELECT EXISTS(SELECT * FROM users WHERE monero_address = $1)", { primary_address });
// If user key is not found in the database, then create one. This is like registering for an account
if(!user_key_found) {
// In reality, this function will return false if user key is not registered in the database
neroshop::print("user key not found in database. Please try again or register", 1);
wallet_controller->close();
return false;
}
// Save user information in memory
int user_id = database->get_integer_params("SELECT id FROM users WHERE monero_address = $1", { primary_address });
// Display message
std::string display_name = database->get_text_params("SELECT name FROM users WHERE monero_address = $1", { primary_address });
neroshop::print("Welcome back, user " + ((!display_name.empty()) ? (display_name + " (id: " + primary_address + ")") : primary_address), 4);
return true;
}
//----------------------------------------------------------------
bool neroshop::Backend::loginWithKeys(WalletController* wallet_controller, UserController * user_controller) {
/*
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
// Get the wallet from the wallet controller
neroshop::Wallet * wallet = wallet_controller->getWallet();
// Initialize monero wallet with existing wallet mnemonic
std::string primary_address;
std::string secret_view_key;
std::string secret_spend_key;
std::cout << "Please enter your primary address:\n";
std::getline(std::cin, primary_address);
std::cout << "Please enter your secret view key:\n";
std::getline(std::cin, secret_view_key);
std::cout << "Please enter your secret spend key (optional):\n";
std::getline(std::cin, secret_spend_key);
// todo: allow user to specify a custom location for the wallet keyfile or use a default location
wallet_controller->restoreFromKeys(primary_address, secret_view_key, secret_spend_key);
// Get the hash of the primary address
std::string user_auth_key;// = neroshop::algo::sha256(primary_address);
Validator::generate_sha256_hash(primary_address, user_auth_key); // temp
neroshop::print("Primary address: \033[1;33m" + primary_address + "\033[1;37m\nSHA256 hash: " + user_auth_key);
//$ echo -n "528qdm2pXnYYesCy5VdmBneWeaSZutEijFVAKjpVHeVd4unsCSM55CjgViQsK9WFNHK1eZgcCuZ3fRqYpzKDokqSKp4yp38" | sha256sum
// Check database to see if user key (hash of primary address) exists
bool user_key_found = database->get_integer_params("SELECT EXISTS(SELECT * FROM users WHERE key = $1)", { user_auth_key });
// If user key is not found in the database, then create one. This is like registering for an account
if(!user_key_found) {
// In reality, this function will return false if user key is not registered in the database
neroshop::print("user key not found in database. Please try again or register", 1);
wallet_controller->close();
return false;
}
// Save user information in memory
int user_id = database->get_integer_params("SELECT id FROM users WHERE key = $1", { user_auth_key });
// Display message
std::string display_name = database->get_text_params("SELECT name FROM users WHERE monero_address = $1", { primary_address });
neroshop::print("Welcome back, user " + ((!display_name.empty()) ? (display_name + " (id: " + primary_address + ")") : primary_address), 4);
return true;
*/
return false;
}
//----------------------------------------------------------------
bool neroshop::Backend::loginWithHW(WalletController* wallet_controller, UserController * user_controller) {
return false;
}
//----------------------------------------------------------------
//----------------------------------------------------------------
void neroshop::Backend::startServerDaemon() {
#ifdef Q_OS_WIN
QString program = "neromon.exe";
#else
QString program = "./neromon";
#endif
// Note: If the calling process exits, the detached process will continue to run unaffected.
qint64 pid = -1;
bool success = QProcess::startDetached(program, {}, QString(), &pid);
if(!success) { throw std::runtime_error("neroshop daemon process could not be started"); }
std::cout << "\033[35mneromon started (pid: " << pid << ")\033[0m\n";
}
//----------------------------------------------------------------
void neroshop::Backend::waitForServerDaemon() {
::sleep(2);
}
//----------------------------------------------------------------
void neroshop::Backend::connectToServerDaemon() {
// We will use JSON to write the request data that will sent to the server
neroshop::Client * client = neroshop::Client::get_main_client();
int client_port = 40441;
std::string client_ip = "0.0.0.0";//"localhost";//0.0.0.0 means anyone can connect to your server
if(!client->connect(client_port, client_ip)) {
// free process
////delete server_process; // kills process
////server_process = nullptr;
// exit application
exit(0);
} else std::cout << client->read() << std::endl;
}
//----------------------------------------------------------------
//----------------------------------------------------------------
#include <nlohmann/json.hpp>
void neroshop::Backend::testWriteJson(const std::vector<std::string>& args) {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
/*QJsonObject jsonObject; // base Object (required)
//QJsonObject json_attributes_object; // subObject (optional)
QJsonArray json_color_array; // An array of colors
json_color_array.insert(json_color_array.size(), QJsonValue("red"));
json_color_array.insert(json_color_array.size(), QJsonValue("green"));
json_color_array.insert(json_color_array.size(), QJsonValue("blue"));
QJsonArray json_size_array; // An array of sizes
json_size_array.insert(json_size_array.size(), QJsonValue("XS"));
json_size_array.insert(json_size_array.size(), QJsonValue("S"));
json_size_array.insert(json_size_array.size(), QJsonValue("M"));
json_size_array.insert(json_size_array.size(), QJsonValue("L"));
json_size_array.insert(json_size_array.size(), QJsonValue("XL"));
//jsonObject.insert
jsonObject.insert(QString("weight"), QJsonValue(12.5));
jsonObject.insert(QString("color"), json_color_array);
jsonObject.insert(QString("size"), json_size_array);
//jsonObject.insert("attributes", json_attributes_object);
// Convert JSON to QString
QJsonDocument doc(jsonObject);
QString strJson = doc.toJson(QJsonDocument::Compact); // https://doc.qt.io/qt-6/qjsondocument.html#JsonFormat-enum
// Display JSON as string
std::cout << strJson.toStdString() << std::endl;*/
// Okay, so we (the server) have received a request from the client
std::string request = "{\n \"id\": \"2127323435\",\n \"jsonrpc\": \"2.0\",\n \"method\": \"query\",\n \"params\": {\n \"sql\": \"SELECT * FROM products WHERE condition = 'New' AND weight <= 0.0;\"\n }\n}";//"{\"id\":\"2127323435\",\"jsonrpc\":\"2.0\",\"method\":\"query\",\"params\":{\"1\":\"New\",\"2\":\"0.0\",\"count\":2,\"sql\":\"SELECT * FROM products WHERE condition = $1 AND weight <= $2;\"}}";
// First process the request ---------------------------------------
QJsonObject response_object; // JSON-RPC Response object
QJsonParseError parse_error;
const auto json_doc = QJsonDocument::fromJson(QString::fromStdString(request).toUtf8(), &parse_error);
if (parse_error.error != QJsonParseError::NoError) {
neroshop::print("Error parsing client request", 1);
response_object.insert(QString("jsonrpc"), QJsonValue("2.0"));
// "result" MUST NOT exist if there was an error invoking the method
QJsonObject error_object; // "error" MUST be an Object as defined in section 5.1
error_object.insert(QString("code"), QJsonValue("-32700"));
error_object.insert(QString("message"), QJsonValue("Parse error"));
QString data;
switch(parse_error.error) {
case QJsonParseError::UnterminatedObject:
data = "Unterminated object";
break;
case QJsonParseError::MissingNameSeparator:
data = "Missing name separator";
break;
case QJsonParseError::UnterminatedArray:
data = "Unterminated array";
break;
case QJsonParseError::MissingValueSeparator:
data = "Missing value separator";
break;
case QJsonParseError::IllegalValue:
data = "Illegal value";
break;
case QJsonParseError::TerminationByNumber:
data = "Termination by number";
break;
case QJsonParseError::IllegalNumber:
data = "Illegal number";
break;
case QJsonParseError::IllegalEscapeSequence:
data = "Illegal escape sequence";
break;
case QJsonParseError::IllegalUTF8String:
data = "Illegal utf8 string";
break;
case QJsonParseError::UnterminatedString:
data = "Unterminated string";
break;
case QJsonParseError::MissingObject:
data = "Missing object";
break;
case QJsonParseError::DeepNesting:
data = "Deep nesting";
break;
case QJsonParseError::DocumentTooLarge:
data = "Document too large";
break;
case QJsonParseError::GarbageAtEnd:
data = "Garbage at end";
break;
}
if(!data.isEmpty()) error_object.insert(QString("data"), QJsonValue(data)); // additional information about the error which may be omitted
response_object.insert(QString("error"), QJsonValue(error_object));
response_object.insert(QString("id"), QJsonValue(QJsonValue::Null)); // "id" MUST be Null if there was an error in detecting the id in the Request object (e.g. Parse error/Invalid Request)
// TODO: return a response with an error_object
std::cout << "\033[91m" << QJsonDocument(response_object).toJson().toStdString() << "\033[0m\n";
return;
}
std::cout << "Request received:\n\033[33m" << json_doc.toJson().toStdString() << "\033[0m\n";
QJsonObject request_object = json_doc.object();
// Retrieve keys and values from request_object
QJsonValue jsonrpc_version = request_object.value("jsonrpc");
assert(jsonrpc_version.isString());
assert(jsonrpc_version.toString() == "2.0");
QJsonValue method = request_object.value("method");
assert(method.isString());
QJsonValue params = request_object.value("params"); // "params" MAY be omitted
QJsonValue id = request_object.value("id");
if(id.isUndefined()) {
// a request object without an "id" member is assumed to be NOTIFICATION ...
// a Notification signifies the Client's lack of interest in the corresponding Response object, and as such no Response object needs to be returned to the client. The Server MUST NOT reply to a Notification, including those that are within a batch request.
// exit function, with an empty string object perhaps
//return "";
}
// Execute the request (run function and get return value)
if(params.isObject()) {//if(params.isArray()) { //if(params.isUndefined()) {
if(method.toString() == "query") { std::cout << "method= query\n";
QJsonValue sql = params.toObject().value("sql");
assert(sql.isString());
std::vector<std::string> args;
QJsonValue arg_count = params.toObject().value("count");
if(!arg_count.isUndefined()) {
for(int index = 0; index < arg_count.toInt(); index++) {
QJsonValue argument = params.toObject().value(QString::fromStdString(std::to_string(index + 1)));
assert(argument.isString());
std::cout << argument.toString().toStdString() << " (arg: " << (index + 1) << ")" << std::endl;
args.push_back(argument.toString().toStdString()); // Store arguments in std::vector
}
assert(args.size() == arg_count.toInt()); // Make sure the number of args is the same as the count
// TODO: check whether sqlite method whether its a select or non-select statement
// select statements should use database->get_*params functions
database->execute_params(sql.toString().toStdString(), args); // Execute query with n args
}
else if(arg_count.isUndefined()) { // No args
// TODO: check whether sqlite method whether its a select or non-select statement
// select statements should use database->get_* functions
database->execute(sql.toString().toStdString()); // Execute query with zero args
}
}
//-------------------------------------------------------
else { std::cout << "method= " << method.toString().toStdString() << "\n";
//params.value("");
}
}
// After execution, get the result then return it as a JSON-RPC response message
////response_object.insert(QString("jsonrpc"), QJsonValue("2.0"));
/*
response_object.insert(QString("jsonrpc"), QJsonValue("2.0"));
response_object.insert(QString("result"), QJsonValue()); // "result" MUST NOT exist if there was an error invoking the method.
QJsonObject error_object; // "error" MUST be an Object as defined in section 5.1
//error_object.insert(QString("code"), QJsonValue());
//error_object.insert(QString("message"), QJsonValue());
//error_object.insert(QString("data"), QJsonValue());
response_object.insert(QString("error"), QJsonValue(error_object)); // "error" MUST NOT exist if there was no error triggered during invocation.
response_object.insert(QString("id"), QJsonValue(QString::fromStdString(random_id))); // "id" MUST be the same as the request object's id
// Convert JSON to string then display it (for debugging purposes)
QJsonDocument json_doc(response_object);
QString json_str = json_doc.toJson();////(QJsonDocument::Compact);
std::cout << json_str.toStdString() << "\n";*/
/*std::string command = "SELECT * FROM users;";
nlohmann::json json;
json["jsonrpc"] = "2.0";
json["method"] = "select";////get_method_type(command);
json["params"]["sql"] = command;
json["params"]["count"] = args.size();
for(int index = 0; index < args.size(); index++) {
std::string arg_key = std::to_string(index + 1);
std::string arg_value = args[index];
json["params"][arg_key] = arg_value;
}
// Generate random number for id (id can be either a string or an integer or null which is not recommended)
std::random_device rd;
std::mt19937 gen(rd());
std::uniform_int_distribution<> distr(1, 9);
std::string random_id;
for(int n = 0; n < 10; ++n) {
int random_integer = distr(gen);
random_id = random_id + std::to_string(random_integer);
}
json["id"] = random_id;
// Dump JSON to string
std::string request = json.dump(4);
#ifdef NEROSHOP_DEBUG
std::cout << request << std::endl;
#endif*/
}
//----------------------------------------------------------------
void neroshop::Backend::testfts5() {
neroshop::db::Sqlite3 * database = neroshop::get_database();
if(!database) throw std::runtime_error("database is NULL");
database->execute("CREATE VIRTUAL TABLE email USING fts5(sender, title, body);");
}
//----------------------------------------------------------------
//----------------------------------------------------------------