2024-07-04 11:02:45 +05:00
|
|
|
using Microsoft.EntityFrameworkCore;
|
2022-07-04 17:24:59 +05:00
|
|
|
using Microsoft.EntityFrameworkCore.ChangeTracking;
|
2021-11-15 14:56:11 +05:00
|
|
|
using Microsoft.EntityFrameworkCore.Metadata;
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Text;
|
2023-03-30 12:57:32 +05:00
|
|
|
using System.Text.Json.Serialization;
|
|
|
|
using System.Text.Json;
|
2021-11-15 14:56:11 +05:00
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
namespace AsbCloudDb
|
|
|
|
{
|
2023-12-05 14:48:56 +05:00
|
|
|
public static class EFExtensions
|
2021-11-15 14:56:11 +05:00
|
|
|
{
|
2023-12-05 14:48:56 +05:00
|
|
|
private static readonly JsonSerializerOptions jsonSerializerOptions = new()
|
2022-06-20 12:42:35 +05:00
|
|
|
{
|
|
|
|
AllowTrailingCommas = true,
|
|
|
|
WriteIndented = true,
|
2023-12-05 14:48:56 +05:00
|
|
|
NumberHandling = JsonNumberHandling.AllowReadingFromString | JsonNumberHandling.AllowNamedFloatingPointLiterals,
|
2022-06-20 12:42:35 +05:00
|
|
|
};
|
|
|
|
|
|
|
|
public static Microsoft.EntityFrameworkCore.Metadata.Builders.PropertyBuilder<TProperty> HasJsonConversion<TProperty>(
|
|
|
|
this Microsoft.EntityFrameworkCore.Metadata.Builders.PropertyBuilder<TProperty> builder)
|
|
|
|
=> HasJsonConversion(builder, jsonSerializerOptions);
|
|
|
|
|
|
|
|
public static Microsoft.EntityFrameworkCore.Metadata.Builders.PropertyBuilder<TProperty> HasJsonConversion<TProperty>(
|
|
|
|
this Microsoft.EntityFrameworkCore.Metadata.Builders.PropertyBuilder<TProperty> builder,
|
2023-12-05 14:48:56 +05:00
|
|
|
JsonSerializerOptions jsonSerializerOptions)
|
2022-06-20 12:42:35 +05:00
|
|
|
{
|
|
|
|
builder.HasConversion(
|
2023-12-05 14:48:56 +05:00
|
|
|
s => JsonSerializer.Serialize(s, jsonSerializerOptions),
|
|
|
|
s => JsonSerializer.Deserialize<TProperty>(s, jsonSerializerOptions)!);
|
2022-07-04 17:24:59 +05:00
|
|
|
|
|
|
|
ValueComparer<TProperty> valueComparer = new (
|
2022-07-04 17:33:32 +05:00
|
|
|
(a,b) =>
|
|
|
|
(a!=null) && (b != null)
|
|
|
|
? a.GetHashCode() == b.GetHashCode()
|
|
|
|
: (a == null) && (b == null),
|
|
|
|
i => (i == null) ?-1 : i.GetHashCode(),
|
2022-07-04 17:24:59 +05:00
|
|
|
i => i);
|
|
|
|
|
|
|
|
builder.Metadata.SetValueComparer(valueComparer);
|
2022-06-20 12:42:35 +05:00
|
|
|
return builder;
|
|
|
|
}
|
|
|
|
|
2024-07-08 11:09:43 +05:00
|
|
|
static Dictionary<Type, IQueryStringFactory> QueryFactories { get; set; } = [];
|
2022-05-06 10:58:52 +05:00
|
|
|
|
2021-12-08 15:29:41 +05:00
|
|
|
static QueryStringFactory<T> GetQueryStringFactory<T>(DbSet<T> dbSet)
|
2021-11-15 14:56:11 +05:00
|
|
|
where T : class
|
|
|
|
{
|
|
|
|
var t = typeof(T);
|
2022-05-06 10:58:52 +05:00
|
|
|
var factory = (QueryStringFactory<T>?)QueryFactories.GetValueOrDefault(t);
|
2021-11-15 14:56:11 +05:00
|
|
|
if (factory is null)
|
|
|
|
{
|
2021-12-08 15:29:41 +05:00
|
|
|
factory = new QueryStringFactory<T>(dbSet);
|
2021-11-15 14:56:11 +05:00
|
|
|
QueryFactories.Add(t, factory);
|
|
|
|
}
|
2022-04-11 18:00:34 +05:00
|
|
|
|
2021-11-15 14:56:11 +05:00
|
|
|
return factory;
|
|
|
|
}
|
|
|
|
|
2021-12-08 15:29:41 +05:00
|
|
|
public static Task<int> ExecInsertOrUpdateAsync<T>(this Microsoft.EntityFrameworkCore.Infrastructure.DatabaseFacade database, DbSet<T> dbSet, IEnumerable<T> items, CancellationToken token)
|
2021-11-15 14:56:11 +05:00
|
|
|
where T : class
|
|
|
|
{
|
2021-12-08 15:29:41 +05:00
|
|
|
var factory = GetQueryStringFactory(dbSet);
|
2021-11-15 14:56:11 +05:00
|
|
|
var query = factory.MakeInsertOrUpdateSql(items);
|
2022-07-19 10:29:38 +05:00
|
|
|
|
2021-11-15 14:56:11 +05:00
|
|
|
return database.ExecuteSqlRawAsync(query, token);
|
|
|
|
}
|
2021-12-17 12:48:58 +05:00
|
|
|
|
2022-10-25 17:23:01 +05:00
|
|
|
public static Task<int> ExecInsertOrIgnoreAsync<T>(this Microsoft.EntityFrameworkCore.Infrastructure.DatabaseFacade database, DbSet<T> dbSet, IEnumerable<T> items, CancellationToken token)
|
|
|
|
where T : class
|
|
|
|
{
|
|
|
|
var factory = GetQueryStringFactory(dbSet);
|
|
|
|
var query = factory.MakeInsertOrIgnoreSql(items);
|
|
|
|
return database.ExecuteSqlRawAsync(query, token);
|
|
|
|
}
|
|
|
|
|
2022-10-10 12:39:54 +05:00
|
|
|
public static Task<int> ExecInsertAsync<T>(this Microsoft.EntityFrameworkCore.Infrastructure.DatabaseFacade database, DbSet<T> dbSet, IEnumerable<T> items, CancellationToken token)
|
|
|
|
where T : class
|
|
|
|
{
|
|
|
|
var factory = GetQueryStringFactory(dbSet);
|
|
|
|
var query = factory.MakeInsertSql(items);
|
|
|
|
|
|
|
|
return database.ExecuteSqlRawAsync(query, token);
|
|
|
|
}
|
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
public static string GetTableName<T>(this DbSet<T> dbSet)
|
|
|
|
where T : class
|
|
|
|
{
|
|
|
|
var factory = GetQueryStringFactory(dbSet);
|
|
|
|
return factory.TableName;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static IEnumerable<string> GetColumnsNames<T>(this DbSet<T> dbSet)
|
|
|
|
where T : class
|
|
|
|
{
|
|
|
|
var factory = GetQueryStringFactory(dbSet);
|
|
|
|
return factory.Columns;
|
|
|
|
}
|
2022-06-10 17:30:42 +05:00
|
|
|
|
2023-12-05 14:48:56 +05:00
|
|
|
public static EntityEntry<T> Upsert<T>(this DbSet<T> dbSet, T value)
|
2022-06-10 17:30:42 +05:00
|
|
|
where T : class
|
|
|
|
{
|
|
|
|
return dbSet.Contains(value)
|
|
|
|
? dbSet.Update(value)
|
2022-06-15 14:57:37 +05:00
|
|
|
: dbSet.Add(value);
|
2022-06-10 17:30:42 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
public static (int updated, int inserted) UpsertRange<T>(this DbSet<T> dbSet, IEnumerable<T> values)
|
|
|
|
where T : class
|
|
|
|
{
|
2022-06-15 14:57:37 +05:00
|
|
|
(int updated, int inserted) stat = (0, 0);
|
2022-06-10 17:30:42 +05:00
|
|
|
foreach (var value in values)
|
|
|
|
if (dbSet.Contains(value))
|
|
|
|
{
|
|
|
|
stat.updated++;
|
|
|
|
dbSet.Update(value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
stat.inserted++;
|
|
|
|
dbSet.Add(value);
|
|
|
|
}
|
|
|
|
return stat;
|
|
|
|
}
|
2022-10-17 14:42:47 +05:00
|
|
|
|
|
|
|
public static IQueryable<T> SkipTake<T>(this IQueryable<T> query, int? skip, int? take)
|
|
|
|
{
|
|
|
|
if (skip > 0)
|
|
|
|
query = query.Skip((int)skip);
|
|
|
|
|
|
|
|
if (take > 0)
|
|
|
|
query = query.Take((int)take);
|
|
|
|
|
|
|
|
return query;
|
|
|
|
}
|
2021-11-15 14:56:11 +05:00
|
|
|
}
|
|
|
|
|
2022-04-11 18:00:34 +05:00
|
|
|
interface IQueryStringFactory { }
|
2021-11-15 14:56:11 +05:00
|
|
|
|
|
|
|
class QueryStringFactory<T> : IQueryStringFactory
|
|
|
|
where T : class
|
|
|
|
{
|
2021-11-17 10:52:03 +05:00
|
|
|
private readonly string insertHeader;
|
2021-11-15 14:56:11 +05:00
|
|
|
private readonly string pk;
|
2021-11-17 10:52:03 +05:00
|
|
|
private readonly string conflictBody;
|
2021-11-15 16:52:12 +05:00
|
|
|
private readonly IEnumerable<IClrPropertyGetter> getters;
|
2021-11-15 14:56:11 +05:00
|
|
|
|
2021-12-17 12:48:58 +05:00
|
|
|
public string TableName { get; }
|
|
|
|
public IEnumerable<string> Columns { get; }
|
|
|
|
|
2024-07-08 11:09:43 +05:00
|
|
|
public QueryStringFactory(DbSet<T> dbSet)
|
2021-11-15 14:56:11 +05:00
|
|
|
{
|
2024-07-08 11:09:43 +05:00
|
|
|
var properties = dbSet.EntityType.GetProperties();
|
|
|
|
var pkColsNames = dbSet.EntityType.FindPrimaryKey()?.Properties.Select(p => p.GetColumnName());
|
2021-11-15 14:56:11 +05:00
|
|
|
pk = pkColsNames is null ? string.Empty : $"({string.Join(", ", pkColsNames)})";
|
|
|
|
|
2024-07-08 11:09:43 +05:00
|
|
|
TableName = dbSet.EntityType.GetTableName()!;
|
2022-07-19 10:29:38 +05:00
|
|
|
getters = properties
|
|
|
|
.Where(p => !p.IsShadowProperty())
|
|
|
|
.Select(p => p.GetGetter()).ToList();
|
|
|
|
|
2024-07-08 11:09:43 +05:00
|
|
|
Columns = properties.Select(p => $"\"{p.GetColumnName()}\"");
|
|
|
|
var columnsString = $"({string.Join(", ", Columns)})";
|
2021-11-17 10:52:03 +05:00
|
|
|
|
2024-07-08 11:09:43 +05:00
|
|
|
insertHeader = $"INSERT INTO {TableName} {columnsString} VALUES ";
|
2021-12-17 12:48:58 +05:00
|
|
|
var excludedUpdateSet = string.Join(", ", Columns.Select(n => $"{n} = excluded.{n}"));
|
2021-11-17 10:52:03 +05:00
|
|
|
conflictBody = $" ON CONFLICT {pk} DO UPDATE SET {excludedUpdateSet};";
|
2021-11-15 14:56:11 +05:00
|
|
|
}
|
|
|
|
|
2022-10-25 17:23:01 +05:00
|
|
|
public string MakeInsertOrIgnoreSql(IEnumerable<T> items)
|
|
|
|
{
|
|
|
|
var builder = new StringBuilder(insertHeader, 7);
|
|
|
|
BuildRows(builder, items);
|
|
|
|
builder.Append(" ON CONFLICT DO NOTHING;");
|
|
|
|
return builder.ToString();
|
|
|
|
}
|
|
|
|
|
2021-11-15 14:56:11 +05:00
|
|
|
public string MakeInsertOrUpdateSql(IEnumerable<T> items)
|
|
|
|
{
|
2021-11-17 10:52:03 +05:00
|
|
|
var builder = new StringBuilder(insertHeader, 7);
|
|
|
|
BuildRows(builder, items);
|
2021-11-15 14:56:11 +05:00
|
|
|
if (string.IsNullOrEmpty(pk))
|
2021-11-17 10:52:03 +05:00
|
|
|
builder.Append(" ON CONFLICT DO NOTHING;");
|
2021-11-15 14:56:11 +05:00
|
|
|
else
|
2021-11-17 10:52:03 +05:00
|
|
|
builder.AppendLine(conflictBody);
|
2021-11-15 14:56:11 +05:00
|
|
|
|
2021-11-17 10:52:03 +05:00
|
|
|
return builder.ToString();
|
2021-11-15 14:56:11 +05:00
|
|
|
}
|
|
|
|
|
2022-10-10 12:39:54 +05:00
|
|
|
public string MakeInsertSql(IEnumerable<T> items)
|
|
|
|
{
|
|
|
|
var builder = new StringBuilder(insertHeader, 7);
|
|
|
|
BuildRows(builder, items);
|
|
|
|
builder.Append(';');
|
|
|
|
return builder.ToString();
|
|
|
|
}
|
|
|
|
|
2021-11-17 10:52:03 +05:00
|
|
|
private StringBuilder BuildRows(StringBuilder builder, IEnumerable<T> items)
|
2021-11-15 14:56:11 +05:00
|
|
|
{
|
2021-11-17 10:52:03 +05:00
|
|
|
var list = items.ToList();
|
2021-12-08 15:29:41 +05:00
|
|
|
for (var i = 0; i < list.Count; i++)
|
2021-11-17 10:52:03 +05:00
|
|
|
{
|
2022-04-11 18:00:34 +05:00
|
|
|
if (i > 0)
|
2021-11-17 10:52:03 +05:00
|
|
|
builder.Append(',');
|
|
|
|
BuildRow(builder, list[i]);
|
2022-04-11 18:00:34 +05:00
|
|
|
}
|
2021-11-17 10:52:03 +05:00
|
|
|
return builder;
|
2021-11-15 14:56:11 +05:00
|
|
|
}
|
|
|
|
|
2021-11-17 10:52:03 +05:00
|
|
|
private StringBuilder BuildRow(StringBuilder builder, T item)
|
2021-11-15 14:56:11 +05:00
|
|
|
{
|
|
|
|
var values = getters.Select(getter => FormatValue(getter.GetClrValue(item)));
|
2021-11-17 10:52:03 +05:00
|
|
|
builder.Append('(');
|
|
|
|
builder.AppendJoin(",", values);
|
|
|
|
builder.Append(')');
|
|
|
|
return builder;
|
2021-11-15 14:56:11 +05:00
|
|
|
}
|
|
|
|
|
2022-05-06 10:58:52 +05:00
|
|
|
private static string FormatValue(object? v)
|
2021-11-15 14:56:11 +05:00
|
|
|
=> v switch
|
2022-04-11 18:00:34 +05:00
|
|
|
{
|
2024-05-16 15:10:21 +05:00
|
|
|
null => "NULL",
|
2022-07-19 10:29:38 +05:00
|
|
|
string vStr => $"'{EscapeCurlyBraces(vStr)}'",
|
2022-04-11 18:00:34 +05:00
|
|
|
DateTime vDate => $"'{FormatDateValue(vDate)}'",
|
|
|
|
DateTimeOffset vDate => $"'{FormatDateValue(vDate.UtcDateTime)}'",
|
|
|
|
IFormattable vFormattable => FormatFormattableValue(vFormattable),
|
2024-05-13 16:48:47 +05:00
|
|
|
_ => $"'{EscapeCurlyBraces(JsonSerializer.Serialize(v))}'",
|
2022-04-11 18:00:34 +05:00
|
|
|
};
|
2021-11-15 14:56:11 +05:00
|
|
|
|
2022-07-19 10:29:38 +05:00
|
|
|
private static string EscapeCurlyBraces(string vStr)
|
|
|
|
{
|
|
|
|
var result = vStr
|
|
|
|
.Replace("{", "{{")
|
|
|
|
.Replace("}", "}}");
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-11-15 14:56:11 +05:00
|
|
|
private static string FormatFormattableValue(IFormattable v)
|
|
|
|
=> v switch
|
|
|
|
{
|
|
|
|
double vt => vt.ToString(System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
float vt => vt.ToString(System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
decimal vt => vt.ToString(System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
int vt => vt.ToString(System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
short vt => vt.ToString(System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
uint vt => vt.ToString(System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
ushort vt => vt.ToString(System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
_ => v.ToString(null, System.Globalization.CultureInfo.InvariantCulture),
|
|
|
|
};
|
|
|
|
|
|
|
|
private static string FormatDateValue(DateTime vDate)
|
|
|
|
{
|
|
|
|
if (vDate.Kind == DateTimeKind.Unspecified)
|
|
|
|
vDate = DateTime.SpecifyKind(vDate, DateTimeKind.Utc);
|
|
|
|
return vDate.ToUniversalTime().ToString("yyyy-MM-dd HH:mm:ss.ffffff zzz");
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2023-03-30 12:57:32 +05:00
|
|
|
public class DateOnlyJsonConverter : JsonConverter<DateOnly>
|
|
|
|
{
|
|
|
|
public override DateOnly Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
|
|
|
|
{
|
|
|
|
return DateOnly.FromDateTime(reader.GetDateTime());
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void Write(Utf8JsonWriter writer, DateOnly value, JsonSerializerOptions options)
|
|
|
|
{
|
|
|
|
var isoDate = value.ToString("O");
|
|
|
|
writer.WriteStringValue(isoDate);
|
|
|
|
}
|
|
|
|
}
|
2021-11-15 14:56:11 +05:00
|
|
|
}
|