feat: read database url from env variable
Now we can use migra cli without config initialization
This commit is contained in:
parent
1b8c2a4273
commit
032e1b7287
2 changed files with 90 additions and 55 deletions
|
@ -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)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -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)?;
|
||||||
}
|
}
|
||||||
|
|
Reference in a new issue