Skip to content

Commit

Permalink
Add/unify runtime Parameters support for SQL script execution. (#32)
Browse files Browse the repository at this point in the history
  • Loading branch information
chullybun authored Dec 20, 2022
1 parent 3f242e0 commit e46261b
Show file tree
Hide file tree
Showing 15 changed files with 222 additions and 39 deletions.
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,11 @@

Represents the **NuGet** versions.

## v2.2.0
- *Enhancement:* Enable `Parameters` to be passed via the command line; either adding, or overridding any pre-configured values. Use `-p|--param Name=Value` syntax; e.g. `--param JournalSchema=dbo`.
- *Enhancement:* Enable moustache syntax property placeholder replacements (e.g`{{ParameterName}}`), from the `Parameters`, within SQL scripts to allow changes during execution into the database at runtime.
- *Enhancement:* Added command-line confirmation prompt for a `Drop` or `Reset` as these are considered highly destructive actions. Supports `--accept-prompts` option to bypass prompts within scripted scenarios.

## v2.1.1
- *Fixed:* Multibyte support added to the `DataParser` insert and merge for SQL Server strings using the `N` prefix.

Expand Down
2 changes: 1 addition & 1 deletion Common.targets
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
<Project>
<PropertyGroup>
<Version>2.1.1</Version>
<Version>2.2.0</Version>
<LangVersion>preview</LangVersion>
<Authors>Avanade</Authors>
<Company>Avanade</Company>
Expand Down
19 changes: 14 additions & 5 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -90,6 +90,8 @@ Over time there will be more than one script updating a single object, for examp

The migration scripts must be marked as embedded resources, and reside under the `Migrations` folder within the c# project. A naming convention should be used to ensure they are to be executed in the correct order; it is recommended that the name be prefixed by the date and time, followed by a brief description of the purpose. For example: `20181218-081540-create-demo-person-table.sql`

A migration script can contain basic moustache value replacement syntax such as `{{Company}}`, this will then be replaced at runtime by the corresponding `Company` parameter value; see [`MigrationArgs.Parameters`](./src/DbEx/Migration/MigrationArgsBase.cs). These parameters (`Name=Value` pairs) can also be command-line specified.

It is recommended that each script be enclosed by a transaction that can be rolled back in the case of error; otherwise, a script could be partially applied and will then need manual intervention to resolve.

_Note_: There are _special case_ scripts that will be executed pre- and post- migrations. In that any scripts ending with `.pre.deploy.sql` will always be executed before the migrations are attempted, and any scripts ending with `.post.deploy.sql` will always be executed after all the migrations have successfully executed.
Expand All @@ -108,6 +110,8 @@ The currently supported objects are (order specified implies order in which they

The schema scripts must be marked as embedded resources, and reside under the `Schema` folder within the c# project. Each script should only contain a single `Create` statement. Each script will be parsed to determine type so that the appropriate order can be applied.

A schema script script can contain basic moustache value replacement syntax such as `{{Company}}`, this will then be replaced at runtime by the corresponding `Company` parameter value; see [`MigrationArgs.Parameters`](./src/DbEx/Migration/MigrationArgsBase.cs). These parameters (`Name=Value` pairs) can also be command-line specified.

The `Schema` folder is used to encourage the usage of database schemas. Therefore, directly under should be the schema name, for example `dbo` or `Ref`. Then sub-folders for the object types as per [Azure Data Studio](https://docs.microsoft.com/en-au/sql/azure-data-studio/what-is), for example `Functions`, `Stored Procedures` or `Types\User-Defined Table Types`.

<br/>
Expand Down Expand Up @@ -171,7 +175,7 @@ Demo:
- { FirstName: Wendy, LastName: Jones, Gender: F, Birthday: 1985-03-18 }
```

Finally, runtime values can be used within the YAML using the value lookup notation; this notation is `^(Key)`. This will either reference the [`DataParserArgs`](./src/DbEx/Migration/Data/DataParserArgs.cs) `RuntimeParameters` property using the specified key. There are two special parameters, being `UserName` and `DateTimeNow`, that reference the same named `DataParserArgs` properties. Where not found the extended notation `^(Namespace.Type.Property.Method().etc, AssemblyName)` is used. Where the `AssemblyName` is not specified then the default `mscorlib` is assumed. The `System` root namespace is optional, i.e. it will be attempted by default. The initial property or method for a `Type` must be `static`, in that the `Type` will not be instantiated. Example as follows.
Finally, runtime values can be used within the YAML using the value lookup notation; this notation is `^(Key)`. This will either reference the [`DataParserArgs`](./src/DbEx/Migration/Data/DataParserArgs.cs) `Parameters` property using the specified key. There are two special parameters, being `UserName` and `DateTimeNow`, that reference the same named `DataParserArgs` properties. Where not found the extended notation `^(Namespace.Type.Property.Method().etc, AssemblyName)` is used. Where the `AssemblyName` is not specified then the default `mscorlib` is assumed. The `System` root namespace is optional, i.e. it will be attempted by default. The initial property or method for a `Type` must be `static`, in that the `Type` will not be instantiated. Example as follows. These parameters (`Name=Value` pairs) can also be command-line specified.

``` yaml
Demo:
Expand All @@ -194,8 +198,9 @@ Xxx Database Tool.
Usage: Xxx [options] <command> <args>

Arguments:
command Database migration command.
Allowed values are: None, Drop, Create, Migrate, Schema, Deploy, Reset, Data, DeployWithData, All, DropAndAll, ResetAndData, ResetAndAll, Execute, Script.
command Database migration command (see https://github.com/Avanade/dbex#commands-functions).
Allowed values are: None, Drop, Create, Migrate, CodeGen, Schema, Deploy, Reset, Data, DeployWithData, Database, DropAndDatabase, All, DropAndAll,
ResetAndData, ResetAndDatabase, ResetAndAll, Execute, Script.
args Additional arguments; 'Script' arguments (first being the script name) -or- 'Execute' (each a SQL statement to invoke).

Options:
Expand All @@ -205,7 +210,9 @@ Options:
-so|--schema-order Database schema name (multiple can be specified in priority order).
-o|--output Output directory path.
-a|--assembly Assembly containing embedded resources (multiple can be specified in probing order).
-p|--param Parameter expressed as a 'Name=Value' pair (multiple can be specified).
-eo|--entry-assembly-only Use the entry assembly only (ignore all other assemblies).
--accept-prompts Accept prompts; command should _not_ stop and wait for user confirmation (DROP or RESET commands).
```
The [`DbEx.Test.Console`](./tests/DbEx.Test.Console) demonstrates how this can be leveraged. The command-line arguments need to be passed through to support the standard options. Additional methods exist to specify defaults or change behaviour as required. An example [`Program.cs`](./tests/DbEx.Test.Console/Program.cs) is as follows.
Expand Down Expand Up @@ -243,6 +250,8 @@ To simplify the process for the developer _DbEx_ enables the creation of new mig

This requires the usage of the `Script` command, plus zero or more optional arguments where the first is the sub-command (these are will depend on the script being created). The optional arguments must appear in the order listed; where not specified it will default within the script file. Depending on the database provider not all of the following will be supported.

The following shows the `Script` sub-commands for SQL Server. Use `--help` to see the commands available at rubntime.

Sub-command | Argument(s) | Description
-|-|-
[N/A](./src/DbEx.SqlServer/Resources/ScriptDefault_sql.hbs) | N/A | Creates a new empty skeleton script file.
Expand Down Expand Up @@ -290,10 +299,10 @@ The [`Database`](./src/DbEx/DatabaseExtensions.cs) class provides a `SelectSchem

To simplify the database management here are some further considerations that may make life easier over time; especially where you adopt the philosophy that the underlying busines logic (within the application APIs) is primarily responsible for the consistency of the data; and the data source (the database) is being largely used for storage and advanced query:

- **Nullable everything** - all columns (except) the primary key should be defined as nullable. The business logic should validate the request to ensure data is provided where mandatory. Makes changes to the database schema easier over time without this constraint.
- **Minimise constraints** - do not use database constraints unless absolutely necessary; only leverage where the database is the best and/or most efficient means to perform; i.e. uniqueness. The business logic should validate the request to ensure that any related data is provided, is valid and consistent.
- **No cross-schema referencing** - avoid referencing across `Schemas` where possible as this will impact the Migrations as part of this tooling; and we should not be using constraints as per prior point. Each schema is considered independent of others except in special cases, such as `dbo` or `sec` (security where used) for example.
- **No cross-schema referencing** - avoid referencing across `Schemas` where possible as this may impact the Migrations as part of this tooling; and we should not be using constraints as per prior point. Each schema is considered independent of others (where using a schema per domain) except in special cases, such as `dbo` or `sec` (security where used) for example.
- **JSON for schema-less** - where there is data that needs to be persisted, but rarely searched on, a schema-less approach should be considered such that a JSON object is persisted into a single column versus having to define additional tables and/or columns. This can further simplify the database requirements where the data is hierarchical in nature. To enable the [`ObjectToJsonConverter`](https://github.com/Avanade/CoreEx/blob/main/src/CoreEx/Mapping/Converters/ObjectToJsonConverter.cs) and [`AutoMapperObjectToJsonConverter`](https://github.com/Avanade/CoreEx/blob/main/src/CoreEx.AutoMapper/Converters/AutoMapperObjectToJsonConverter.cs) can be used within the corresponding mapper to enable.
- **Nullable everything** - all columns (except) the primary key should be defined as nullable. The business logic should validate the request to ensure data is provided where mandatory. Makes changes to the database schema easier over time without this constraint.

<br/>

Expand Down
13 changes: 6 additions & 7 deletions src/DbEx.MySql/Migration/MySqlMigration.cs
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,9 @@
using DbEx.DbSchema;
using DbEx.Migration;
using DbUp.Support;
using Microsoft.Extensions.Logging;
using MySql.Data.MySqlClient;
using OnRamp.Utility;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;

Expand All @@ -28,7 +25,7 @@ public class MySqlMigration : DatabaseMigrationBase
private readonly string _databaseName;
private readonly IDatabase _database;
private readonly IDatabase _masterDatabase;
private List<string> _resetBypass = new List<string>();
private readonly List<string> _resetBypass = new();

/// <summary>
/// Initializes an instance of the <see cref="MySqlMigration"/> class.
Expand All @@ -50,8 +47,10 @@ public MySqlMigration(MigrationArgsBase args) : base(args)
if (SchemaObjectTypes.Length == 0)
SchemaObjectTypes = new string[] { "FUNCTION", "VIEW", "PROCEDURE" };

Journal.Schema = null;
Journal.Table = "schemaversions";
// Add/set standard parameters.
Args.Parameter(MigrationArgsBase.DatabaseNameParamName, _databaseName, true);
Args.Parameter(MigrationArgsBase.JournalSchemaParamName, null, true);
Args.Parameter(MigrationArgsBase.JournalTableParamName, "schemaversions");
}

/// <inheritdoc/>
Expand Down Expand Up @@ -92,7 +91,7 @@ protected override async Task ExecuteScriptAsync(DatabaseMigrationScript script,

foreach (var sql in new SqlCommandSplitter().SplitScriptIntoCommands(sr.ReadToEnd()))
{
await Database.SqlStatement(sql).NonQueryAsync(cancellationToken).ConfigureAwait(false);
await Database.SqlStatement(ReplaceSqlRuntimeParameters(sql)).NonQueryAsync(cancellationToken).ConfigureAwait(false);
}
}
}
Expand Down
10 changes: 6 additions & 4 deletions src/DbEx.SqlServer/Migration/SqlServerMigration.cs
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ public class SqlServerMigration : DatabaseMigrationBase
private readonly string _databaseName;
private readonly IDatabase _database;
private readonly IDatabase _masterDatabase;
private List<string> _resetBypass = new List<string>();
private readonly List<string> _resetBypass = new();

/// <summary>
/// Initializes an instance of the <see cref="SqlServerMigration"/> class.
Expand Down Expand Up @@ -58,8 +58,10 @@ public SqlServerMigration(MigrationArgsBase args) : base(args)
if (!Args.SchemaOrder.Contains("dbo"))
Args.SchemaOrder.Insert(0, "dbo");

Journal.Schema = "dbo";
Journal.Table = "SchemaVersions";
// Add/set standard parameters.
Args.Parameter(MigrationArgsBase.DatabaseNameParamName, _databaseName, true);
Args.Parameter(MigrationArgsBase.JournalSchemaParamName, "dbo");
Args.Parameter(MigrationArgsBase.JournalTableParamName, "SchemaVersions");
}

/// <inheritdoc/>
Expand Down Expand Up @@ -109,7 +111,7 @@ protected override async Task ExecuteScriptAsync(DatabaseMigrationScript script,

foreach (var sql in new SqlCommandSplitter().SplitScriptIntoCommands(sr.ReadToEnd()))
{
await Database.SqlStatement(sql).NonQueryAsync(cancellationToken).ConfigureAwait(false);
await Database.SqlStatement(ReplaceSqlRuntimeParameters(sql)).NonQueryAsync(cancellationToken).ConfigureAwait(false);
}
}
}
Expand Down
36 changes: 33 additions & 3 deletions src/DbEx/Console/MigrationConsoleBase.cs
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,8 @@ namespace DbEx.Console
/// <para>The underlying command line parsing is provided by <see href="https://natemcmaster.github.io/CommandLineUtils/"/>.</para></remarks>
public abstract class MigrationConsoleBase
{
private const string EntryAssemblyOnlyOptionName = "EO";
private const string EntryAssemblyOnlyOptionName = "entry-assembly-only";
private const string AcceptPromptsOptionName = "accept-prompts";
private CommandArgument<MigrationCommand>? _commandArg;
private CommandArgument? _additionalArgs;
private CommandOption? _helpOption;
Expand Down Expand Up @@ -121,7 +122,9 @@ public async Task<int> RunAsync(string[] args, CancellationToken cancellationTok
ConsoleOptions.Add(nameof(MigrationArgs.SchemaOrder), app.Option("-so|--schema-order", "Database schema name (multiple can be specified in priority order).", CommandOptionType.MultipleValue));
ConsoleOptions.Add(nameof(MigrationArgs.OutputDirectory), app.Option("-o|--output", "Output directory path.", CommandOptionType.MultipleValue).Accepts(v => v.ExistingDirectory("Output directory path does not exist.")));
ConsoleOptions.Add(nameof(MigrationArgs.Assemblies), app.Option("-a|--assembly", "Assembly containing embedded resources (multiple can be specified in probing order).", CommandOptionType.MultipleValue));
ConsoleOptions.Add(nameof(MigrationArgs.Parameters), app.Option("-p|--param", "Parameter expressed as a 'Name=Value' pair (multiple can be specified).", CommandOptionType.MultipleValue));
ConsoleOptions.Add(EntryAssemblyOnlyOptionName, app.Option("-eo|--entry-assembly-only", "Use the entry assembly only (ignore all other assemblies).", CommandOptionType.NoValue));
ConsoleOptions.Add(AcceptPromptsOptionName, app.Option("--accept-prompts", "Accept prompts; command should _not_ stop and wait for user confirmation (DROP or RESET commands).", CommandOptionType.NoValue));
_additionalArgs = app.Argument("args", "Additional arguments; 'Script' arguments (first being the script name) -or- 'Execute' (each a SQL statement to invoke).", multipleValues: true);

OnBeforeExecute(app);
Expand Down Expand Up @@ -153,6 +156,10 @@ public async Task<int> RunAsync(string[] args, CancellationToken cancellationTok
Args.AddAssembly(Assembly.GetEntryAssembly()!);
});

vr = ValidateMultipleValue(nameof(MigrationArgs.Parameters), ctx, (ctx, co) => new ParametersValidator(Args).GetValidationResult(co, ctx));
if (vr != ValidationResult.Success)
return vr;

if (_additionalArgs.Values.Count > 0 && !(Args.MigrationCommand.HasFlag(MigrationCommand.Script) || Args.MigrationCommand.HasFlag(MigrationCommand.Execute)))
return new ValidationResult($"Additional arguments can only be specified when the command is '{nameof(MigrationCommand.Script)}' or '{nameof(MigrationCommand.Execute)}'.", new string[] { "args" });

Expand Down Expand Up @@ -191,7 +198,25 @@ public async Task<int> RunAsync(string[] args, CancellationToken cancellationTok
Args.OverrideConnectionString(cs?.Value());

// Invoke any additional.
return OnValidation(ctx)!;
var res = OnValidation(ctx)!;

// Action any command input.
var nco = GetCommandOption(AcceptPromptsOptionName);
if (nco == null || !nco.HasValue())
{
if (Args.MigrationCommand.HasFlag(MigrationCommand.Drop))
{
if (!Prompt.GetYesNo("DROP: Confirm that where the specified database already exists it should be dropped?", false, ConsoleColor.Yellow))
return new ValidationResult("Database drop was not confirmed; no execution occurred.");
}
else if (Args.MigrationCommand.HasFlag(MigrationCommand.Reset))
{
if (!Prompt.GetYesNo("RESET: Confirm that the existing data within the database should be reset (deleted)?", false, ConsoleColor.Yellow))
return new ValidationResult("Data reset was not confirmed; no execution occurred.");
}
}

return res;
});

// Set up the code generation execution.
Expand Down Expand Up @@ -385,12 +410,17 @@ public static void WriteStandardizedArgs(DatabaseMigrationBase migrator, Action<

migrator.Args.Logger.LogInformation("{Content}", $"Command = {migrator.Args.MigrationCommand}");
migrator.Args.Logger.LogInformation("{Content}", $"Provider = {migrator.Provider}");
migrator.Args.Logger.LogInformation("{Content}", $"Database = {migrator.DatabaseName}");
migrator.Args.Logger.LogInformation("{Content}", $"SchemaOrder = {string.Join(", ", migrator.Args.SchemaOrder.ToArray())}");
migrator.Args.Logger.LogInformation("{Content}", $"OutDir = {migrator.Args.OutputDirectory?.FullName}");

additional?.Invoke(migrator.Args.Logger);

migrator.Args.Logger.LogInformation("{Content}", $"Parameters{(migrator.Args.Parameters.Count == 0 ? " = none" : ":")}");
foreach (var p in migrator.Args.Parameters.OrderBy(x => x.Key))
{
migrator.Args.Logger.LogInformation("{Content}", $" {p.Key} = {p.Value}");
}

migrator.Args.Logger.LogInformation("{Content}", $"Assemblies{(migrator.Args.Assemblies.Count == 0 ? " = none" : ":")}");
foreach (var a in migrator.Args.Assemblies)
{
Expand Down
Loading

0 comments on commit e46261b

Please sign in to comment.