|
1 | | -using Aikido.Zen.Core.Helpers; |
| 1 | +using System; |
| 2 | +using System.Reflection; |
2 | 3 | using HarmonyLib; |
3 | | -using Microsoft.Data.Sqlite; |
4 | | -using MySql.Data.MySqlClient; |
5 | | -using System.Data.Common; |
6 | | -using Npgsql; |
7 | 4 | using Aikido.Zen.Core.Models; |
8 | | -using MySqlX.XDevAPI.Relational; |
9 | | -using System.Reflection; |
| 5 | +using Aikido.Zen.Core.Helpers; |
10 | 6 |
|
11 | 7 | namespace Aikido.Zen.DotNetCore.Patches |
12 | 8 | { |
13 | 9 | internal static class SqlClientPatches |
14 | 10 | { |
15 | | - // we need to patch from inside the framework, because we have to pass the context, which is constructed in a framework specific manner |
16 | 11 | public static void ApplyPatches(Harmony harmony) |
17 | 12 | { |
18 | | - |
19 | | - // Generic |
20 | | - PatchMethod(harmony, typeof(DbCommand), "ExecuteNonQueryAsync"); |
21 | | - PatchMethod(harmony, typeof(DbCommand), "ExecuteReaderAsync", typeof(System.Data.CommandBehavior)); |
22 | | - PatchMethod(harmony, typeof(DbCommand), "ExecuteScalarAsync"); |
| 13 | + // Use reflection to get the types dynamically |
| 14 | + PatchMethod(harmony, "System.Data.Common", "DbCommand", "ExecuteNonQueryAsync"); |
| 15 | + PatchMethod(harmony, "System.Data.Common", "DbCommand", "ExecuteReaderAsync", "System.Data.CommandBehavior"); |
| 16 | + PatchMethod(harmony, "System.Data.Common", "DbCommand", "ExecuteScalarAsync"); |
23 | 17 |
|
24 | 18 | // SQL Server |
25 | | - PatchMethod(harmony, typeof(Microsoft.Data.SqlClient.SqlCommand), "ExecuteNonQuery"); |
26 | | - PatchMethod(harmony, typeof(Microsoft.Data.SqlClient.SqlCommand), "ExecuteScalar"); |
27 | | - PatchMethod(harmony, typeof(Microsoft.Data.SqlClient.SqlCommand), "ExecuteReader", typeof(System.Data.CommandBehavior)); |
28 | | - PatchMethod(harmony, typeof(System.Data.SqlClient.SqlCommand), "ExecuteNonQuery"); |
29 | | - PatchMethod(harmony, typeof(System.Data.SqlClient.SqlCommand), "ExecuteScalar"); |
30 | | - PatchMethod(harmony, typeof(System.Data.SqlClient.SqlCommand), "ExecuteReader", typeof(System.Data.CommandBehavior)); |
| 19 | + PatchMethod(harmony, "Microsoft.Data.SqlClient", "SqlCommand", "ExecuteNonQuery"); |
| 20 | + PatchMethod(harmony, "Microsoft.Data.SqlClient", "SqlCommand", "ExecuteScalar"); |
| 21 | + PatchMethod(harmony, "Microsoft.Data.SqlClient", "SqlCommand", "ExecuteReader", "System.Data.CommandBehavior"); |
| 22 | + PatchMethod(harmony, "System.Data.SqlClient", "SqlCommand", "ExecuteNonQuery"); |
| 23 | + PatchMethod(harmony, "System.Data.SqlClient", "SqlCommand", "ExecuteScalar"); |
| 24 | + PatchMethod(harmony, "System.Data.SqlClient", "SqlCommand", "ExecuteReader", "System.Data.CommandBehavior"); |
31 | 25 |
|
32 | 26 | // SQLite |
33 | | - PatchMethod(harmony, typeof(SqliteCommand), "ExecuteNonQuery"); |
34 | | - PatchMethod(harmony, typeof(SqliteCommand), "ExecuteScalar"); |
35 | | - PatchMethod(harmony, typeof(SqliteCommand), "ExecuteReader", typeof(System.Data.CommandBehavior)); |
| 27 | + PatchMethod(harmony, "Microsoft.Data.Sqlite", "SqliteCommand", "ExecuteNonQuery"); |
| 28 | + PatchMethod(harmony, "Microsoft.Data.Sqlite", "SqliteCommand", "ExecuteScalar"); |
| 29 | + PatchMethod(harmony, "Microsoft.Data.Sqlite", "SqliteCommand", "ExecuteReader", "System.Data.CommandBehavior"); |
36 | 30 |
|
37 | 31 | // MySql, MariaDB |
38 | | - PatchMethod(harmony, typeof(MySqlCommand), "ExecuteNonQuery"); |
39 | | - PatchMethod(harmony, typeof(MySqlCommand), "ExecuteScalar"); |
40 | | - PatchMethod(harmony, typeof(MySqlCommand), "ExecuteReader", typeof(System.Data.CommandBehavior)); |
41 | | - PatchMethod(harmony, typeof(MySqlConnector.MySqlCommand), "ExecuteNonQuery"); |
42 | | - PatchMethod(harmony, typeof(MySqlConnector.MySqlCommand), "ExecuteScalar"); |
43 | | - PatchMethod(harmony, typeof(MySqlConnector.MySqlCommand), "ExecuteReader", typeof(System.Data.CommandBehavior)); |
| 32 | + PatchMethod(harmony, "MySql.Data", "MySqlClient.MySqlCommand", "ExecuteNonQuery"); |
| 33 | + PatchMethod(harmony, "MySql.Data", "MySqlClient.MySqlCommand", "ExecuteScalar"); |
| 34 | + PatchMethod(harmony, "MySql.Data", "MySqlClient.MySqlCommand", "ExecuteReader", "System.Data.CommandBehavior"); |
| 35 | + PatchMethod(harmony, "MySqlConnector", "MySqlCommand", "ExecuteNonQuery"); |
| 36 | + PatchMethod(harmony, "MySqlConnector", "MySqlCommand", "ExecuteScalar"); |
| 37 | + PatchMethod(harmony, "MySqlConnector", "MySqlCommand", "ExecuteReader", "System.Data.CommandBehavior"); |
44 | 38 |
|
45 | 39 | // PostgreSQL |
46 | | - PatchMethod(harmony, typeof(NpgsqlCommand), "ExecuteNonQuery"); |
47 | | - PatchMethod(harmony, typeof(NpgsqlCommand), "ExecuteScalar"); |
48 | | - PatchMethod(harmony, typeof(NpgsqlCommand), "ExecuteReader", typeof(System.Data.CommandBehavior)); |
| 40 | + PatchMethod(harmony, "Npgsql", "NpgsqlCommand", "ExecuteNonQuery"); |
| 41 | + PatchMethod(harmony, "Npgsql", "NpgsqlCommand", "ExecuteScalar"); |
| 42 | + PatchMethod(harmony, "Npgsql", "NpgsqlCommand", "ExecuteReader", "System.Data.CommandBehavior"); |
49 | 43 |
|
50 | 44 | // MySqlX |
51 | | - PatchMethod(harmony, typeof(Table), "Select"); |
52 | | - PatchMethod(harmony, typeof(Table), "Insert"); |
53 | | - PatchMethod(harmony, typeof(Table), "Update"); |
54 | | - PatchMethod(harmony, typeof(Table), "Delete"); |
| 45 | + PatchMethod(harmony, "MySqlX", "XDevAPI.Relational.Table", "Select"); |
| 46 | + PatchMethod(harmony, "MySqlX", "XDevAPI.Relational.Table", "Insert"); |
| 47 | + PatchMethod(harmony, "MySqlX", "XDevAPI.Relational.Table", "Update"); |
| 48 | + PatchMethod(harmony, "MySqlX", "XDevAPI.Relational.Table", "Delete"); |
55 | 49 | } |
56 | 50 |
|
57 | | - private static void PatchMethod(Harmony harmony, Type type, string methodName, params Type[] parameters) |
| 51 | + private static void PatchMethod(Harmony harmony, string assemblyName, string typeName, string methodName, params string[] parameterTypeNames) |
58 | 52 | { |
59 | | - var method = AccessTools.Method(type, methodName, parameters); |
| 53 | + var method = ReflectionHelper.GetMethodFromAssembly(assemblyName, typeName, methodName, parameterTypeNames); |
60 | 54 | if (method != null) |
61 | 55 | { |
62 | 56 | harmony.Patch(method, new HarmonyMethod(typeof(SqlClientPatches).GetMethod(nameof(OnCommandExecuting), BindingFlags.Static | BindingFlags.NonPublic))); |
63 | 57 | } |
64 | 58 | } |
65 | 59 |
|
66 | | - private static bool OnCommandExecuting(object[] __args, MethodBase __originalMethod, DbCommand __instance) |
| 60 | + private static bool OnCommandExecuting(object[] __args, MethodBase __originalMethod, object __instance) |
67 | 61 | { |
| 62 | + var dbCommand = __instance as System.Data.Common.DbCommand; |
| 63 | + if (dbCommand == null) return true; |
68 | 64 | var assembly = __instance.GetType().Assembly.FullName?.Split(", Culture=")[0]; |
69 | | - return Aikido.Zen.Core.Patches.SqlClientPatcher.OnCommandExecuting(__args, __originalMethod, __instance, assembly, Zen.GetContext()); |
| 65 | + return Aikido.Zen.Core.Patches.SqlClientPatcher.OnCommandExecuting(__args, __originalMethod, dbCommand, assembly, Zen.GetContext()); |
70 | 66 | } |
71 | 67 | } |
72 | 68 | } |
0 commit comments