From affd44045f41a2dd58c5667ce3938e42568fbe84 Mon Sep 17 00:00:00 2001 From: Simon S <54024766+smnsht@users.noreply.github.com> Date: Mon, 28 Aug 2023 21:12:43 +0300 Subject: [PATCH] 31355: SQLite: Add EF.Functions.Unhex (#31454) Fixes #31355 Co-authored-by: Brice Lambson --- .../Extensions/SqliteDbFunctionsExtensions.cs | 27 +++++++++++++ .../Internal/SqliteHexMethodTranslator.cs | 21 +++++++++- .../BuiltInDataTypesSqliteTest.cs | 40 +++++++++++++++++++ 3 files changed, 86 insertions(+), 2 deletions(-) diff --git a/src/EFCore.Sqlite.Core/Extensions/SqliteDbFunctionsExtensions.cs b/src/EFCore.Sqlite.Core/Extensions/SqliteDbFunctionsExtensions.cs index 585a0936030..f52da2ec8f4 100644 --- a/src/EFCore.Sqlite.Core/Extensions/SqliteDbFunctionsExtensions.cs +++ b/src/EFCore.Sqlite.Core/Extensions/SqliteDbFunctionsExtensions.cs @@ -43,6 +43,33 @@ public static bool Glob(this DbFunctions _, string matchExpression, string patte public static string Hex(this DbFunctions _, byte[] bytes) => throw new InvalidOperationException(CoreStrings.FunctionOnClient(nameof(Hex))); + /// + /// Maps to the SQLite unhex function which returns a BLOB representing decoding of the hexadecimal string. + /// + /// + /// See Database functions, and + /// Accessing SQLite databases with EF Core for more information and examples. + /// + /// The instance. + /// The hexadecimal string. + /// Decoded hexadecimal string as binary value. + public static byte[] Unhex(this DbFunctions _, string value) + => throw new InvalidOperationException(CoreStrings.FunctionOnClient(nameof(Unhex))); + + /// + /// Maps to the SQLite unhex function which returns a BLOB representing decoding of the hexadecimal string. + /// + /// + /// See Database functions, and + /// Accessing SQLite databases with EF Core for more information and examples. + /// + /// The instance. + /// The hexadecimal string. + /// Characters that are ignored in . + /// Decoded hexadecimal string as binary value. + public static byte[] Unhex(this DbFunctions _, string value, string ignoreChars) + => throw new InvalidOperationException(CoreStrings.FunctionOnClient(nameof(Unhex))); + /// /// Maps to the SQLite substr function which returns a subarray of the specified value. The subarray starts /// at and continues to the end of the value. diff --git a/src/EFCore.Sqlite.Core/Query/Internal/SqliteHexMethodTranslator.cs b/src/EFCore.Sqlite.Core/Query/Internal/SqliteHexMethodTranslator.cs index 68e759f7a85..ede730114ed 100644 --- a/src/EFCore.Sqlite.Core/Query/Internal/SqliteHexMethodTranslator.cs +++ b/src/EFCore.Sqlite.Core/Query/Internal/SqliteHexMethodTranslator.cs @@ -13,9 +13,15 @@ namespace Microsoft.EntityFrameworkCore.Sqlite.Query.Internal; /// public class SqliteHexMethodTranslator : IMethodCallTranslator { - private static readonly MethodInfo MethodInfo = typeof(SqliteDbFunctionsExtensions) + private static readonly MethodInfo HexMethodInfo = typeof(SqliteDbFunctionsExtensions) .GetMethod(nameof(SqliteDbFunctionsExtensions.Hex), new[] { typeof(DbFunctions), typeof(byte[]) })!; + private static readonly MethodInfo UnhexMethodInfo = typeof(SqliteDbFunctionsExtensions) + .GetMethod(nameof(SqliteDbFunctionsExtensions.Unhex), new[] { typeof(DbFunctions), typeof(string) })!; + + private static readonly MethodInfo UnhexWithIgnoreCharsMethodInfo = typeof(SqliteDbFunctionsExtensions) + .GetMethod(nameof(SqliteDbFunctionsExtensions.Unhex), new[] { typeof(DbFunctions), typeof(string), typeof(string) })!; + private readonly ISqlExpressionFactory _sqlExpressionFactory; /// @@ -41,7 +47,7 @@ public SqliteHexMethodTranslator(ISqlExpressionFactory sqlExpressionFactory) IReadOnlyList arguments, IDiagnosticsLogger logger) { - if (method.Equals(MethodInfo)) + if (method.Equals(HexMethodInfo)) { return _sqlExpressionFactory.Function( "hex", @@ -51,6 +57,17 @@ public SqliteHexMethodTranslator(ISqlExpressionFactory sqlExpressionFactory) typeof(string)); } + if (method.Equals(UnhexMethodInfo) + || method.Equals(UnhexWithIgnoreCharsMethodInfo)) + { + return _sqlExpressionFactory.Function( + "unhex", + arguments.Skip(1), + nullable: true, + arguments.Skip(1).Select(_ => true).ToArray(), + typeof(byte[])); + } + return null; } } diff --git a/test/EFCore.Sqlite.FunctionalTests/BuiltInDataTypesSqliteTest.cs b/test/EFCore.Sqlite.FunctionalTests/BuiltInDataTypesSqliteTest.cs index 08fc4c6e23a..c1a73c88ac5 100644 --- a/test/EFCore.Sqlite.FunctionalTests/BuiltInDataTypesSqliteTest.cs +++ b/test/EFCore.Sqlite.FunctionalTests/BuiltInDataTypesSqliteTest.cs @@ -1546,6 +1546,46 @@ SELECT hex("o"."Bytes") Assert.Equal(expectedResults, results); } + [ConditionalFact] + public virtual void Can_query_using_unhex_function() + { + using var context = CreateContext(); + + var results = context.Set() + .Select(e => EF.Functions.Unhex(EF.Functions.Hex(e.Bytes))).ToList(); + + AssertSql( +""" +SELECT unhex(hex("o"."Bytes")) +FROM "ObjectBackedDataTypes" AS "o" +"""); + + var expectedResults = context.Set().AsEnumerable() + .Select(e => e.Bytes).ToList(); + + Assert.Equal(expectedResults, results); + } + + [ConditionalFact] + public virtual void Can_query_using_unhex_function_with_ignore_chars() + { + using var context = CreateContext(); + + var results = context.Set() + .Select(e => EF.Functions.Unhex(EF.Functions.Hex(e.Bytes) + "!?", "!?")).ToList(); + + AssertSql( +""" +SELECT unhex(COALESCE(hex("o"."Bytes"), '') || '!?', '!?') +FROM "ObjectBackedDataTypes" AS "o" +"""); + + var expectedResults = context.Set().AsEnumerable() + .Select(e => e.Bytes).ToList(); + + Assert.Equal(expectedResults, results); + } + [ConditionalFact] public virtual void Can_query_using_substr_function() {