From 3518acf1b97eb698186053e79c57c7a09da07817 Mon Sep 17 00:00:00 2001 From: Viktor Bahr Date: Mon, 9 May 2022 15:31:12 +0200 Subject: [PATCH] CLI command to generate a new migration (#656) * feat(cli): add 'migration generate' subcommand This subcommend will create a new, empty migration. * feat(deps): add chrono crate This crate will allow me to fetch the current date and time required for generating the migration filename. * feat(cli): generate migration filename * feat(cli): read template, replace migration name * feat(cli): write modified content to file * feat(deps): add regex crate Allows me to parse the lib.rs file containing the migrator logic. * fix(cli): add missing chrono import * feat(cli): mod declaration for new migration This modifies the existing migator file, adding a module declaration for the newly generated migration. * feat(cli): regenerate migration vector * feat(cli): write updated migrator file to disk This completes updating the migrator file with the new migration information. * docs(cli): additional docstring * refactor(cli): move logic into functions * test(cli): create new migration happy path * test(cli): update migrator happy path * fix(cli): dedicated tmp dir for test This avoids conflicts with the other tests. * style(cli): align generated code with cargofmt As suggested by @billy1624 in the review of #656. * feat(cli): harden regex against extra spaces As suggested by @billy1624 in the review of #656. Co-authored-by: Billy Chan --- sea-orm-cli/Cargo.toml | 2 + sea-orm-cli/src/cli.rs | 11 +++ sea-orm-cli/src/commands.rs | 129 ++++++++++++++++++++++++++++++++++++ 3 files changed, 142 insertions(+) diff --git a/sea-orm-cli/Cargo.toml b/sea-orm-cli/Cargo.toml index 145c1fdbf..58a215a58 100644 --- a/sea-orm-cli/Cargo.toml +++ b/sea-orm-cli/Cargo.toml @@ -38,6 +38,8 @@ sqlx = { version = "^0.5", default-features = false, features = [ "mysql", "post tracing-subscriber = { version = "0.3", features = ["env-filter"] } tracing = { version = "0.1" } url = "^2.2" +chrono = "0.4" +regex = "1" [dev-dependencies] smol = "1.2.5" diff --git a/sea-orm-cli/src/cli.rs b/sea-orm-cli/src/cli.rs index 362e49146..1a0ebfe68 100644 --- a/sea-orm-cli/src/cli.rs +++ b/sea-orm-cli/src/cli.rs @@ -95,6 +95,17 @@ pub fn build_cli() -> App<'static, 'static> { .about("Initialize migration directory") .arg(arg_migration_dir.clone()), ) + .subcommand( + SubCommand::with_name("generate") + .about("Generate a new, empty migration") + .arg( + Arg::with_name("MIGRATION_NAME") + .help("Name of the new migation") + .required(true) + .takes_value(true), + ) + .arg(arg_migration_dir.clone()), + ) .arg(arg_migration_dir.clone()); for subcommand in get_subcommands() { migrate_subcommands = diff --git a/sea-orm-cli/src/commands.rs b/sea-orm-cli/src/commands.rs index c9a1ef002..b0752274f 100644 --- a/sea-orm-cli/src/commands.rs +++ b/sea-orm-cli/src/commands.rs @@ -1,4 +1,6 @@ +use chrono::Local; use clap::ArgMatches; +use regex::Regex; use sea_orm_codegen::{EntityTransformer, OutputFile, WithSerde}; use std::{error::Error, fmt::Display, fs, io::Write, path::Path, process::Command, str::FromStr}; use url::Url; @@ -220,6 +222,22 @@ pub fn run_migrate_command(matches: &ArgMatches<'_>) -> Result<(), Box) -> Result<(), Box Result<(), Box> { + let migration_filepath = Path::new(migration_dir) + .join("src") + .join(format!("{}.rs", &migration_name)); + println!("Creating migration file `{}`", migration_filepath.display()); + // TODO: make OS agnostic + let migration_template = + include_str!("../template/migration/src/m20220101_000001_create_table.rs"); + let migration_content = + migration_template.replace("m20220101_000001_create_table", &migration_name); + let mut migration_file = fs::File::create(migration_filepath)?; + migration_file.write_all(migration_content.as_bytes())?; + Ok(()) +} + +fn update_migrator(migration_name: &str, migration_dir: &str) -> Result<(), Box> { + let migrator_filepath = Path::new(migration_dir).join("src").join("lib.rs"); + println!( + "Adding migration `{}` to `{}`", + migration_name, + migrator_filepath.display() + ); + let migrator_content = fs::read_to_string(&migrator_filepath)?; + let mut updated_migrator_content = migrator_content.clone(); + + // create a backup of the migrator file in case something goes wrong + let migrator_backup_filepath = migrator_filepath.clone().with_file_name("lib.rs.bkp"); + fs::copy(&migrator_filepath, &migrator_backup_filepath)?; + let mut migrator_file = fs::File::create(&migrator_filepath)?; + + // find existing mod declarations, add new line + let mod_regex = Regex::new(r"mod\s+(?Pm\d{8}_\d{6}_\w+);")?; + let mods: Vec<_> = mod_regex.captures_iter(&migrator_content).collect(); + let mods_end = mods.last().unwrap().get(0).unwrap().end() + 1; + updated_migrator_content.insert_str(mods_end, format!("mod {};\n", migration_name).as_str()); + + // build new vector from declared migration modules + let mut migrations: Vec<&str> = mods + .iter() + .map(|cap| cap.name("name").unwrap().as_str()) + .collect(); + migrations.push(migration_name); + let mut boxed_migrations = migrations + .iter() + .map(|migration| format!(" Box::new({}::Migration),", migration)) + .collect::>() + .join("\n"); + boxed_migrations.push_str("\n"); + let boxed_migrations = format!("vec![\n{} ]\n", boxed_migrations); + let vec_regex = Regex::new(r"vec!\[[\s\S]+\]\n")?; + let updated_migrator_content = vec_regex.replace(&updated_migrator_content, &boxed_migrations); + + migrator_file.write_all(updated_migrator_content.as_bytes())?; + fs::remove_file(&migrator_backup_filepath)?; + Ok(()) +} + pub fn handle_error(error: E) where E: Display, @@ -371,4 +446,58 @@ mod tests { smol::block_on(run_generate_command(matches.subcommand().1.unwrap())).unwrap(); } + #[test] + fn test_create_new_migration() { + let migration_name = "test_name"; + let migration_dir = "/tmp/sea_orm_cli_test_new_migration/"; + fs::create_dir_all(format!("{}src", migration_dir)).unwrap(); + create_new_migration(migration_name, migration_dir).unwrap(); + let migration_filepath = Path::new(migration_dir) + .join("src") + .join(format!("{}.rs", migration_name)); + assert!(migration_filepath.exists()); + let migration_content = fs::read_to_string(migration_filepath).unwrap(); + let migration_content = + migration_content.replace(&migration_name, "m20220101_000001_create_table"); + assert_eq!( + &migration_content, + include_str!("../template/migration/src/m20220101_000001_create_table.rs") + ); + fs::remove_dir_all("/tmp/sea_orm_cli_test_new_migration/").unwrap(); + } + + #[test] + fn test_update_migrator() { + let migration_name = "test_name"; + let migration_dir = "/tmp/sea_orm_cli_test_update_migrator/"; + fs::create_dir_all(format!("{}src", migration_dir)).unwrap(); + let migrator_filepath = Path::new(migration_dir).join("src").join("lib.rs"); + fs::copy("./template/migration/src/lib.rs", &migrator_filepath).unwrap(); + update_migrator(migration_name, migration_dir).unwrap(); + assert!(&migrator_filepath.exists()); + let migrator_content = fs::read_to_string(&migrator_filepath).unwrap(); + let mod_regex = Regex::new(r"mod (?P\w+);").unwrap(); + let migrations: Vec<&str> = mod_regex + .captures_iter(&migrator_content) + .map(|cap| cap.name("name").unwrap().as_str()) + .collect(); + assert_eq!(migrations.len(), 2); + assert_eq!( + *migrations.first().unwrap(), + "m20220101_000001_create_table" + ); + assert_eq!(migrations.last().unwrap(), &migration_name); + let boxed_regex = Regex::new(r"Box::new\((?P\S+)::Migration\)").unwrap(); + let migrations: Vec<&str> = boxed_regex + .captures_iter(&migrator_content) + .map(|cap| cap.name("name").unwrap().as_str()) + .collect(); + assert_eq!(migrations.len(), 2); + assert_eq!( + *migrations.first().unwrap(), + "m20220101_000001_create_table" + ); + assert_eq!(migrations.last().unwrap(), &migration_name); + fs::remove_dir_all("/tmp/sea_orm_cli_test_update_migrator/").unwrap(); + } }