1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
use crate::{json_rpc::types::*, CoreError, CoreResult};
use psl::parser_database::SourceFile;
use schema_connector::{migrations_directory::*, DiffTarget, SchemaConnector};
use std::{path::Path, sync::Arc};
use user_facing_errors::schema_engine::MigrationNameTooLong;

/// Create a new migration.
pub async fn create_migration(
    input: CreateMigrationInput,
    connector: &mut dyn SchemaConnector,
) -> CoreResult<CreateMigrationOutput> {
    let connector_type = connector.connector_type();

    if input.migration_name.len() > 200 {
        return Err(CoreError::user_facing(MigrationNameTooLong));
    }

    // Check for provider switch
    error_on_changed_provider(&input.migrations_directory_path, connector_type)?;

    // Infer the migration.
    let previous_migrations = list_migrations(Path::new(&input.migrations_directory_path))?;

    // We need to start with the 'to', which is the Schema, in order to grab the
    // namespaces, in case we've got MultiSchema enabled.
    let to = connector
        .database_schema_from_diff_target(
            DiffTarget::Datamodel(SourceFile::new_allocated(Arc::from(
                input.prisma_schema.into_boxed_str(),
            ))),
            None,
            None,
        )
        .await?;

    let namespaces = connector.extract_namespaces(&to);
    // We pass the namespaces here, because we want to describe all of these namespaces.
    let from = connector
        .database_schema_from_diff_target(DiffTarget::Migrations(&previous_migrations), None, namespaces)
        .await?;
    let migration = connector.diff(from, to);

    if connector.migration_is_empty(&migration) && !input.draft {
        tracing::info!("Database is up-to-date, returning without creating new migration.");

        return Ok(CreateMigrationOutput {
            generated_migration_name: None,
        });
    }

    let destructive_change_diagnostics = connector.destructive_change_checker().pure_check(&migration);

    let migration_script = connector.render_script(&migration, &destructive_change_diagnostics)?;

    // Write the migration script to a file.
    let directory = create_migration_directory(Path::new(&input.migrations_directory_path), &input.migration_name)
        .map_err(|_| CoreError::from_msg("Failed to create a new migration directory.".into()))?;

    directory
        .write_migration_script(&migration_script, connector.migration_file_extension())
        .map_err(|err| {
            CoreError::from_msg(format!(
                "Failed to write the migration script to `{:?}`\n{}",
                directory.path(),
                err,
            ))
        })?;

    write_migration_lock_file(&input.migrations_directory_path, connector_type).map_err(|err| {
        CoreError::from_msg(format!(
            "Failed to write the migration lock file to `{:?}`\n{}",
            &input.migrations_directory_path, err
        ))
    })?;

    Ok(CreateMigrationOutput {
        generated_migration_name: Some(directory.migration_name().to_owned()),
    })
}