// file : mod/database-module.cxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #include <mod/database-module.hxx> #include <odb/database.hxx> #include <odb/exceptions.hxx> #include <odb/transaction.hxx> #include <libbrep/build-package.hxx> #include <libbrep/build-package-odb.hxx> #include <mod/database.hxx> #include <mod/module-options.hxx> namespace brep { using namespace odb::core; // While currently the user-defined copy constructor is not required (we // don't need to deep copy nullptr's), it is a good idea to keep the // placeholder ready for less trivial cases. // database_module:: database_module (const database_module& r) : handler (r), retry_ (r.retry_), package_db_ (r.initialized_ ? r.package_db_ : nullptr), build_db_ (r.initialized_ ? r.build_db_ : nullptr) { } void database_module:: init (const options::package_db& o, size_t retry) { package_db_ = shared_database (o.package_db_user (), o.package_db_role (), o.package_db_password (), o.package_db_name (), o.package_db_host (), o.package_db_port (), o.package_db_max_connections ()); retry_ = retry_ < retry ? retry : retry_; } void database_module:: init (const options::build_db& o, size_t retry) { build_db_ = shared_database (o.build_db_user (), o.build_db_role (), o.build_db_password (), o.build_db_name (), o.build_db_host (), o.build_db_port (), o.build_db_max_connections ()); retry_ = retry_ < retry ? retry : retry_; } bool database_module:: handle (request& rq, response& rs, log& l) try { return handler::handle (rq, rs, l); } catch (const odb::recoverable& e) { if (retry_-- > 0) { HANDLER_DIAG; l1 ([&]{trace << e << "; " << retry_ + 1 << " retries left";}); throw retry (); } throw; } void database_module:: update_tenant_service_state ( const connection_ptr& conn, const string& tid, const function<optional<string> (const tenant_service&)>& f) { assert (f != nullptr); // Shouldn't be called otherwise. // Must be initialized via the init(options::build_db) function call. // assert (build_db_ != nullptr); for (size_t retry (retry_);; ) { try { transaction tr (conn->begin ()); shared_ptr<build_tenant> t (build_db_->find<build_tenant> (tid)); if (t != nullptr && t->service) { tenant_service& s (*t->service); if (optional<string> data = f (s)) { s.data = move (*data); build_db_->update (t); } } tr.commit (); // Bail out if we have successfully updated the service state. // break; } catch (const odb::recoverable& e) { if (retry-- == 0) throw; HANDLER_DIAG; l1 ([&]{trace << e << "; " << retry + 1 << " tenant service " << "state update retries left";}); } } } }