init repo
This commit is contained in:
commit
53a4657834
24
Cargo.toml
Normal file
24
Cargo.toml
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
[package]
|
||||||
|
name = "connectors"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["JesusPerez <jpl@jesusperez.pro>"]
|
||||||
|
edition = "2018"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
anyhow = "1.0.40"
|
||||||
|
async-trait = "0.1.51"
|
||||||
|
|
||||||
|
envmnt = "0.9.0"
|
||||||
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
|
serde_derive = "1.0"
|
||||||
|
tokio = { version = "1.5.0", features = ["full"] }
|
||||||
|
|
||||||
|
redis = { version = "0.21.2", features = [ "tokio-comp", "cluster"] }
|
||||||
|
slab = "0.4.4"
|
||||||
|
sqlx = {version = "0.5.7", default-features = false, features = ["macros","runtime-tokio-rustls","sqlite", "mysql", "postgres", "decimal", "chrono"]}
|
||||||
|
sthash = "0.2.11"
|
||||||
|
tempfile = "3.2.0"
|
||||||
|
thiserror = "1.0.29"
|
||||||
|
# tikv-client = { git = "https://github.com/tikv/client-rust.git" }
|
||||||
|
|
||||||
|
datastores = { version = "0.1.0", path = "../defs" }
|
21
LICENSE
Normal file
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2020-2021 Jesús Pérez Lorenzo
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
41
README.md
Normal file
41
README.md
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
# Data Stores Connections Library
|
||||||
|
|
||||||
|
## Rust Library to connect and manage [Data Stores](https://en.wikipedia.org/wiki/Data_store) for **LibreClouds** [Klouds](https://rlung.librecloud.online/LibreCloud/Klouds#klouds)
|
||||||
|
|
||||||
|
Part of the following developments:
|
||||||
|
|
||||||
|
- [CloudMandala](https://rlung.librecloud.online/LibreCloud/CloudMandala#cloudmandala)
|
||||||
|
- [Zteron](https://rlung.librecloud.online/LibreCloud/CloudMandala#cloudmandala)
|
||||||
|
|
||||||
|
It includes definitions and functions for the following [Data Stores](https://en.wikipedia.org/wiki/Data_store):
|
||||||
|
|
||||||
|
- Redis
|
||||||
|
- MySQL
|
||||||
|
- Postgres
|
||||||
|
- Sqlite
|
||||||
|
- Slab
|
||||||
|
|
||||||
|
|
||||||
|
### How to use
|
||||||
|
|
||||||
|
1 - Clone or download this lib in a path, better outside of target development
|
||||||
|
|
||||||
|
2 - Get current version from <u>Cargo.toml</u>
|
||||||
|
|
||||||
|
3 - Include a line like the one below in <u>target development Cargo.toml</u> (adjust version & path) and **use** whatever is need.
|
||||||
|
|
||||||
|
```toml
|
||||||
|
connectors = { version = "0.1.0", path = "../lib/datastores/connectors" }
|
||||||
|
```
|
||||||
|
|
||||||
|
### Connectors
|
||||||
|
|
||||||
|
[Data Store Connectors](LibreCloud/lib_datastores_connectors)
|
||||||
|
|
||||||
|
## Author
|
||||||
|
|
||||||
|
- [Jesús Pérez](https://info.jesusperez.pro).
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
MIT
|
300
src/defs.rs
Normal file
300
src/defs.rs
Normal file
@ -0,0 +1,300 @@
|
|||||||
|
use serde::{Serialize,Deserialize};
|
||||||
|
use std::collections::{HashMap};
|
||||||
|
use anyhow::{anyhow,Result};
|
||||||
|
use datastores::defs::DataStore;
|
||||||
|
|
||||||
|
use crate::redis::RedisPool;
|
||||||
|
use crate::mysql::MysqlPool;
|
||||||
|
use crate::postgres::PostgresPool;
|
||||||
|
use crate::sqlite::SqlitePool;
|
||||||
|
//use crate::tikv::TikvPool;
|
||||||
|
|
||||||
|
use async_trait::async_trait;
|
||||||
|
|
||||||
|
//use tkdr::crypt_lib::decrypt; // FIXME
|
||||||
|
fn decrypt(data: &str, _key: &str) -> String {
|
||||||
|
data.to_owned()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
pub trait PoolHandle {
|
||||||
|
async fn connect(&self) -> Self;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum DataPool {
|
||||||
|
Redis(RedisPool),
|
||||||
|
Mysql(MysqlPool),
|
||||||
|
Postgres(PostgresPool),
|
||||||
|
Sqlite(SqlitePool),
|
||||||
|
//Tikv(TikvPool),
|
||||||
|
File(String),
|
||||||
|
Slab(String),
|
||||||
|
NoPool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialEq for DataPool {
|
||||||
|
fn eq(&self, other: &Self) -> bool {
|
||||||
|
match self {
|
||||||
|
DataPool::Redis(source) =>
|
||||||
|
match other {
|
||||||
|
DataPool::Redis(target) => source.id == target.id,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
DataPool::Mysql(source) =>
|
||||||
|
match other {
|
||||||
|
DataPool::Mysql(target) => source.id == target.id,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
DataPool::Postgres(source) =>
|
||||||
|
match other {
|
||||||
|
DataPool::Postgres(target) => source.id == target.id,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
DataPool::Sqlite(source) =>
|
||||||
|
match other {
|
||||||
|
DataPool::Sqlite(target) => source.id == target.id,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
DataPool::File(source) =>
|
||||||
|
match other {
|
||||||
|
DataPool::File(target) => source == target,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
DataPool::Slab(source) =>
|
||||||
|
match other {
|
||||||
|
DataPool::Slab(target) => source == target,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
DataPool::NoPool =>
|
||||||
|
match other {
|
||||||
|
DataPool::NoPool => true,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl Default for DataPool {
|
||||||
|
fn default() -> Self {
|
||||||
|
DataPool::NoPool
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Default)]
|
||||||
|
pub struct AppDataConn {
|
||||||
|
name: String,
|
||||||
|
datastores: HashMap<String,DataPool>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AppDataConn {
|
||||||
|
pub async fn new(name: String, stores_settings: Vec<StoreSettings>, key: &str) -> Self {
|
||||||
|
let mut datastores: HashMap<String,DataPool> = HashMap::new();
|
||||||
|
for settings in stores_settings.iter() {
|
||||||
|
match settings.datastore {
|
||||||
|
DataStore::Redis => {
|
||||||
|
datastores.insert(
|
||||||
|
settings.id.to_owned(),
|
||||||
|
DataPool::Redis(RedisPool::new(settings.to_owned()).connect_pool().await)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
DataStore::Mysql => {
|
||||||
|
datastores.insert(
|
||||||
|
settings.id.to_owned(),
|
||||||
|
DataPool::Mysql(MysqlPool::new(settings.to_owned(),key).await.connect_pool().await)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
DataStore::Postgres => {
|
||||||
|
datastores.insert(
|
||||||
|
settings.id.to_owned(),
|
||||||
|
DataPool::Postgres(PostgresPool::new(settings.to_owned(),key).await.connect_pool().await)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
DataStore::Sqlite => {
|
||||||
|
datastores.insert(
|
||||||
|
settings.id.to_owned(),
|
||||||
|
DataPool::Sqlite(SqlitePool::new(settings.to_owned()).connect_pool().await)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
_ => continue,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
name,
|
||||||
|
datastores,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn get_conn(&self,key: &str) -> &DataPool {
|
||||||
|
self.datastores.get(key).unwrap_or(&DataPool::NoPool)
|
||||||
|
}
|
||||||
|
pub async fn get_redis(&self,key: &str) -> Result<&redis::aio::Connection> {
|
||||||
|
match self.datastores.get(key).unwrap_or(&DataPool::NoPool) {
|
||||||
|
DataPool::Redis(redis_conn) =>
|
||||||
|
if let Some(pool) = &redis_conn.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Redis pool not available"))
|
||||||
|
},
|
||||||
|
_ => Err(anyhow!("{} is not a Redis connection",key)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn get_mysql(&self,key: &str) -> Result<&sqlx::pool::PoolConnection<sqlx::MySql>> {
|
||||||
|
match self.datastores.get(key).unwrap_or(&DataPool::NoPool) {
|
||||||
|
DataPool::Mysql(mysql_conn) =>
|
||||||
|
if let Some(pool) = &mysql_conn.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Mysql pool not available"))
|
||||||
|
},
|
||||||
|
_ => Err(anyhow!("{} is not a Mysql connection",key)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn get_postgres(&self,key: &str) -> Result<&sqlx::pool::PoolConnection<sqlx::Postgres>> {
|
||||||
|
match self.datastores.get(key).unwrap_or(&DataPool::NoPool) {
|
||||||
|
DataPool::Postgres(postgres_conn) =>
|
||||||
|
if let Some(pool) = &postgres_conn.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Postgres pool not available"))
|
||||||
|
},
|
||||||
|
_ => Err(anyhow!("{} is not a Postgres connection",key)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn get_sqlite(&self,key: &str) -> Result<&sqlx::Pool<sqlx::Sqlite>> {
|
||||||
|
match self.datastores.get(key).unwrap_or(&DataPool::NoPool) {
|
||||||
|
DataPool::Sqlite(sqlite_conn) =>
|
||||||
|
if let Some(pool) = &sqlite_conn.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Sqlite pool not available"))
|
||||||
|
},
|
||||||
|
_ => Err(anyhow!("{} is not a Sqlite connection",key)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn check_connections(&self, datastores_settings: Vec<StoreSettings>) -> bool {
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
let mut status = false;
|
||||||
|
for con in &datastores_settings {
|
||||||
|
match con.datastore {
|
||||||
|
DataStore::Redis => {
|
||||||
|
status = match self.get_redis(&con.id).await {
|
||||||
|
Ok(_pool) => {
|
||||||
|
if debug > 0 {
|
||||||
|
println!("app_data_conn found redis pool");
|
||||||
|
}
|
||||||
|
true
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
if StoreSettings::check_required_id(datastores_settings.to_owned(),&con.id) {
|
||||||
|
panic!("Error app_data_conn required: {}",e);
|
||||||
|
} else {
|
||||||
|
println!("Error app_data_conn: {}",e);
|
||||||
|
}
|
||||||
|
false
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
status
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Serialize, Deserialize, Default)]
|
||||||
|
pub struct StoreSettings {
|
||||||
|
pub id: String,
|
||||||
|
pub host: String,
|
||||||
|
pub port: u32,
|
||||||
|
pub user: String,
|
||||||
|
pub pass: String,
|
||||||
|
pub datastore: DataStore,
|
||||||
|
pub database: String,
|
||||||
|
pub prefix: String,
|
||||||
|
pub max_conn: u32,
|
||||||
|
pub required: bool,
|
||||||
|
}
|
||||||
|
impl StoreSettings {
|
||||||
|
pub fn get_credentials(&self,key: &str) -> (String,String) {
|
||||||
|
let user: String;
|
||||||
|
let pass: String;
|
||||||
|
if key.is_empty() {
|
||||||
|
user = self.user.to_owned();
|
||||||
|
pass = self.pass.to_owned();
|
||||||
|
} else {
|
||||||
|
user = decrypt(&self.user, key);
|
||||||
|
pass = decrypt(&self.pass, key);
|
||||||
|
}
|
||||||
|
(user,pass)
|
||||||
|
}
|
||||||
|
pub fn url_db(&self, key: &str) -> String {
|
||||||
|
let store = self.get_store();
|
||||||
|
let (user,pass) = self.get_credentials(key);
|
||||||
|
let user_pass: String;
|
||||||
|
if ! user.is_empty() || ! pass.is_empty() {
|
||||||
|
user_pass = format!("{}:{}", &user, &pass);
|
||||||
|
} else {
|
||||||
|
user_pass = String::from("");
|
||||||
|
}
|
||||||
|
let host = &self.host;
|
||||||
|
let port = &self.port;
|
||||||
|
let database = &self.database;
|
||||||
|
format!(
|
||||||
|
"{}://{}@{}:{}/{}",
|
||||||
|
store, &user_pass, &host, &port, &database
|
||||||
|
)
|
||||||
|
}
|
||||||
|
pub fn url_keyval(&self) -> String {
|
||||||
|
let store = self.get_store();
|
||||||
|
let host = &self.host;
|
||||||
|
let port = &self.port;
|
||||||
|
format!("{}://{}:{}", store, &host, &port)
|
||||||
|
}
|
||||||
|
pub fn url_local(&self) -> String {
|
||||||
|
let store = self.get_store();
|
||||||
|
format!("{}.{}", store, &self.database)
|
||||||
|
}
|
||||||
|
pub fn get_store(&self) -> String {
|
||||||
|
match self.datastore {
|
||||||
|
DataStore::File => String::from("file"),
|
||||||
|
DataStore::Mysql => String::from("mysql"),
|
||||||
|
DataStore::Postgres => String::from("postgres"),
|
||||||
|
DataStore::Sqlite => String::from("sqlite"),
|
||||||
|
DataStore::Redis => String::from("redis"),
|
||||||
|
// DataStore::Tikv => String::from("tikv"),
|
||||||
|
DataStore::Slab => String::from("slab"),
|
||||||
|
DataStore::Unknown => String::from("Unknown"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn find_storesetting_id(cfg_store_settings: Vec<StoreSettings>, id: &str) -> Result<Vec<StoreSettings>> {
|
||||||
|
let datastore_found: Vec<StoreSettings> = cfg_store_settings.iter().filter(|itm| itm.id == id).cloned().collect();
|
||||||
|
if datastore_found.len() > 0 {
|
||||||
|
Ok(datastore_found)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("No DataStore Settings found for: {}",&id))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn check_required_id(cfg_store_settings: Vec<StoreSettings>, id: &str) -> bool {
|
||||||
|
match StoreSettings::find_storesetting_id(cfg_store_settings.to_owned(),id) {
|
||||||
|
Ok(ds_found) =>
|
||||||
|
if let Some(item) = ds_found.get(0) {
|
||||||
|
item.required
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Error check required: {}",e);
|
||||||
|
false
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct DataStorePool<T>
|
||||||
|
where T : PoolHandle + Default
|
||||||
|
{
|
||||||
|
pub pool: T,
|
||||||
|
pub typ: DataStore,
|
||||||
|
}
|
39
src/errors.rs
Normal file
39
src/errors.rs
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
/// Error definition
|
||||||
|
//
|
||||||
|
use thiserror::Error;
|
||||||
|
use std::fmt;
|
||||||
|
|
||||||
|
// #[derive(Debug, Fail)]
|
||||||
|
|
||||||
|
#[derive(Error, Eq, PartialEq, Clone, Debug)]
|
||||||
|
pub enum DatabaseErrors {
|
||||||
|
// #[fail(display = "{}. Reason: {}", 0, 1)]
|
||||||
|
ConnectionFailed(String, String),
|
||||||
|
// #[fail(display = "{}. Reason: {}", 0, 1)]
|
||||||
|
PoolCreationFailed(String, String),
|
||||||
|
}
|
||||||
|
impl fmt::Display for DatabaseErrors {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
match self {
|
||||||
|
DatabaseErrors::ConnectionFailed(s,t) => write!(f,"connection {}: {} ",s,t),
|
||||||
|
DatabaseErrors::PoolCreationFailed(s,t) => write!(f,"pool {}: {} ",s,t),
|
||||||
|
}
|
||||||
|
//write!(f, "{:?}", self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Database Store Errors
|
||||||
|
#[derive(Error, Debug)]
|
||||||
|
pub enum DataStoreError {
|
||||||
|
#[error("data store disconnected")]
|
||||||
|
Disconnect(#[from] std::io::Error),
|
||||||
|
#[error("the data for key `{0}` is not available")]
|
||||||
|
Redaction(String),
|
||||||
|
#[error("invalid header (expected {expected:?}, found {found:?})")]
|
||||||
|
InvalidHeader {
|
||||||
|
expected: String,
|
||||||
|
found: String,
|
||||||
|
},
|
||||||
|
#[error("unknown data store error")]
|
||||||
|
Unknown,
|
||||||
|
}
|
7
src/lib.rs
Normal file
7
src/lib.rs
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
pub mod defs;
|
||||||
|
pub mod errors;
|
||||||
|
pub mod mysql;
|
||||||
|
pub mod postgres;
|
||||||
|
pub mod redis;
|
||||||
|
pub mod sqlite;
|
||||||
|
// pub mod tikv;
|
82
src/mysql.rs
Normal file
82
src/mysql.rs
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
/// `MySql` Connector
|
||||||
|
//
|
||||||
|
use sqlx::mysql::MySqlPoolOptions;
|
||||||
|
use anyhow::{Result,anyhow};
|
||||||
|
use async_trait::async_trait;
|
||||||
|
|
||||||
|
use crate::defs::{StoreSettings,PoolHandle};
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct MysqlPool {
|
||||||
|
pub id: String,
|
||||||
|
pub client: Option<sqlx::Pool<sqlx::MySql>>,
|
||||||
|
pub conn: Option<sqlx::pool::PoolConnection<sqlx::MySql>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for MysqlPool {
|
||||||
|
fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
|
||||||
|
write!(fmt, "MysqlPool client: {:?}", self.client)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MysqlPool {
|
||||||
|
pub async fn new(pool_settings: StoreSettings,key: &str) -> Self {
|
||||||
|
let url = pool_settings.url_db(key);
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Open Mysql {}: {}",pool_settings.id, &url);
|
||||||
|
}
|
||||||
|
match MySqlPoolOptions::new().max_connections(pool_settings.max_conn)
|
||||||
|
// .connect_lazy(&url);
|
||||||
|
.connect(&url).await {
|
||||||
|
Ok(cli) => Self {
|
||||||
|
id: pool_settings.id,
|
||||||
|
client: Some(cli),
|
||||||
|
conn: None,
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Error Mysql new pool: {}",e);
|
||||||
|
Self::default()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn connect_pool(self) -> Self {
|
||||||
|
self.connect().await
|
||||||
|
}
|
||||||
|
pub async fn get_conn(self) -> Result<sqlx::pool::PoolConnection<sqlx::MySql>> {
|
||||||
|
if let Some(pool) = self.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
let cli=self.connect_pool().await;
|
||||||
|
if let Some(pool) = cli.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Mysql pool not available"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[async_trait]
|
||||||
|
impl PoolHandle for MysqlPool {
|
||||||
|
async fn connect(&self) -> Self {
|
||||||
|
let cli = self;
|
||||||
|
let mut conn = None; // DataPool::NoPool;
|
||||||
|
if let Some(client) = &cli.client {
|
||||||
|
if let Some(cli_pool) = client.try_acquire() {
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Got Mysql connection");
|
||||||
|
}
|
||||||
|
conn = Some(cli_pool);
|
||||||
|
} else {
|
||||||
|
eprintln!("Error Mysql connection ");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
id: cli.id.to_owned(),
|
||||||
|
client: cli.client.to_owned(),
|
||||||
|
conn,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
82
src/postgres.rs
Normal file
82
src/postgres.rs
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
/// Postgres Connector
|
||||||
|
//
|
||||||
|
use sqlx::postgres::PgPoolOptions;
|
||||||
|
use anyhow::{Result,anyhow};
|
||||||
|
use async_trait::async_trait;
|
||||||
|
|
||||||
|
use crate::defs::{StoreSettings,PoolHandle};
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct PostgresPool {
|
||||||
|
pub id: String,
|
||||||
|
pub client: Option<sqlx::Pool<sqlx::Postgres>>,
|
||||||
|
pub conn: Option<sqlx::pool::PoolConnection<sqlx::Postgres>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for PostgresPool {
|
||||||
|
fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
|
||||||
|
write!(fmt, "PostgresPool client: {:?}", self.client)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PostgresPool {
|
||||||
|
pub async fn new(pool_settings: StoreSettings,key: &str) -> Self {
|
||||||
|
let url = pool_settings.url_db(key);
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Open Postgres {}: {}",pool_settings.id, &url);
|
||||||
|
}
|
||||||
|
match PgPoolOptions::new().max_connections(pool_settings.max_conn)
|
||||||
|
// .connect_lazy(&url);
|
||||||
|
.connect(&url).await {
|
||||||
|
Ok(cli) => Self {
|
||||||
|
id: pool_settings.id,
|
||||||
|
client: Some(cli),
|
||||||
|
conn: None,
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Error Postgres new pool: {}",e);
|
||||||
|
Self::default()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn connect_pool(self) -> Self {
|
||||||
|
self.connect().await
|
||||||
|
}
|
||||||
|
pub async fn get_conn(self) -> Result<sqlx::pool::PoolConnection<sqlx::Postgres>> {
|
||||||
|
if let Some(pool) = self.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
let cli=self.connect_pool().await;
|
||||||
|
if let Some(pool) = cli.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Postgres pool not available"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[async_trait]
|
||||||
|
impl PoolHandle for PostgresPool {
|
||||||
|
async fn connect(&self) -> Self {
|
||||||
|
let cli = self;
|
||||||
|
let mut conn = None; // DataPool::NoPool;
|
||||||
|
if let Some(client) = &cli.client {
|
||||||
|
if let Some(cli_pool) = client.try_acquire() {
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Got Postgres connection");
|
||||||
|
}
|
||||||
|
conn = Some(cli_pool);
|
||||||
|
} else {
|
||||||
|
eprintln!("Error Postgres connection ");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
id: cli.id.to_owned(),
|
||||||
|
client: cli.client.to_owned(),
|
||||||
|
conn,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
82
src/redis.rs
Normal file
82
src/redis.rs
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
/// Redis Connector
|
||||||
|
//
|
||||||
|
use async_trait::async_trait;
|
||||||
|
use anyhow::{anyhow,Result};
|
||||||
|
|
||||||
|
use crate::defs::{StoreSettings,PoolHandle};
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct RedisPool {
|
||||||
|
pub id: String,
|
||||||
|
pub client: Option<redis::Client>,
|
||||||
|
pub conn: Option<redis::aio::Connection>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for RedisPool {
|
||||||
|
fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
|
||||||
|
write!(fmt, "RedisPool client: {:?}", self.client)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RedisPool {
|
||||||
|
pub fn new(pool_settings: StoreSettings) -> Self {
|
||||||
|
let url = pool_settings.url_keyval();
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Open Redis {}: {}",pool_settings.id, &url);
|
||||||
|
}
|
||||||
|
match redis::Client::open(url) {
|
||||||
|
Ok(cli) => Self {
|
||||||
|
id: pool_settings.id,
|
||||||
|
client: Some(cli.to_owned()),
|
||||||
|
conn: None,
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Error Redis new pool: {}",e);
|
||||||
|
Self::default()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn connect_pool(self) -> Self {
|
||||||
|
self.connect().await
|
||||||
|
}
|
||||||
|
pub async fn get_conn(self) -> Result<redis::aio::Connection> {
|
||||||
|
if let Some(pool) = self.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
let cli=self.connect_pool().await;
|
||||||
|
if let Some(pool) = cli.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Redis pool not available"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[async_trait]
|
||||||
|
impl PoolHandle for RedisPool {
|
||||||
|
async fn connect(&self) -> Self {
|
||||||
|
let cli = self;
|
||||||
|
let mut conn = None;
|
||||||
|
if let Some(client) = &cli.client {
|
||||||
|
match client.get_async_connection().await {
|
||||||
|
Ok(cli_pool) => {
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Got Redis connection");
|
||||||
|
}
|
||||||
|
conn = Some(cli_pool);
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Error Redis connection: {}",e);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
id: cli.id.to_owned(),
|
||||||
|
client: cli.client.to_owned(),
|
||||||
|
conn,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
80
src/sqlite.rs
Normal file
80
src/sqlite.rs
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
/// Sqlite Connection
|
||||||
|
//
|
||||||
|
use anyhow::{Result,anyhow};
|
||||||
|
use async_trait::async_trait;
|
||||||
|
|
||||||
|
use crate::defs::{StoreSettings,PoolHandle};
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct SqlitePool {
|
||||||
|
pub id: String,
|
||||||
|
// TODO Pool with Sqlite not working
|
||||||
|
// pub client: Option<sqlx::Pool<sqlx::Sqlite>>,
|
||||||
|
// pub conn: Option<sqlx::pool::PoolConnection<sqlx::Sqlite>>,
|
||||||
|
pub client: Option<String>,
|
||||||
|
pub conn: Option<sqlx::Pool<sqlx::Sqlite>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for SqlitePool {
|
||||||
|
fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
|
||||||
|
write!(fmt, "SqlitePool client: {:?}", self.client)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SqlitePool {
|
||||||
|
pub fn new(pool_settings: StoreSettings) -> Self {
|
||||||
|
let url = pool_settings.url_local();
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Open Sqlite {}: {}",pool_settings.id, &url);
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
id: pool_settings.id.to_owned(),
|
||||||
|
client: Some(url),
|
||||||
|
conn: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub async fn connect_pool(self) -> Self {
|
||||||
|
self.connect().await
|
||||||
|
}
|
||||||
|
pub async fn get_conn(self) -> Result<sqlx::Pool<sqlx::Sqlite>> {
|
||||||
|
if let Some(pool) = self.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
let cli=self.connect_pool().await;
|
||||||
|
if let Some(pool) = cli.conn {
|
||||||
|
Ok(pool)
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("Sqlite pool not available"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[async_trait]
|
||||||
|
impl PoolHandle for SqlitePool {
|
||||||
|
async fn connect(&self) -> Self {
|
||||||
|
let cli = self;
|
||||||
|
let mut conn = None; // DataPool::NoPool;
|
||||||
|
if let Some(client) = &cli.client {
|
||||||
|
// TODO Pool with Sqlite not working
|
||||||
|
// match client.acquire().await {
|
||||||
|
match sqlx::sqlite::SqlitePool::connect(&client).await {
|
||||||
|
Ok(pool) => {
|
||||||
|
let debug = envmnt::get_isize("DEBUG",0);
|
||||||
|
if debug > 0 {
|
||||||
|
println!("Got Sqlite connection");
|
||||||
|
}
|
||||||
|
conn = Some(pool);
|
||||||
|
},
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Error Sqlite new pool: {}",e);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Self {
|
||||||
|
id: cli.id.to_owned(),
|
||||||
|
client: cli.client.to_owned(),
|
||||||
|
conn,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
31
src/tikv.rs
Normal file
31
src/tikv.rs
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
/// Tikv Connection
|
||||||
|
//
|
||||||
|
// Copyright 2020, Jesús Pérez Lorenzo
|
||||||
|
//
|
||||||
|
use crate::models::Tikv;
|
||||||
|
use anyhow::Context as AnyContext;
|
||||||
|
use std::path::PathBuf;
|
||||||
|
use tikv_client::{Config, Key, KvPair, RawClient, Result, TransactionClient, ToOwnedRange, Value};
|
||||||
|
use crate::models::config::StoreKeyValue;
|
||||||
|
|
||||||
|
/// Tikv pool connection
|
||||||
|
pub async fn get_tikv_pool(tikv: &Tikv) -> anyhow::Result<RawClient> {
|
||||||
|
let tikv_url = config.url("tikv");
|
||||||
|
// Optionally encrypt the traffic.
|
||||||
|
let config = if let (Some(ca), Some(cert), Some(key)) = (tikv.ca, tikv.cert, tikv.key) {
|
||||||
|
Config::new(tikv.pd).with_security(ca, cert, key)
|
||||||
|
} else {
|
||||||
|
Config::new(tikv.pd)
|
||||||
|
};
|
||||||
|
// When we first create a client we receive a `Connect` structure which must be resolved before
|
||||||
|
// the client is actually connected and usable.
|
||||||
|
// let client = if tikv.trans {
|
||||||
|
// TransactionClient::new(config).await?
|
||||||
|
// //let mut txn = txn_client.begin().await?;
|
||||||
|
// } else {
|
||||||
|
let client = RawClient::new(config)
|
||||||
|
.await
|
||||||
|
.context(format!("get tikv new tikv pool {}", redis_url))?;
|
||||||
|
//}
|
||||||
|
Ok(client)
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user