V10: fix build warnings infrastructure (#12369)
* Run code cleanup * Run dotnet format * Start manual fixes * Manual fixing of warnings * Fix nullability in columnalias * Fix tests * Fix up after merge * Start updating after review * Update editorconfig to contain new static & const rules * Fix up editorconfig to not contain duplicate rules * Fix up static member names * Fix up according to review * Update src/Umbraco.Infrastructure/DependencyInjection/UmbracoBuilder.DistributedCache.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/DependencyInjection/UmbracoBuilder.Repositories.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/DependencyInjection/UmbracoBuilder.Repositories.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Examine/ContentIndexPopulator.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Examine/ContentIndexPopulator.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Examine/ContentValueSetValidator.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Examine/ContentValueSetValidator.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Examine/ContentValueSetValidator.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Examine/ExamineUmbracoIndexingHandler.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Examine/PublishedContentIndexPopulator.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Extensions/InstanceIdentifiableExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/HostedServices/RecurringHostedServiceBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/HostedServices/ReportSiteTask.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Logging/Serilog/LoggerConfigExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Logging/Serilog/LoggerConfigExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Macros/MacroTagParser.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Macros/MacroTagParser.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Macros/MacroTagParser.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Migrations/Expressions/Alter/Table/IAlterTableColumnOptionBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Migrations/Upgrade/V_10_0_0/AddMemberPropertiesAsColumns.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/ModelsBuilder/Building/TextBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/ModelsBuilder/Building/TextBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/ModelsBuilder/Building/TextBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/ModelsBuilder/Building/TextBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/ModelsBuilder/Building/TextBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/ModelsBuilder/Building/TextBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/ModelsBuilder/Building/TextBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Dtos/ExternalLoginDto.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/AccessMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/AuditEntryMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/MediaMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/MemberMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/PropertyGroupMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/PropertyGroupMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/PropertyTypeMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/PropertyTypeMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/RelationTypeMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Mappers/RelationTypeMapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/NPocoMapperCollectionBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Querying/ExpressionVisitorBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Infrastructure/Persistence/Repositories/Implement/ExternalLoginRepository.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Fix [..] to substring * Fix after merge with 10/dev * Fox ContentValueSetValidator.cs * Update LoggerConfigExtensions Co-authored-by: Nikolaj Geisle <niko737@edu.ucl.dk> Co-authored-by: Mole <nikolajlauridsen@protonmail.ch>
This commit is contained in:
@@ -1,6 +1,5 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Data;
|
||||
using System.Data.Common;
|
||||
using System.Text.RegularExpressions;
|
||||
using Microsoft.Data.SqlClient;
|
||||
using NPoco;
|
||||
@@ -9,290 +8,320 @@ using Umbraco.Cms.Infrastructure.Persistence;
|
||||
using Umbraco.Cms.Infrastructure.Persistence.FaultHandling;
|
||||
using Umbraco.Cms.Infrastructure.Persistence.SqlSyntax;
|
||||
|
||||
namespace Umbraco.Extensions
|
||||
namespace Umbraco.Extensions;
|
||||
|
||||
/// <summary>
|
||||
/// Provides extension methods to NPoco Database class.
|
||||
/// </summary>
|
||||
public static partial class NPocoDatabaseExtensions
|
||||
{
|
||||
/// <summary>
|
||||
/// Provides extension methods to NPoco Database class.
|
||||
/// Iterates over the result of a paged data set with a db reader
|
||||
/// </summary>
|
||||
public static partial class NPocoDatabaseExtensions
|
||||
/// <typeparam name="T"></typeparam>
|
||||
/// <param name="database"></param>
|
||||
/// <param name="pageSize">
|
||||
/// The number of rows to load per page
|
||||
/// </param>
|
||||
/// <param name="sql"></param>
|
||||
/// <param name="sqlCount">
|
||||
/// Specify a custom Sql command to get the total count, if null is specified than the
|
||||
/// auto-generated sql count will be used
|
||||
/// </param>
|
||||
/// <returns></returns>
|
||||
/// <remarks>
|
||||
/// NPoco's normal Page returns a List{T} but sometimes we don't want all that in memory and instead want to
|
||||
/// iterate over each row with a reader using Query vs Fetch.
|
||||
/// </remarks>
|
||||
public static IEnumerable<T> QueryPaged<T>(this IDatabase database, long pageSize, Sql sql, Sql? sqlCount)
|
||||
{
|
||||
/// <summary>
|
||||
/// Iterates over the result of a paged data set with a db reader
|
||||
/// </summary>
|
||||
/// <typeparam name="T"></typeparam>
|
||||
/// <param name="database"></param>
|
||||
/// <param name="pageSize">
|
||||
/// The number of rows to load per page
|
||||
/// </param>
|
||||
/// <param name="sql"></param>
|
||||
/// <param name="sqlCount">Specify a custom Sql command to get the total count, if null is specified than the auto-generated sql count will be used</param>
|
||||
/// <returns></returns>
|
||||
/// <remarks>
|
||||
/// NPoco's normal Page returns a List{T} but sometimes we don't want all that in memory and instead want to
|
||||
/// iterate over each row with a reader using Query vs Fetch.
|
||||
/// </remarks>
|
||||
public static IEnumerable<T> QueryPaged<T>(this IDatabase database, long pageSize, Sql sql, Sql? sqlCount)
|
||||
{
|
||||
var sqlString = sql.SQL;
|
||||
var sqlArgs = sql.Arguments;
|
||||
var sqlString = sql.SQL;
|
||||
var sqlArgs = sql.Arguments;
|
||||
|
||||
int? itemCount = null;
|
||||
long pageIndex = 0;
|
||||
do
|
||||
int? itemCount = null;
|
||||
long pageIndex = 0;
|
||||
do
|
||||
{
|
||||
// Get the paged queries
|
||||
database.BuildPageQueries<T>(pageIndex * pageSize, pageSize, sqlString, ref sqlArgs, out var generatedSqlCount, out var sqlPage);
|
||||
|
||||
// get the item count once
|
||||
if (itemCount == null)
|
||||
{
|
||||
// Get the paged queries
|
||||
database.BuildPageQueries<T>(pageIndex * pageSize, pageSize, sqlString, ref sqlArgs, out var generatedSqlCount, out var sqlPage);
|
||||
itemCount = database.ExecuteScalar<int>(
|
||||
sqlCount?.SQL ?? generatedSqlCount,
|
||||
sqlCount?.Arguments ?? sqlArgs);
|
||||
}
|
||||
|
||||
// get the item count once
|
||||
if (itemCount == null)
|
||||
{
|
||||
itemCount = database.ExecuteScalar<int>(sqlCount?.SQL ?? generatedSqlCount, sqlCount?.Arguments ?? sqlArgs);
|
||||
}
|
||||
pageIndex++;
|
||||
pageIndex++;
|
||||
|
||||
// iterate over rows without allocating all items to memory (Query vs Fetch)
|
||||
foreach (var row in database.Query<T>(sqlPage, sqlArgs))
|
||||
{
|
||||
yield return row;
|
||||
}
|
||||
// iterate over rows without allocating all items to memory (Query vs Fetch)
|
||||
foreach (T row in database.Query<T>(sqlPage, sqlArgs))
|
||||
{
|
||||
yield return row;
|
||||
}
|
||||
}
|
||||
while (pageIndex * pageSize < itemCount);
|
||||
}
|
||||
|
||||
} while ((pageIndex * pageSize) < itemCount);
|
||||
/// <summary>
|
||||
/// Iterates over the result of a paged data set with a db reader
|
||||
/// </summary>
|
||||
/// <typeparam name="T"></typeparam>
|
||||
/// <param name="database"></param>
|
||||
/// <param name="pageSize">
|
||||
/// The number of rows to load per page
|
||||
/// </param>
|
||||
/// <param name="sql"></param>
|
||||
/// <returns></returns>
|
||||
/// <remarks>
|
||||
/// NPoco's normal Page returns a List{T} but sometimes we don't want all that in memory and instead want to
|
||||
/// iterate over each row with a reader using Query vs Fetch.
|
||||
/// </remarks>
|
||||
public static IEnumerable<T> QueryPaged<T>(this IDatabase database, long pageSize, Sql sql) =>
|
||||
database.QueryPaged<T>(pageSize, sql, null);
|
||||
|
||||
// NOTE
|
||||
//
|
||||
// proper way to do it with TSQL and SQLCE
|
||||
// IF EXISTS (SELECT ... FROM table WITH (UPDLOCK,HOLDLOCK)) WHERE ...)
|
||||
// BEGIN
|
||||
// UPDATE table SET ... WHERE ...
|
||||
// END
|
||||
// ELSE
|
||||
// BEGIN
|
||||
// INSERT INTO table (...) VALUES (...)
|
||||
// END
|
||||
//
|
||||
// works in READ COMMITED, TSQL & SQLCE lock the constraint even if it does not exist, so INSERT is OK
|
||||
//
|
||||
// TODO: use the proper database syntax, not this kludge
|
||||
|
||||
/// <summary>
|
||||
/// Safely inserts a record, or updates if it exists, based on a unique constraint.
|
||||
/// </summary>
|
||||
/// <param name="db"></param>
|
||||
/// <param name="poco"></param>
|
||||
/// <returns>
|
||||
/// The action that executed, either an insert or an update. If an insert occurred and a PK value got generated, the
|
||||
/// poco object
|
||||
/// passed in will contain the updated value.
|
||||
/// </returns>
|
||||
/// <remarks>
|
||||
/// <para>
|
||||
/// We cannot rely on database-specific options because SQLCE
|
||||
/// does not support any of them. Ideally this should be achieved with proper transaction isolation levels but that
|
||||
/// would mean revisiting
|
||||
/// isolation levels globally. We want to keep it simple for the time being and manage it manually.
|
||||
/// </para>
|
||||
/// <para>We handle it by trying to update, then insert, etc. until something works, or we get bored.</para>
|
||||
/// <para>
|
||||
/// Note that with proper transactions, if T2 begins after T1 then we are sure that the database will contain T2's
|
||||
/// value
|
||||
/// once T1 and T2 have completed. Whereas here, it could contain T1's value.
|
||||
/// </para>
|
||||
/// </remarks>
|
||||
public static RecordPersistenceType InsertOrUpdate<T>(this IUmbracoDatabase db, T poco)
|
||||
where T : class =>
|
||||
db.InsertOrUpdate(poco, null, null);
|
||||
|
||||
/// <summary>
|
||||
/// Safely inserts a record, or updates if it exists, based on a unique constraint.
|
||||
/// </summary>
|
||||
/// <param name="db"></param>
|
||||
/// <param name="poco"></param>
|
||||
/// <param name="updateArgs"></param>
|
||||
/// <param name="updateCommand">If the entity has a composite key they you need to specify the update command explicitly</param>
|
||||
/// <returns>
|
||||
/// The action that executed, either an insert or an update. If an insert occurred and a PK value got generated, the
|
||||
/// poco object
|
||||
/// passed in will contain the updated value.
|
||||
/// </returns>
|
||||
/// <remarks>
|
||||
/// <para>
|
||||
/// We cannot rely on database-specific options because SQLCE
|
||||
/// does not support any of them. Ideally this should be achieved with proper transaction isolation levels but that
|
||||
/// would mean revisiting
|
||||
/// isolation levels globally. We want to keep it simple for the time being and manage it manually.
|
||||
/// </para>
|
||||
/// <para>We handle it by trying to update, then insert, etc. until something works, or we get bored.</para>
|
||||
/// <para>
|
||||
/// Note that with proper transactions, if T2 begins after T1 then we are sure that the database will contain T2's
|
||||
/// value
|
||||
/// once T1 and T2 have completed. Whereas here, it could contain T1's value.
|
||||
/// </para>
|
||||
/// </remarks>
|
||||
public static RecordPersistenceType InsertOrUpdate<T>(
|
||||
this IUmbracoDatabase db,
|
||||
T poco,
|
||||
string? updateCommand,
|
||||
object? updateArgs)
|
||||
where T : class
|
||||
{
|
||||
if (poco == null)
|
||||
{
|
||||
throw new ArgumentNullException(nameof(poco));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Iterates over the result of a paged data set with a db reader
|
||||
/// </summary>
|
||||
/// <typeparam name="T"></typeparam>
|
||||
/// <param name="database"></param>
|
||||
/// <param name="pageSize">
|
||||
/// The number of rows to load per page
|
||||
/// </param>
|
||||
/// <param name="sql"></param>
|
||||
/// <returns></returns>
|
||||
/// <remarks>
|
||||
/// NPoco's normal Page returns a List{T} but sometimes we don't want all that in memory and instead want to
|
||||
/// iterate over each row with a reader using Query vs Fetch.
|
||||
/// </remarks>
|
||||
public static IEnumerable<T> QueryPaged<T>(this IDatabase database, long pageSize, Sql sql) => database.QueryPaged<T>(pageSize, sql, null);
|
||||
// TODO: NPoco has a Save method that works with the primary key
|
||||
// in any case, no point trying to update if there's no primary key!
|
||||
|
||||
// NOTE
|
||||
//
|
||||
// proper way to do it with TSQL and SQLCE
|
||||
// IF EXISTS (SELECT ... FROM table WITH (UPDLOCK,HOLDLOCK)) WHERE ...)
|
||||
// BEGIN
|
||||
// UPDATE table SET ... WHERE ...
|
||||
// END
|
||||
// ELSE
|
||||
// BEGIN
|
||||
// INSERT INTO table (...) VALUES (...)
|
||||
// END
|
||||
//
|
||||
// works in READ COMMITED, TSQL & SQLCE lock the constraint even if it does not exist, so INSERT is OK
|
||||
//
|
||||
// TODO: use the proper database syntax, not this kludge
|
||||
|
||||
/// <summary>
|
||||
/// Safely inserts a record, or updates if it exists, based on a unique constraint.
|
||||
/// </summary>
|
||||
/// <param name="db"></param>
|
||||
/// <param name="poco"></param>
|
||||
/// <returns>The action that executed, either an insert or an update. If an insert occurred and a PK value got generated, the poco object
|
||||
/// passed in will contain the updated value.</returns>
|
||||
/// <remarks>
|
||||
/// <para>We cannot rely on database-specific options because SQLCE
|
||||
/// does not support any of them. Ideally this should be achieved with proper transaction isolation levels but that would mean revisiting
|
||||
/// isolation levels globally. We want to keep it simple for the time being and manage it manually.</para>
|
||||
/// <para>We handle it by trying to update, then insert, etc. until something works, or we get bored.</para>
|
||||
/// <para>Note that with proper transactions, if T2 begins after T1 then we are sure that the database will contain T2's value
|
||||
/// once T1 and T2 have completed. Whereas here, it could contain T1's value.</para>
|
||||
/// </remarks>
|
||||
public static RecordPersistenceType InsertOrUpdate<T>(this IUmbracoDatabase db, T poco)
|
||||
where T : class
|
||||
// try to update
|
||||
var rowCount = updateCommand.IsNullOrWhiteSpace() || updateArgs is null
|
||||
? db.Update(poco)
|
||||
: db.Update<T>(updateCommand!, updateArgs);
|
||||
if (rowCount > 0)
|
||||
{
|
||||
return db.InsertOrUpdate(poco, null, null);
|
||||
return RecordPersistenceType.Update;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Safely inserts a record, or updates if it exists, based on a unique constraint.
|
||||
/// </summary>
|
||||
/// <param name="db"></param>
|
||||
/// <param name="poco"></param>
|
||||
/// <param name="updateArgs"></param>
|
||||
/// <param name="updateCommand">If the entity has a composite key they you need to specify the update command explicitly</param>
|
||||
/// <returns>The action that executed, either an insert or an update. If an insert occurred and a PK value got generated, the poco object
|
||||
/// passed in will contain the updated value.</returns>
|
||||
/// <remarks>
|
||||
/// <para>We cannot rely on database-specific options because SQLCE
|
||||
/// does not support any of them. Ideally this should be achieved with proper transaction isolation levels but that would mean revisiting
|
||||
/// isolation levels globally. We want to keep it simple for the time being and manage it manually.</para>
|
||||
/// <para>We handle it by trying to update, then insert, etc. until something works, or we get bored.</para>
|
||||
/// <para>Note that with proper transactions, if T2 begins after T1 then we are sure that the database will contain T2's value
|
||||
/// once T1 and T2 have completed. Whereas here, it could contain T1's value.</para>
|
||||
/// </remarks>
|
||||
public static RecordPersistenceType InsertOrUpdate<T>(this IUmbracoDatabase db,
|
||||
T poco,
|
||||
string? updateCommand,
|
||||
object? updateArgs)
|
||||
where T : class
|
||||
// failed: does not exist, need to insert
|
||||
// RC1 race cond here: another thread may insert a record with the same constraint
|
||||
var i = 0;
|
||||
while (i++ < 4)
|
||||
{
|
||||
if (poco == null)
|
||||
throw new ArgumentNullException(nameof(poco));
|
||||
try
|
||||
{
|
||||
// try to insert
|
||||
db.Insert(poco);
|
||||
return RecordPersistenceType.Insert;
|
||||
}
|
||||
catch (SqlException)
|
||||
{
|
||||
// assuming all db engines will throw SQLException exception
|
||||
// failed: exists (due to race cond RC1)
|
||||
// RC2 race cond here: another thread may remove the record
|
||||
|
||||
// TODO: NPoco has a Save method that works with the primary key
|
||||
// in any case, no point trying to update if there's no primary key!
|
||||
|
||||
// try to update
|
||||
var rowCount = updateCommand.IsNullOrWhiteSpace() || updateArgs is null
|
||||
// try to update
|
||||
rowCount = updateCommand.IsNullOrWhiteSpace() || updateArgs is null
|
||||
? db.Update(poco)
|
||||
: db.Update<T>(updateCommand!, updateArgs);
|
||||
if (rowCount > 0)
|
||||
return RecordPersistenceType.Update;
|
||||
|
||||
// failed: does not exist, need to insert
|
||||
// RC1 race cond here: another thread may insert a record with the same constraint
|
||||
|
||||
var i = 0;
|
||||
while (i++ < 4)
|
||||
{
|
||||
try
|
||||
if (rowCount > 0)
|
||||
{
|
||||
// try to insert
|
||||
db.Insert(poco);
|
||||
return RecordPersistenceType.Insert;
|
||||
return RecordPersistenceType.Update;
|
||||
}
|
||||
catch (SqlException) // assuming all db engines will throw that exception
|
||||
{
|
||||
// failed: exists (due to race cond RC1)
|
||||
// RC2 race cond here: another thread may remove the record
|
||||
|
||||
// try to update
|
||||
rowCount = updateCommand.IsNullOrWhiteSpace() || updateArgs is null
|
||||
? db.Update(poco)
|
||||
: db.Update<T>(updateCommand!, updateArgs);
|
||||
if (rowCount > 0)
|
||||
return RecordPersistenceType.Update;
|
||||
|
||||
// failed: does not exist (due to race cond RC2), need to insert
|
||||
// loop
|
||||
}
|
||||
}
|
||||
|
||||
// this can go on forever... have to break at some point and report an error.
|
||||
throw new DataException("Record could not be inserted or updated.");
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// This will escape single @ symbols for npoco values so it doesn't think it's a parameter
|
||||
/// </summary>
|
||||
/// <param name="value"></param>
|
||||
/// <returns></returns>
|
||||
public static string EscapeAtSymbols(string value)
|
||||
{
|
||||
if (value.Contains("@") == false) return value;
|
||||
|
||||
//this fancy regex will only match a single @ not a double, etc...
|
||||
var regex = new Regex("(?<!@)@(?!@)");
|
||||
return regex.Replace(value, "@@");
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns the underlying connection as a typed connection - this is used to unwrap the profiled mini profiler stuff
|
||||
/// </summary>
|
||||
/// <typeparam name="TConnection"></typeparam>
|
||||
/// <param name="connection"></param>
|
||||
/// <returns></returns>
|
||||
public static TConnection GetTypedConnection<TConnection>(IDbConnection connection)
|
||||
where TConnection : class, IDbConnection
|
||||
{
|
||||
var c = connection;
|
||||
for (;;)
|
||||
{
|
||||
switch (c)
|
||||
{
|
||||
case TConnection ofType:
|
||||
return ofType;
|
||||
case RetryDbConnection retry:
|
||||
c = retry.Inner;
|
||||
break;
|
||||
case ProfiledDbConnection profiled:
|
||||
c = profiled.WrappedConnection;
|
||||
break;
|
||||
default:
|
||||
throw new NotSupportedException(connection.GetType().FullName);
|
||||
}
|
||||
// failed: does not exist (due to race cond RC2), need to insert
|
||||
// loop
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns the underlying transaction as a typed transaction - this is used to unwrap the profiled mini profiler stuff
|
||||
/// </summary>
|
||||
/// <typeparam name="TTransaction"></typeparam>
|
||||
/// <param name="transaction"></param>
|
||||
/// <returns></returns>
|
||||
public static TTransaction GetTypedTransaction<TTransaction>(IDbTransaction? transaction)
|
||||
where TTransaction : class, IDbTransaction
|
||||
// this can go on forever... have to break at some point and report an error.
|
||||
throw new DataException("Record could not be inserted or updated.");
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// This will escape single @ symbols for npoco values so it doesn't think it's a parameter
|
||||
/// </summary>
|
||||
/// <param name="value"></param>
|
||||
/// <returns></returns>
|
||||
public static string EscapeAtSymbols(string value)
|
||||
{
|
||||
if (value.Contains("@") == false)
|
||||
{
|
||||
var t = transaction;
|
||||
for (;;)
|
||||
return value;
|
||||
}
|
||||
|
||||
// this fancy regex will only match a single @ not a double, etc...
|
||||
var regex = new Regex("(?<!@)@(?!@)");
|
||||
return regex.Replace(value, "@@");
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns the underlying connection as a typed connection - this is used to unwrap the profiled mini profiler stuff
|
||||
/// </summary>
|
||||
/// <typeparam name="TConnection"></typeparam>
|
||||
/// <param name="connection"></param>
|
||||
/// <returns></returns>
|
||||
public static TConnection GetTypedConnection<TConnection>(IDbConnection connection)
|
||||
where TConnection : class, IDbConnection
|
||||
{
|
||||
IDbConnection? c = connection;
|
||||
for (; ;)
|
||||
{
|
||||
switch (c)
|
||||
{
|
||||
switch (t)
|
||||
{
|
||||
case TTransaction ofType:
|
||||
return ofType;
|
||||
case ProfiledDbTransaction profiled:
|
||||
t = profiled.WrappedTransaction;
|
||||
break;
|
||||
default:
|
||||
throw new NotSupportedException(transaction?.GetType().FullName);
|
||||
}
|
||||
case TConnection ofType:
|
||||
return ofType;
|
||||
case RetryDbConnection retry:
|
||||
c = retry.Inner;
|
||||
break;
|
||||
case ProfiledDbConnection profiled:
|
||||
c = profiled.WrappedConnection;
|
||||
break;
|
||||
default:
|
||||
throw new NotSupportedException(connection.GetType().FullName);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns the underlying command as a typed command - this is used to unwrap the profiled mini profiler stuff
|
||||
/// </summary>
|
||||
/// <typeparam name="TCommand"></typeparam>
|
||||
/// <param name="command"></param>
|
||||
/// <returns></returns>
|
||||
public static TCommand GetTypedCommand<TCommand>(IDbCommand command)
|
||||
where TCommand : class, IDbCommand
|
||||
{
|
||||
var c = command;
|
||||
for (;;)
|
||||
{
|
||||
switch (c)
|
||||
{
|
||||
case TCommand ofType:
|
||||
return ofType;
|
||||
case FaultHandlingDbCommand faultHandling:
|
||||
c = faultHandling.Inner;
|
||||
break;
|
||||
case ProfiledDbCommand profiled:
|
||||
c = profiled.InternalCommand;
|
||||
break;
|
||||
default:
|
||||
throw new NotSupportedException(command.GetType().FullName);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static void TruncateTable(this IDatabase db, ISqlSyntaxProvider sqlSyntax, string tableName)
|
||||
{
|
||||
var sql = new Sql(string.Format(
|
||||
sqlSyntax.TruncateTable,
|
||||
sqlSyntax.GetQuotedTableName(tableName)));
|
||||
db.Execute(sql);
|
||||
}
|
||||
|
||||
public static IsolationLevel GetCurrentTransactionIsolationLevel(this IDatabase database)
|
||||
{
|
||||
var transaction = database.Transaction;
|
||||
return transaction?.IsolationLevel ?? IsolationLevel.Unspecified;
|
||||
}
|
||||
|
||||
public static IEnumerable<TResult> FetchByGroups<TResult, TSource>(this IDatabase db, IEnumerable<TSource> source, int groupSize, Func<IEnumerable<TSource>, Sql<ISqlContext>> sqlFactory)
|
||||
{
|
||||
return source.SelectByGroups(x => db.Fetch<TResult>(sqlFactory(x)), groupSize);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns the underlying transaction as a typed transaction - this is used to unwrap the profiled mini profiler stuff
|
||||
/// </summary>
|
||||
/// <typeparam name="TTransaction"></typeparam>
|
||||
/// <param name="transaction"></param>
|
||||
/// <returns></returns>
|
||||
public static TTransaction GetTypedTransaction<TTransaction>(IDbTransaction? transaction)
|
||||
where TTransaction : class, IDbTransaction
|
||||
{
|
||||
IDbTransaction? t = transaction;
|
||||
for (; ;)
|
||||
{
|
||||
switch (t)
|
||||
{
|
||||
case TTransaction ofType:
|
||||
return ofType;
|
||||
case ProfiledDbTransaction profiled:
|
||||
t = profiled.WrappedTransaction;
|
||||
break;
|
||||
default:
|
||||
throw new NotSupportedException(transaction?.GetType().FullName);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns the underlying command as a typed command - this is used to unwrap the profiled mini profiler stuff
|
||||
/// </summary>
|
||||
/// <typeparam name="TCommand"></typeparam>
|
||||
/// <param name="command"></param>
|
||||
/// <returns></returns>
|
||||
public static TCommand GetTypedCommand<TCommand>(IDbCommand command)
|
||||
where TCommand : class, IDbCommand
|
||||
{
|
||||
IDbCommand? c = command;
|
||||
for (; ;)
|
||||
{
|
||||
switch (c)
|
||||
{
|
||||
case TCommand ofType:
|
||||
return ofType;
|
||||
case FaultHandlingDbCommand faultHandling:
|
||||
c = faultHandling.Inner;
|
||||
break;
|
||||
case ProfiledDbCommand profiled:
|
||||
c = profiled.InternalCommand;
|
||||
break;
|
||||
default:
|
||||
throw new NotSupportedException(command.GetType().FullName);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static void TruncateTable(this IDatabase db, ISqlSyntaxProvider sqlSyntax, string tableName)
|
||||
{
|
||||
var sql = new Sql(string.Format(
|
||||
sqlSyntax.TruncateTable,
|
||||
sqlSyntax.GetQuotedTableName(tableName)));
|
||||
db.Execute(sql);
|
||||
}
|
||||
|
||||
public static IsolationLevel GetCurrentTransactionIsolationLevel(this IDatabase database)
|
||||
{
|
||||
DbTransaction? transaction = database.Transaction;
|
||||
return transaction?.IsolationLevel ?? IsolationLevel.Unspecified;
|
||||
}
|
||||
|
||||
public static IEnumerable<TResult> FetchByGroups<TResult, TSource>(this IDatabase db, IEnumerable<TSource> source, int groupSize, Func<IEnumerable<TSource>, Sql<ISqlContext>> sqlFactory) =>
|
||||
source.SelectByGroups(x => db.Fetch<TResult>(sqlFactory(x)), groupSize);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user