368 lines
13 KiB
C#
368 lines
13 KiB
C#
using System.Data;
|
|
using System.Data.Common;
|
|
using LinqToDB;
|
|
using LinqToDB.Common;
|
|
using LinqToDB.Data;
|
|
using LinqToDB.DataProvider;
|
|
using LinqToDB.SqlQuery;
|
|
using Nop.Core;
|
|
using Nop.Data.DataProviders.LinqToDB;
|
|
using Nop.Data.Mapping;
|
|
using Npgsql;
|
|
|
|
namespace Nop.Data.DataProviders;
|
|
|
|
public partial class PostgreSqlDataProvider : BaseDataProvider, INopDataProvider
|
|
{
|
|
#region Fields
|
|
|
|
protected static readonly Lazy<IDataProvider> _dataProvider = new(() => new LinqToDBPostgreSQLDataProvider(), true);
|
|
|
|
#endregion
|
|
|
|
#region Utilities
|
|
|
|
/// <summary>
|
|
/// Creates the database connection by the current data configuration
|
|
/// </summary>
|
|
protected override DataConnection CreateDataConnection()
|
|
{
|
|
var dataContext = CreateDataConnection(LinqToDbDataProvider);
|
|
dataContext.MappingSchema.SetDataType(
|
|
typeof(string),
|
|
new SqlDataType(new DbDataType(typeof(string), "citext")));
|
|
|
|
return dataContext;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the connection string builder
|
|
/// </summary>
|
|
/// <returns>The connection string builder</returns>
|
|
protected static NpgsqlConnectionStringBuilder GetConnectionStringBuilder()
|
|
{
|
|
return new NpgsqlConnectionStringBuilder(GetCurrentConnectionString());
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a connection to the database for a current data provider
|
|
/// </summary>
|
|
/// <param name="connectionString">Connection string</param>
|
|
/// <returns>Connection to a database</returns>
|
|
protected override DbConnection GetInternalDbConnection(string connectionString)
|
|
{
|
|
ArgumentException.ThrowIfNullOrEmpty(connectionString);
|
|
|
|
return new NpgsqlConnection(connectionString);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the name of the sequence associated with a identity column
|
|
/// </summary>
|
|
/// <param name="dataConnection">A database connection object</param>
|
|
/// <typeparam name="TEntity">Entity type</typeparam>
|
|
/// <returns>Returns the name of the sequence, or NULL if no sequence is associated with the column</returns>
|
|
protected virtual string GetSequenceName<TEntity>(DataConnection dataConnection) where TEntity : BaseEntity
|
|
{
|
|
ArgumentNullException.ThrowIfNull(dataConnection);
|
|
|
|
var descriptor = NopMappingSchema.GetEntityDescriptor(typeof(TEntity))
|
|
?? throw new NopException($"Mapped entity descriptor is not found: {typeof(TEntity).Name}");
|
|
|
|
var tableName = descriptor.EntityName;
|
|
var columnName = descriptor.Fields.FirstOrDefault(x => x.IsIdentity && x.IsPrimaryKey)?.Name;
|
|
|
|
if (string.IsNullOrEmpty(columnName))
|
|
throw new NopException("A table's primary key does not have an identity constraint");
|
|
|
|
return dataConnection.Query<string>($"SELECT pg_get_serial_sequence('\"{tableName}\"', '{columnName}');")
|
|
.FirstOrDefault();
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Methods
|
|
|
|
/// <summary>
|
|
/// Creates the database by using the loaded connection string
|
|
/// </summary>
|
|
/// <param name="collation"></param>
|
|
/// <param name="triesToConnect"></param>
|
|
public void CreateDatabase(string collation, int triesToConnect = 10)
|
|
{
|
|
if (DatabaseExists())
|
|
return;
|
|
|
|
var builder = GetConnectionStringBuilder();
|
|
|
|
//gets database name
|
|
var databaseName = builder.Database;
|
|
|
|
//now create connection string to 'postgres' - default administrative connection database.
|
|
builder.Database = "postgres";
|
|
|
|
using (var connection = GetInternalDbConnection(builder.ConnectionString))
|
|
{
|
|
var query = $"CREATE DATABASE \"{databaseName}\" WITH OWNER = '{builder.Username}'";
|
|
if (!string.IsNullOrWhiteSpace(collation))
|
|
query = $"{query} LC_COLLATE = '{collation}'";
|
|
|
|
var command = connection.CreateCommand();
|
|
command.CommandText = query;
|
|
command.Connection.Open();
|
|
|
|
command.ExecuteNonQuery();
|
|
}
|
|
|
|
//try connect
|
|
if (triesToConnect <= 0)
|
|
return;
|
|
|
|
//sometimes on slow servers (hosting) there could be situations when database requires some time to be created.
|
|
//but we have already started creation of tables and sample data.
|
|
//as a result there is an exception thrown and the installation process cannot continue.
|
|
//that's why we are in a cycle of "triesToConnect" times trying to connect to a database with a delay of one second.
|
|
for (var i = 0; i <= triesToConnect; i++)
|
|
{
|
|
if (i == triesToConnect)
|
|
throw new Exception("Unable to connect to the new database. Please try one more time");
|
|
|
|
if (!DatabaseExists())
|
|
{
|
|
Thread.Sleep(1000);
|
|
}
|
|
else
|
|
{
|
|
builder.Database = databaseName;
|
|
using var connection = GetInternalDbConnection(builder.ConnectionString) as NpgsqlConnection;
|
|
var command = connection.CreateCommand();
|
|
command.CommandText = "CREATE EXTENSION IF NOT EXISTS citext; CREATE EXTENSION IF NOT EXISTS pgcrypto;";
|
|
command.Connection.Open();
|
|
command.ExecuteNonQuery();
|
|
connection.ReloadTypes();
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks if the specified database exists, returns true if database exists
|
|
/// </summary>
|
|
/// <returns>Returns true if the database exists.</returns>
|
|
public bool DatabaseExists()
|
|
{
|
|
try
|
|
{
|
|
using var connection = GetInternalDbConnection(GetCurrentConnectionString());
|
|
|
|
//just try to connect
|
|
connection.Open();
|
|
|
|
return true;
|
|
}
|
|
catch
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks if the specified database exists, returns true if database exists
|
|
/// </summary>
|
|
/// <returns>
|
|
/// A task that represents the asynchronous operation
|
|
/// The task result contains the returns true if the database exists.
|
|
/// </returns>
|
|
public async Task<bool> DatabaseExistsAsync()
|
|
{
|
|
try
|
|
{
|
|
await using var connection = GetInternalDbConnection(GetCurrentConnectionString());
|
|
|
|
//just try to connect
|
|
await connection.OpenAsync();
|
|
|
|
return true;
|
|
}
|
|
catch
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the current identity value
|
|
/// </summary>
|
|
/// <typeparam name="TEntity">Entity type</typeparam>
|
|
/// <returns>
|
|
/// A task that represents the asynchronous operation
|
|
/// The task result contains the integer identity; null if cannot get the result
|
|
/// </returns>
|
|
public virtual Task<int?> GetTableIdentAsync<TEntity>() where TEntity : BaseEntity
|
|
{
|
|
using var currentConnection = CreateDataConnection();
|
|
|
|
var seqName = GetSequenceName<TEntity>(currentConnection);
|
|
|
|
var result = currentConnection.Query<int>($"SELECT COALESCE(last_value + CASE WHEN is_called THEN 1 ELSE 0 END, 1) as Value FROM {seqName};")
|
|
.FirstOrDefault();
|
|
|
|
return Task.FromResult<int?>(result);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Set table identity (is supported)
|
|
/// </summary>
|
|
/// <typeparam name="TEntity">Entity type</typeparam>
|
|
/// <param name="ident">Identity value</param>
|
|
/// <returns>A task that represents the asynchronous operation</returns>
|
|
public virtual async Task SetTableIdentAsync<TEntity>(int ident) where TEntity : BaseEntity
|
|
{
|
|
var currentIdent = await GetTableIdentAsync<TEntity>();
|
|
if (!currentIdent.HasValue || ident <= currentIdent.Value)
|
|
return;
|
|
|
|
using var currentConnection = CreateDataConnection();
|
|
|
|
var seqName = GetSequenceName<TEntity>(currentConnection);
|
|
|
|
await currentConnection.ExecuteAsync($"select setval('{seqName}', {ident}, false);");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a backup of the database
|
|
/// </summary>
|
|
/// <returns>A task that represents the asynchronous operation</returns>
|
|
public virtual Task BackupDatabaseAsync(string fileName)
|
|
{
|
|
throw new DataException("This database provider does not support backup");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Inserts record into table. Returns inserted entity with identity
|
|
/// </summary>
|
|
/// <param name="entity"></param>
|
|
/// <typeparam name="TEntity"></typeparam>
|
|
/// <returns>Inserted entity</returns>
|
|
public override TEntity InsertEntity<TEntity>(TEntity entity)
|
|
{
|
|
using var dataContext = CreateDataConnection();
|
|
try
|
|
{
|
|
entity.Id = dataContext.InsertWithInt32Identity(entity);
|
|
}
|
|
// Ignore when we try insert foreign entity via InsertWithInt32IdentityAsync method
|
|
catch (SqlException ex) when (ex.Message.StartsWith("Identity field must be defined for"))
|
|
{
|
|
dataContext.Insert(entity);
|
|
}
|
|
|
|
return entity;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Inserts record into table. Returns inserted entity with identity
|
|
/// </summary>
|
|
/// <param name="entity"></param>
|
|
/// <typeparam name="TEntity"></typeparam>
|
|
/// <returns>
|
|
/// A task that represents the asynchronous operation
|
|
/// The task result contains the inserted entity
|
|
/// </returns>
|
|
public override async Task<TEntity> InsertEntityAsync<TEntity>(TEntity entity)
|
|
{
|
|
using var dataContext = CreateDataConnection();
|
|
try
|
|
{
|
|
entity.Id = await dataContext.InsertWithInt32IdentityAsync(entity);
|
|
}
|
|
// Ignore when we try insert foreign entity via InsertWithInt32IdentityAsync method
|
|
catch (SqlException ex) when (ex.Message.StartsWith("Identity field must be defined for"))
|
|
{
|
|
await dataContext.InsertAsync(entity);
|
|
}
|
|
|
|
return entity;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Restores the database from a backup
|
|
/// </summary>
|
|
/// <param name="backupFileName">The name of the backup file</param>
|
|
/// <returns>A task that represents the asynchronous operation</returns>
|
|
public virtual Task RestoreDatabaseAsync(string backupFileName)
|
|
{
|
|
throw new DataException("This database provider does not support backup");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Re-index database tables
|
|
/// </summary>
|
|
/// <returns>A task that represents the asynchronous operation</returns>
|
|
public virtual async Task ReIndexTablesAsync()
|
|
{
|
|
using var currentConnection = CreateDataConnection();
|
|
await currentConnection.ExecuteAsync($"REINDEX DATABASE \"{currentConnection.Connection.Database}\";");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Build the connection string
|
|
/// </summary>
|
|
/// <param name="nopConnectionString">Connection string info</param>
|
|
/// <returns>Connection string</returns>
|
|
public virtual string BuildConnectionString(INopConnectionStringInfo nopConnectionString)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(nopConnectionString);
|
|
|
|
if (nopConnectionString.IntegratedSecurity)
|
|
throw new NopException("Data provider supports connection only with login and password");
|
|
|
|
var builder = new NpgsqlConnectionStringBuilder
|
|
{
|
|
Host = nopConnectionString.ServerName,
|
|
//Cast DatabaseName to lowercase to avoid case-sensitivity problems
|
|
Database = nopConnectionString.DatabaseName.ToLowerInvariant(),
|
|
Username = nopConnectionString.Username,
|
|
Password = nopConnectionString.Password,
|
|
};
|
|
|
|
return builder.ConnectionString;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the name of a foreign key
|
|
/// </summary>
|
|
/// <param name="foreignTable">Foreign key table</param>
|
|
/// <param name="foreignColumn">Foreign key column name</param>
|
|
/// <param name="primaryTable">Primary table</param>
|
|
/// <param name="primaryColumn">Primary key column name</param>
|
|
/// <returns>Name of a foreign key</returns>
|
|
public virtual string CreateForeignKeyName(string foreignTable, string foreignColumn, string primaryTable, string primaryColumn)
|
|
{
|
|
return $"FK_{foreignTable}_{foreignColumn}_{primaryTable}_{primaryColumn}";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the name of an index
|
|
/// </summary>
|
|
/// <param name="targetTable">Target table name</param>
|
|
/// <param name="targetColumn">Target column name</param>
|
|
/// <returns>Name of an index</returns>
|
|
public virtual string GetIndexName(string targetTable, string targetColumn)
|
|
{
|
|
return $"IX_{targetTable}_{targetColumn}";
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Properties
|
|
|
|
protected override IDataProvider LinqToDbDataProvider => _dataProvider.Value;
|
|
|
|
public int SupportedLengthOfBinaryHash => 0;
|
|
|
|
public bool BackupSupported => false;
|
|
|
|
#endregion
|
|
} |