|
1 | 1 | using System;
|
| 2 | +using System.Collections.Concurrent; |
2 | 3 | using System.Collections.Generic;
|
3 | 4 | using System.Collections.ObjectModel;
|
4 | 5 | using System.Data;
|
5 | 6 | using System.Data.Common;
|
6 | 7 | using System.Reflection;
|
7 | 8 | using System.Text;
|
| 9 | +using System.Threading; |
8 | 10 | using MySqlConnector;
|
9 | 11 | using Pchp.Core;
|
10 | 12 | using Pchp.Library.Database;
|
@@ -91,42 +93,86 @@ public static ConnectionResource ValidConnection(Context ctx, PhpResource link =
|
91 | 93 | }
|
92 | 94 |
|
93 | 95 | /// <summary>
|
94 |
| - /// Returns the last insert ID from an IDbCommand by unwrapping it to the internal MySqlCommand |
| 96 | + /// Casts the <paramref name="object"/> to <typeparamref name="TResult"/>. |
| 97 | + /// Otherwise it reflects the <paramref name="object"/>, looking for a <paramref name="getterMethodName"/> and trying to invoke that to get the underlying reference. |
95 | 98 | /// </summary>
|
96 |
| - /// <param name="command">Generic IDbCommand to work with</param> |
97 |
| - /// <returns>Last insert ID</returns> |
98 |
| - public static long LastInsertedId( |
99 |
| - IDbCommand command) |
| 99 | + /// <typeparam name="TIn">Abstract type.</typeparam> |
| 100 | + /// <typeparam name="TResult">Expected actual type of <paramref name="object"/>.</typeparam> |
| 101 | + /// <param name="object">Reference to object.</param> |
| 102 | + /// <param name="lazyCache">Lazily created dictionary remembering method used ot obtain the underlying value.</param> |
| 103 | + /// <param name="getterMethodName">Method name used to obtain the underlying value.</param> |
| 104 | + /// <returns>Value of type <typeparamref name="TResult"/>.</returns> |
| 105 | + /// <remarks>Used to get an underlying value of wrapping classes like the ones provided by MiniProfiler.</remarks> |
| 106 | + /// <exception cref="ArgumentNullException"><paramref name="object"/> is null.</exception> |
| 107 | + /// <exception cref="InvalidOperationException"><paramref name="getterMethodName"/> is not defined on <paramref name="object"/>.</exception> |
| 108 | + /// <exception cref="NullReferenceException"><paramref name="object"/>' getter method returned null or an unexpected value.</exception> |
| 109 | + static TResult/*!*/GetUnderlyingValue<TIn, TResult>(TIn @object, ref ConcurrentDictionary<Type, MethodInfo> lazyCache, string getterMethodName) where TResult : class |
100 | 110 | {
|
101 |
| - // If we have a MySqlCommand, just use it |
102 |
| - if (command is MySqlCommand mySqlCommand) return mySqlCommand.LastInsertedId; |
103 |
| - |
104 |
| - // If we did not get one back, try to unwrap it as likely it's wrapped by a profiler like MiniProfiler |
105 |
| - if (_innerCommandMethod == null) |
106 |
| - _innerCommandMethod = command.GetType().GetMethod("get_InternalCommand", BindingFlags.Instance | BindingFlags.Public); |
107 |
| - mySqlCommand = _innerCommandMethod?.Invoke(command, null) as MySqlCommand; |
108 |
| - if (mySqlCommand == null) throw new NullReferenceException("Could not get internal command for wrapped command!"); |
109 |
| - return mySqlCommand.LastInsertedId; |
| 111 | + // we have TResult in most cases: |
| 112 | + if (@object is TResult result) |
| 113 | + { |
| 114 | + return result; |
| 115 | + } |
| 116 | + |
| 117 | + if (@object == null) |
| 118 | + { |
| 119 | + throw new ArgumentNullException(nameof(@object)); |
| 120 | + } |
| 121 | + |
| 122 | + // enure cache |
| 123 | + if (lazyCache == null) |
| 124 | + { |
| 125 | + Interlocked.CompareExchange(ref lazyCache, new ConcurrentDictionary<Type, MethodInfo>(), null); |
| 126 | + } |
| 127 | + |
| 128 | + // resolve {getterMethodName} method |
| 129 | + var method = lazyCache.GetOrAdd( |
| 130 | + @object.GetType(), |
| 131 | + type => type.GetMethod(getterMethodName, BindingFlags.Instance | BindingFlags.Public)) |
| 132 | + ?? throw new InvalidOperationException($"'{getterMethodName}' method could not be resolved for {@object.GetType().Name}."); |
| 133 | + |
| 134 | + // checks |
| 135 | + var value = method.Invoke(@object, null) |
| 136 | + ?? throw new NullReferenceException($"{getterMethodName}() returned null."); |
| 137 | + |
| 138 | + return value as TResult |
| 139 | + ?? throw new NullReferenceException($"{@object.GetType().Name}.{getterMethodName}() returned an unexpected value of type {value.GetType().Name}. Expecting '{typeof(TResult).Name}'."); |
110 | 140 | }
|
111 |
| - private static MethodInfo _innerCommandMethod; |
| 141 | + |
| 142 | + /// <summary> |
| 143 | + /// Casts or unwraps given <see cref="IDbCommand"/> to <see cref="MySqlCommand"/>. |
| 144 | + /// </summary> |
| 145 | + /// <returns><see cref="IDbCommand"/> might be wrapped into another class (usually DB profiler class like MiniProfiler).</returns> |
| 146 | + public static MySqlCommand AsMySqlCommand(this IDbCommand command) => GetUnderlyingValue<IDbCommand, MySqlCommand>(command, ref s_iternalCommandMethod, "get_InternalCommand"); |
| 147 | + |
| 148 | + /// <summary> |
| 149 | + /// Casts or unwraps given <see cref="IDbCommand"/> to <see cref="MySqlCommand"/>. |
| 150 | + /// </summary> |
| 151 | + /// <returns><see cref="IDbCommand"/> might be wrapped into another class (usually DB profiler class like MiniProfiler).</returns> |
| 152 | + public static MySqlDataReader AsMySqlDataReader(this IDataReader reader) => GetUnderlyingValue<IDataReader, MySqlDataReader>(reader, ref s_wrappedReaderMethod, "get_WrappedReader"); |
| 153 | + |
| 154 | + /// <summary> |
| 155 | + /// Casts or unwraps given <see cref="IDbConnection"/> to <see cref="MySqlConnection"/>. |
| 156 | + /// </summary> |
| 157 | + /// <returns><see cref="IDbConnection"/> might be wrapped into another class (usually DB profiler class like MiniProfiler).</returns> |
| 158 | + public static MySqlConnection AsMySqlConnection(this IDbConnection connection) => GetUnderlyingValue<IDbConnection, MySqlConnection>(connection, ref s_wrappedConnectionMethod, "get_WrappedConnection"); |
| 159 | + |
| 160 | + static ConcurrentDictionary<Type, MethodInfo> |
| 161 | + s_iternalCommandMethod, |
| 162 | + s_wrappedReaderMethod, |
| 163 | + s_wrappedConnectionMethod; |
| 164 | + |
| 165 | + /// <summary> |
| 166 | + /// Returns the last insert ID from the MySqlCommand, eventually using the underlying MySqlCommand of another IDbCommand. |
| 167 | + /// </summary> |
| 168 | + /// <param name="command">Generic <see cref="IDbCommand"/> to work with</param> |
| 169 | + /// <returns>Last insert ID</returns> |
| 170 | + public static long LastInsertedId(IDbCommand command) => command.AsMySqlCommand().LastInsertedId; |
112 | 171 |
|
113 | 172 | /// <summary>
|
114 | 173 | /// Returns metadata about the columns in the result set.
|
115 | 174 | /// </summary>
|
116 |
| - /// <returns>A <see cref="System.Collections.ObjectModel.ReadOnlyCollection{DbColumn}"/> containing metadata about the result set.</returns> |
117 |
| - public static ReadOnlyCollection<DbColumn> GetColumnSchema( |
118 |
| - IDataReader reader) |
119 |
| - { |
120 |
| - // If we have a MySqlCommand, just use it |
121 |
| - if (reader is MySqlDataReader mySqlDataReader) return mySqlDataReader.GetColumnSchema(); |
122 |
| - |
123 |
| - // If we did not get one back, try to unwrap it as likely it's wrapped by a profiler like MiniProfiler |
124 |
| - if (_innerDataReaderMethod == null) |
125 |
| - _innerDataReaderMethod = reader.GetType().GetMethod("get_WrappedReader", BindingFlags.Instance | BindingFlags.Public); |
126 |
| - mySqlDataReader = _innerDataReaderMethod?.Invoke(reader, null) as MySqlDataReader; |
127 |
| - if (mySqlDataReader == null) throw new NullReferenceException("Could not get MySqlDataReader for wrapped reader!"); |
128 |
| - return mySqlDataReader.GetColumnSchema(); |
129 |
| - } |
130 |
| - private static MethodInfo _innerDataReaderMethod; |
| 175 | + /// <returns>A <see cref="ReadOnlyCollection{DbColumn}"/> containing metadata about the result set.</returns> |
| 176 | + public static ReadOnlyCollection<DbColumn> GetColumnSchema(IDataReader reader) => reader.AsMySqlDataReader().GetColumnSchema(); |
131 | 177 | }
|
132 | 178 | }
|
0 commit comments