2018-02-10 00:00:55 +00:00
|
|
|
use std::ops::Deref;
|
|
|
|
|
2018-03-07 17:41:34 +00:00
|
|
|
use diesel::r2d2;
|
|
|
|
use diesel::r2d2::ConnectionManager;
|
2018-12-30 22:34:31 +00:00
|
|
|
use diesel::{Connection as DieselConnection, ConnectionError};
|
2018-02-10 00:00:55 +00:00
|
|
|
|
|
|
|
use rocket::http::Status;
|
|
|
|
use rocket::request::{self, FromRequest};
|
|
|
|
use rocket::{Outcome, Request, State};
|
|
|
|
|
2019-05-03 13:46:29 +00:00
|
|
|
use crate::error::Error;
|
2019-06-02 11:35:01 +00:00
|
|
|
use chrono::prelude::*;
|
|
|
|
use std::process::Command;
|
2019-05-03 13:46:29 +00:00
|
|
|
|
2018-12-07 01:05:45 +00:00
|
|
|
use crate::CONFIG;
|
2018-02-10 00:00:55 +00:00
|
|
|
|
|
|
|
/// An alias to the database connection used
|
2019-05-26 21:02:41 +00:00
|
|
|
#[cfg(feature = "sqlite")]
|
2019-05-27 20:58:52 +00:00
|
|
|
type Connection = diesel::sqlite::SqliteConnection;
|
2019-05-26 21:02:41 +00:00
|
|
|
#[cfg(feature = "mysql")]
|
2019-05-27 20:58:52 +00:00
|
|
|
type Connection = diesel::mysql::MysqlConnection;
|
2019-09-12 20:12:22 +00:00
|
|
|
#[cfg(feature = "postgresql")]
|
|
|
|
type Connection = diesel::pg::PgConnection;
|
2018-02-10 00:00:55 +00:00
|
|
|
|
2019-05-27 20:58:52 +00:00
|
|
|
/// An alias to the type for a pool of Diesel connections.
|
2018-02-10 00:00:55 +00:00
|
|
|
type Pool = r2d2::Pool<ConnectionManager<Connection>>;
|
|
|
|
|
|
|
|
/// Connection request guard type: a wrapper around an r2d2 pooled connection.
|
|
|
|
pub struct DbConn(pub r2d2::PooledConnection<ConnectionManager<Connection>>);
|
|
|
|
|
|
|
|
pub mod models;
|
2019-05-26 21:02:41 +00:00
|
|
|
#[cfg(feature = "sqlite")]
|
|
|
|
#[path = "schemas/sqlite/schema.rs"]
|
2018-12-30 22:34:31 +00:00
|
|
|
pub mod schema;
|
2019-05-26 21:02:41 +00:00
|
|
|
#[cfg(feature = "mysql")]
|
|
|
|
#[path = "schemas/mysql/schema.rs"]
|
|
|
|
pub mod schema;
|
2019-09-12 20:12:22 +00:00
|
|
|
#[cfg(feature = "postgresql")]
|
|
|
|
#[path = "schemas/postgresql/schema.rs"]
|
|
|
|
pub mod schema;
|
2019-05-26 21:02:41 +00:00
|
|
|
|
2018-02-10 00:00:55 +00:00
|
|
|
/// Initializes a database pool.
|
|
|
|
pub fn init_pool() -> Pool {
|
2019-01-25 17:23:51 +00:00
|
|
|
let manager = ConnectionManager::new(CONFIG.database_url());
|
2018-02-10 00:00:55 +00:00
|
|
|
|
2018-12-30 22:34:31 +00:00
|
|
|
r2d2::Pool::builder().build(manager).expect("Failed to create pool")
|
2018-02-10 00:00:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_connection() -> Result<Connection, ConnectionError> {
|
2019-06-02 11:35:01 +00:00
|
|
|
Connection::establish(&CONFIG.database_url())
|
2018-02-10 00:00:55 +00:00
|
|
|
}
|
|
|
|
|
2019-05-03 13:46:29 +00:00
|
|
|
/// Creates a back-up of the database using sqlite3
|
|
|
|
pub fn backup_database() -> Result<(), Error> {
|
2019-10-11 10:08:40 +00:00
|
|
|
use std::path::Path;
|
|
|
|
let db_url = CONFIG.database_url();
|
|
|
|
let db_path = Path::new(&db_url).parent().unwrap();
|
|
|
|
|
2019-05-03 13:46:29 +00:00
|
|
|
let now: DateTime<Utc> = Utc::now();
|
2019-07-09 15:26:34 +00:00
|
|
|
let file_date = now.format("%Y%m%d").to_string();
|
2019-05-03 13:46:29 +00:00
|
|
|
let backup_command: String = format!("{}{}{}", ".backup 'db_", file_date, ".sqlite3'");
|
|
|
|
|
|
|
|
Command::new("sqlite3")
|
2019-10-11 10:08:40 +00:00
|
|
|
.current_dir(db_path)
|
2019-05-03 13:46:29 +00:00
|
|
|
.args(&["db.sqlite3", &backup_command])
|
|
|
|
.output()
|
|
|
|
.expect("Can't open database, sqlite3 is not available, make sure it's installed and available on the PATH");
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-02-10 00:00:55 +00:00
|
|
|
/// Attempts to retrieve a single connection from the managed database pool. If
|
|
|
|
/// no pool is currently managed, fails with an `InternalServerError` status. If
|
|
|
|
/// no connections are available, fails with a `ServiceUnavailable` status.
|
|
|
|
impl<'a, 'r> FromRequest<'a, 'r> for DbConn {
|
|
|
|
type Error = ();
|
|
|
|
|
|
|
|
fn from_request(request: &'a Request<'r>) -> request::Outcome<DbConn, ()> {
|
2019-11-04 13:30:24 +00:00
|
|
|
let pool = request.guard::<State<Pool>>()?;
|
2018-02-10 00:00:55 +00:00
|
|
|
match pool.get() {
|
|
|
|
Ok(conn) => Outcome::Success(DbConn(conn)),
|
2018-12-30 22:34:31 +00:00
|
|
|
Err(_) => Outcome::Failure((Status::ServiceUnavailable, ())),
|
2018-02-10 00:00:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// For the convenience of using an &DbConn as a &Database.
|
|
|
|
impl Deref for DbConn {
|
|
|
|
type Target = Connection;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|