Archived
1
0
Fork 0

feat: read database url from env variable

Now we can use migra cli without config initialization
This commit is contained in:
Dmitriy Pleshevskiy 2021-02-08 07:01:31 +03:00
parent 1b8c2a4273
commit 032e1b7287
2 changed files with 90 additions and 55 deletions

View file

@ -3,21 +3,25 @@ use crate::path::PathBuilder;
use postgres::Client; use postgres::Client;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use std::path::{Path, PathBuf}; use std::path::{Path, PathBuf};
use std::{fs, io}; use std::{env, fs, io};
const MIGRA_TOML_FILENAME: &str = "Migra.toml"; const MIGRA_TOML_FILENAME: &str = "Migra.toml";
const DEFAULT_DATABASE_CONNECTION_ENV: &str = "$DATABASE_URL";
#[derive(Debug, Serialize, Deserialize)] #[derive(Debug, Serialize, Deserialize)]
pub(crate) struct Config { pub(crate) struct Config {
#[serde(skip)] #[serde(skip)]
pub root: PathBuf, root: PathBuf,
pub directory: PathBuf,
pub database: DatabaseConfig, directory: PathBuf,
#[serde(default)]
database: DatabaseConfig,
} }
#[derive(Debug, Serialize, Deserialize)] #[derive(Debug, Clone, Default, Serialize, Deserialize)]
pub(crate) struct DatabaseConfig { pub(crate) struct DatabaseConfig {
pub connection: String, pub connection: Option<String>,
} }
impl Default for Config { impl Default for Config {
@ -26,7 +30,7 @@ impl Default for Config {
root: PathBuf::new(), root: PathBuf::new(),
directory: PathBuf::from("database"), directory: PathBuf::from("database"),
database: DatabaseConfig { database: DatabaseConfig {
connection: String::new(), connection: Some(String::from(DEFAULT_DATABASE_CONNECTION_ENV)),
}, },
} }
} }
@ -57,21 +61,26 @@ impl Config {
let config_path = match config_path { let config_path = match config_path {
Some(mut config_path) if config_path.is_dir() => { Some(mut config_path) if config_path.is_dir() => {
config_path.push(MIGRA_TOML_FILENAME); config_path.push(MIGRA_TOML_FILENAME);
config_path Some(config_path)
} }
Some(config_path) => config_path, Some(config_path) => Some(config_path),
None => recursive_find_config_file()?, None => recursive_find_config_file().ok(),
}; };
let content = fs::read_to_string(&config_path)?; match config_path {
None => Ok(Config::default()),
Some(config_path) => {
let content = fs::read_to_string(&config_path)?;
let mut config: Config = toml::from_str(&content).expect("Cannot parse Migra.toml"); let mut config: Config = toml::from_str(&content).expect("Cannot parse Migra.toml");
config.root = config_path config.root = config_path
.parent() .parent()
.unwrap_or_else(|| Path::new("")) .unwrap_or_else(|| Path::new(""))
.to_path_buf(); .to_path_buf();
Ok(config) Ok(config)
}
}
} }
pub fn initialize() -> Result<(), Box<dyn std::error::Error>> { pub fn initialize() -> Result<(), Box<dyn std::error::Error>> {
@ -90,6 +99,56 @@ impl Config {
} }
} }
impl Config {
pub fn directory_path(&self) -> PathBuf {
PathBuilder::from(&self.root)
.append(&self.directory)
.build()
}
pub fn database_connection(&self) -> String {
let connection = self
.database
.connection
.clone()
.unwrap_or_else(|| String::from(DEFAULT_DATABASE_CONNECTION_ENV));
if let Some(connection_env) = connection.strip_prefix("$") {
env::var(connection_env).unwrap_or_else(|_| {
panic!(
r#"You need to provide "{}" environment variable"#,
connection_env
)
})
} else {
connection
}
}
pub fn migration_dir_path(&self) -> PathBuf {
PathBuilder::from(&self.directory_path())
.append("migrations")
.build()
}
pub fn migrations(&self) -> io::Result<Vec<Migration>> {
let mut entries = self
.migration_dir_path()
.read_dir()?
.map(|res| res.map(|e| e.path()))
.collect::<Result<Vec<_>, io::Error>>()?;
entries.sort();
let migrations = entries
.iter()
.filter_map(Migration::new)
.collect::<Vec<_>>();
Ok(migrations)
}
}
#[derive(Debug)]
pub struct Migration { pub struct Migration {
upgrade_sql: PathBuf, upgrade_sql: PathBuf,
downgrade_sql: PathBuf, downgrade_sql: PathBuf,
@ -134,7 +193,10 @@ impl Migration {
Ok(()) Ok(())
} }
pub fn downgrade(&self, client: &mut Client) -> Result<(), Box<dyn std::error::Error + 'static>> { pub fn downgrade(
&self,
client: &mut Client,
) -> Result<(), Box<dyn std::error::Error + 'static>> {
let content = fs::read_to_string(&self.downgrade_sql)?; let content = fs::read_to_string(&self.downgrade_sql)?;
database::apply_sql(client, &content)?; database::apply_sql(client, &content)?;
@ -144,34 +206,3 @@ impl Migration {
Ok(()) Ok(())
} }
} }
impl Config {
pub fn directory_path(&self) -> PathBuf {
PathBuilder::from(&self.root)
.append(&self.directory)
.build()
}
pub fn migration_dir_path(&self) -> PathBuf {
PathBuilder::from(&self.directory_path())
.append("migrations")
.build()
}
pub fn migrations(&self) -> io::Result<Vec<Migration>> {
let mut entries = self
.migration_dir_path()
.read_dir()?
.map(|res| res.map(|e| e.path()))
.collect::<Result<Vec<_>, io::Error>>()?;
entries.sort();
let migrations = entries
.iter()
.filter_map(Migration::new)
.collect::<Vec<_>>();
Ok(migrations)
}
}

View file

@ -21,7 +21,7 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
Command::Apply(ApplyCommandOpt { file_name }) => { Command::Apply(ApplyCommandOpt { file_name }) => {
let config = Config::read(opt.config)?; let config = Config::read(opt.config)?;
let mut client = database::connect(&config.database.connection)?; let mut client = database::connect(&config.database_connection())?;
let file_path = PathBuilder::from(config.directory_path()) let file_path = PathBuilder::from(config.directory_path())
.append(file_name) .append(file_name)
@ -80,7 +80,7 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
Command::List => { Command::List => {
let config = Config::read(opt.config)?; let config = Config::read(opt.config)?;
let mut client = database::connect(&config.database.connection)?; let mut client = database::connect(&config.database_connection())?;
let applied_migrations = database::applied_migrations(&mut client)?; let applied_migrations = database::applied_migrations(&mut client)?;
println!("Applied migrations:"); println!("Applied migrations:");
@ -95,7 +95,8 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
println!(); println!();
let pending_migrations = config.migrations()? let pending_migrations = config
.migrations()?
.into_iter() .into_iter()
.filter(|m| !applied_migrations.contains(m.name())) .filter(|m| !applied_migrations.contains(m.name()))
.collect::<Vec<_>>(); .collect::<Vec<_>>();
@ -111,7 +112,7 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
Command::Upgrade => { Command::Upgrade => {
let config = Config::read(opt.config)?; let config = Config::read(opt.config)?;
let mut client = database::connect(&config.database.connection)?; let mut client = database::connect(&config.database_connection())?;
let applied_migrations = database::applied_migrations(&mut client)?; let applied_migrations = database::applied_migrations(&mut client)?;
@ -134,13 +135,16 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
Command::Downgrade => { Command::Downgrade => {
let config = Config::read(opt.config)?; let config = Config::read(opt.config)?;
let mut client = database::connect(&config.database.connection)?; let mut client = database::connect(&config.database_connection())?;
let applied_migrations = database::applied_migrations(&mut client)?; let applied_migrations = database::applied_migrations(&mut client)?;
let migrations = config.migrations()?; let migrations = config.migrations()?;
if let Some(first_applied_migration) = applied_migrations.first() { if let Some(first_applied_migration) = applied_migrations.first() {
if let Some(migration) = migrations.iter().find(|m| m.name() == first_applied_migration) { if let Some(migration) = migrations
.iter()
.find(|m| m.name() == first_applied_migration)
{
println!("downgrade {}...", migration.name()); println!("downgrade {}...", migration.name());
migration.downgrade(&mut client)?; migration.downgrade(&mut client)?;
} }