Skip to content

Commit

Permalink
chore: more fixes, tests, docs
Browse files Browse the repository at this point in the history
  • Loading branch information
m4tx committed Jan 8, 2025
1 parent a6cdd82 commit 3a4d342
Show file tree
Hide file tree
Showing 6 changed files with 278 additions and 20 deletions.
16 changes: 12 additions & 4 deletions flareon-cli/src/main.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
extern crate core;

mod migration_generator;
mod utils;

Expand All @@ -26,6 +24,9 @@ enum Commands {
/// Path to the crate directory to generate migrations for (default:
/// current directory)
path: Option<PathBuf>,
/// Name of the app to use in the migration (default: crate name)
#[arg(long)]
app_name: Option<String>,
/// Directory to write the migrations to (default: migrations/ directory
/// in the crate's src/ directory)
#[arg(long)]
Expand All @@ -41,9 +42,16 @@ fn main() -> anyhow::Result<()> {
.init();

match cli.command {
Commands::MakeMigrations { path, output_dir } => {
Commands::MakeMigrations {
path,
app_name,
output_dir,
} => {
let path = path.unwrap_or_else(|| PathBuf::from("."));
let options = MigrationGeneratorOptions { output_dir };
let options = MigrationGeneratorOptions {
app_name,
output_dir,
};
make_migrations(&path, options).with_context(|| "unable to create migrations")?;
}
}
Expand Down
137 changes: 124 additions & 13 deletions flareon-cli/src/migration_generator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ use flareon::db::migrations::{DynMigration, MigrationEngine};
use flareon_codegen::model::{Field, Model, ModelArgs, ModelOpts, ModelType};
use flareon_codegen::symbol_resolver::SymbolResolver;
use log::{debug, info, trace};
use petgraph::graph::{DiGraph, NodeIndex};
use petgraph::graph::DiGraph;
use petgraph::visit::EdgeRef;
use proc_macro2::TokenStream;
use quote::{format_ident, quote, ToTokens};
Expand Down Expand Up @@ -48,6 +48,7 @@ pub fn make_migrations(path: &Path, options: MigrationGeneratorOptions) -> anyho

#[derive(Debug, Clone, Default)]
pub struct MigrationGeneratorOptions {
pub app_name: Option<String>,
pub output_dir: Option<PathBuf>,
}

Expand Down Expand Up @@ -82,6 +83,7 @@ impl MigrationGenerator {
Ok(())
}

/// Generate migrations as a ready-to-write source code.
pub fn generate_migrations_to_write(
&mut self,
source_files: Vec<SourceFile>,
Expand All @@ -95,6 +97,8 @@ impl MigrationGenerator {
}
}

/// Generate migrations and return internal structures that can be used to
/// generate source code.
pub fn generate_migrations(
&mut self,
source_files: Vec<SourceFile>,
Expand Down Expand Up @@ -319,7 +323,10 @@ impl MigrationGenerator {
fn make_create_model_operation(app_model: &ModelInSource) -> DynOperation {
DynOperation::CreateModel {
table_name: app_model.model.table_name.clone(),
model_ty: app_model.model.resolved_ty.clone().expect("resolved_ty is expected to be present when parsing the entire file with symbol resolver"),
model_ty: app_model.model.resolved_ty.clone().expect(
"resolved_ty is expected to be present when \
parsing the entire file with symbol resolver",
),
fields: app_model.model.fields.clone(),
}
}
Expand Down Expand Up @@ -381,7 +388,10 @@ impl MigrationGenerator {
fn make_add_field_operation(app_model: &ModelInSource, field: &Field) -> DynOperation {
DynOperation::AddField {
table_name: app_model.model.table_name.clone(),
model_ty: app_model.model.resolved_ty.clone().expect("resolved_ty is expected to be present when parsing the entire file with symbol resolver"),
model_ty: app_model.model.resolved_ty.clone().expect(
"resolved_ty is expected to be present \
when parsing the entire file with symbol resolver",
),
field: field.clone(),
}
}
Expand Down Expand Up @@ -426,7 +436,7 @@ impl MigrationGenerator {
.map(|dependency| dependency.repr())
.collect();

let app_name = &self.crate_name;
let app_name = self.options.app_name.as_ref().unwrap_or(&self.crate_name);
let migration_name = &migration.migration_name;
let migration_def = quote! {
#[derive(Debug, Copy, Clone)]
Expand Down Expand Up @@ -666,6 +676,8 @@ pub struct GeneratedMigration {
}

impl GeneratedMigration {
/// Get the list of [`DynDependency`] for all foreign keys that point
/// to models that are **not** created in this migration.
fn get_foreign_key_dependencies(&self) -> Vec<DynDependency> {
let create_ops = self.get_create_ops_map();
let ops_adding_foreign_keys = self.get_ops_adding_foreign_keys();
Expand All @@ -682,6 +694,16 @@ impl GeneratedMigration {
dependencies
}

/// Removes dependency cycles by removing operations that create cycles.
///
/// This method tries to minimize the number of operations added by
/// calculating the minimum feedback arc set of the dependency graph.
///
/// This method modifies the `operations` field in place.
///
/// # See also
///
/// * [`Self::remove_dependency`]
fn remove_cycles(&mut self) {
let graph = self.construct_dependency_graph();

Expand All @@ -701,6 +723,11 @@ impl GeneratedMigration {
}
}

/// Remove a dependency between two operations.
///
/// This is done by removing foreign keys from the `from` operation that
/// point to the model created by `to` operation, and creating a new
/// `AddField` operation for each removed foreign key.
#[must_use]
fn remove_dependency(from: &mut DynOperation, to: &DynOperation) -> Vec<DynOperation> {
match from {
Expand All @@ -712,7 +739,10 @@ impl GeneratedMigration {
let to_type = match to {
DynOperation::CreateModel { model_ty, .. } => model_ty,
DynOperation::AddField { .. } => {
unreachable!("AddField operation shouldn't be a dependency of CreateModel because it doesn't create a new model")
unreachable!(
"AddField operation shouldn't be a dependency of CreateModel \
because it doesn't create a new model"
)
}
};
trace!(
Expand Down Expand Up @@ -745,18 +775,36 @@ impl GeneratedMigration {
}
}

/// Topologically sort operations in this migration.
///
/// This is to ensure that operations will be applied in the correct order.
/// If there are no dependencies between operations, the order of operations
/// will not be modified.
///
/// This method modifies the `operations` field in place.
///
/// # Panics
///
/// This method should be called after removing cycles; otherwise it will
/// panic.
fn toposort_operations(&mut self) {
let graph = self.construct_dependency_graph();

let sorted = petgraph::algo::toposort(&graph, None)
.expect("cycles shouldn't exist after removing them");
let mut sorted = sorted
.into_iter()
.map(petgraph::prelude::NodeIndex::index)
.map(petgraph::graph::NodeIndex::index)
.collect::<Vec<_>>();
flareon::__private::apply_permutation(&mut self.operations, &mut sorted);
}

/// Construct a graph that represents reverse dependencies between
/// operations in this migration.
///
/// The graph is directed and has an edge from operation A to operation B
/// if operation B creates a foreign key that points to a model created by
/// operation A.
#[must_use]
fn construct_dependency_graph(&mut self) -> DiGraph<usize, (), usize> {
let create_ops = self.get_create_ops_map();
Expand All @@ -769,7 +817,11 @@ impl GeneratedMigration {
}
for (i, dependency_ty) in &ops_adding_foreign_keys {
if let Some(&dependency) = create_ops.get(dependency_ty) {
graph.add_edge(NodeIndex::new(dependency), NodeIndex::new(*i), ());
graph.add_edge(
petgraph::graph::NodeIndex::new(dependency),
petgraph::graph::NodeIndex::new(*i),
(),
);
}
}

Expand Down Expand Up @@ -855,16 +907,19 @@ impl Repr for Field {
let mut tokens = quote! {
::flareon::db::migrations::Field::new(::flareon::db::Identifier::new(#column_name), <#ty as ::flareon::db::DatabaseField>::TYPE)
};
if self
.auto_value
.expect("auto_value is expected to be present when parsing the entire file with symbol resolver")
{
if self.auto_value.expect(
"auto_value is expected to be present \
when parsing the entire file with symbol resolver",
) {
tokens = quote! { #tokens.auto() }
}
if self.primary_key {
tokens = quote! { #tokens.primary_key() }
}
if let Some(fk_spec) = self.foreign_key.clone().expect("foreign_key is expected to be present when parsing the entire file with symbol resolver") {
if let Some(fk_spec) = self.foreign_key.clone().expect(
"foreign_key is expected to be present \
when parsing the entire file with symbol resolver",
) {
let to_model = &fk_spec.to_model;

tokens = quote! {
Expand Down Expand Up @@ -966,7 +1021,8 @@ fn is_field_foreign_key_to(field: &Field, ty: &syn::Type) -> bool {
/// Returns [`None`] if the field is not a foreign key.
fn foreign_key_for_field(field: &Field) -> Option<syn::Type> {
match field.foreign_key.clone().expect(
"foreign_key is expected to be present when parsing the entire file with symbol resolver",
"foreign_key is expected to be present \
when parsing the entire file with symbol resolver",
) {
None => None,
Some(foreign_key_spec) => Some(foreign_key_spec.to_model),
Expand Down Expand Up @@ -1339,4 +1395,59 @@ mod tests {
model_type: parse_quote!(crate::Table4),
}));
}

#[test]
fn make_add_field_operation() {
let app_model = ModelInSource {
model_item: parse_quote! {
struct TestModel {
id: i32,
field1: i32,
}
},
model: Model {
name: format_ident!("TestModel"),
original_name: "TestModel".to_string(),
resolved_ty: Some(parse_quote!(TestModel)),
model_type: Default::default(),
table_name: "test_model".to_string(),
pk_field: Field {
field_name: format_ident!("id"),
column_name: "id".to_string(),
ty: parse_quote!(i32),
auto_value: MaybeUnknown::Known(true),
primary_key: true,
unique: false,
foreign_key: MaybeUnknown::Known(None),
},
fields: vec![],
},
};

let field = Field {
field_name: format_ident!("new_field"),
column_name: "new_field".to_string(),
ty: parse_quote!(i32),
auto_value: MaybeUnknown::Known(false),
primary_key: false,
unique: false,
foreign_key: MaybeUnknown::Known(None),
};

let operation = MigrationGenerator::make_add_field_operation(&app_model, &field);

match operation {
DynOperation::AddField {
table_name,
model_ty,
field: op_field,
} => {
assert_eq!(table_name, "test_model");
assert_eq!(model_ty, parse_quote!(TestModel));
assert_eq!(op_field.column_name, "new_field");
assert_eq!(op_field.ty, parse_quote!(i32));
}
_ => panic!("Expected AddField operation"),
}
}
}
2 changes: 1 addition & 1 deletion flareon-cli/tests/migration_generator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -120,7 +120,7 @@ fn create_models_foreign_key_cycle() {
}

#[test]
fn create_models_foreign_two_migrations() {
fn create_models_foreign_key_two_migrations() {
let mut generator = test_generator();

let src = include_str!("migration_generator/foreign_key_two_migrations/step_1.rs");
Expand Down
40 changes: 40 additions & 0 deletions flareon/src/db.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1163,4 +1163,44 @@ mod tests {
LimitedString::<5>::new("test").unwrap(),
);
}

#[test]
fn db_field_value_is_auto() {
let auto_value = DbFieldValue::Auto;
assert!(auto_value.is_auto());
assert!(!auto_value.is_value());
}

#[test]
fn db_field_value_is_value() {
let value = DbFieldValue::Value(42.into());
assert!(value.is_value());
assert!(!value.is_auto());
}

#[test]
fn db_field_value_unwrap() {
let value = DbFieldValue::Value(42.into());
assert_eq!(value.unwrap_value(), 42.into());
}

#[test]
#[should_panic(expected = "called DbValue::unwrap_value() on a wrong DbValue variant")]
fn db_field_value_unwrap_panic() {
let auto_value = DbFieldValue::Auto;
let _ = auto_value.unwrap_value();
}

#[test]
fn db_field_value_expect() {
let value = DbFieldValue::Value(42.into());
assert_eq!(value.expect_value("expected a value"), 42.into());
}

#[test]
#[should_panic(expected = "expected a value")]
fn db_field_value_expect_panic() {
let auto_value = DbFieldValue::Auto;
let _ = auto_value.expect_value("expected a value");
}
}
24 changes: 22 additions & 2 deletions flareon/src/db/migrations.rs
Original file line number Diff line number Diff line change
Expand Up @@ -451,7 +451,7 @@ impl Field {
}
}

#[derive(Debug, Copy, Clone)]
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
struct ForeignKeyReference {
model: Identifier,
field: Identifier,
Expand Down Expand Up @@ -848,7 +848,7 @@ mod tests {
}

#[test]
fn test_field_new() {
fn field_new() {
let field = Field::new(Identifier::new("id"), ColumnType::Integer)
.primary_key()
.auto()
Expand All @@ -861,6 +861,26 @@ mod tests {
assert!(field.null);
}

#[test]
fn field_foreign_key() {
let field = Field::new(Identifier::new("parent"), ColumnType::Integer).foreign_key(
Identifier::new("testapp__parent"),
Identifier::new("id"),
ForeignKeyOnDeletePolicy::Restrict,
ForeignKeyOnUpdatePolicy::Restrict,
);

assert_eq!(
field.foreign_key,
Some(ForeignKeyReference {
model: Identifier::new("testapp__parent"),
field: Identifier::new("id"),
on_delete: ForeignKeyOnDeletePolicy::Restrict,
on_update: ForeignKeyOnUpdatePolicy::Restrict,
})
);
}

#[test]
fn test_migration_wrapper() {
let migration = MigrationWrapper::new(TestMigration);
Expand Down
Loading

0 comments on commit 3a4d342

Please sign in to comment.