Skip to content

Commit 738c59b

Browse files
committed
mysql - getting underlying MySql-specific value reflection cleanups
moved to a single helper, added more specific exceptions
1 parent c5990b7 commit 738c59b

File tree

5 files changed

+89
-87
lines changed

5 files changed

+89
-87
lines changed

src/PDO/Peachpie.Library.PDO.MySQL/PDOMySQLDriver.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
using System.Data.Common;
44
using MySqlConnector;
55
using Pchp.Core;
6+
using Peachpie.Library.MySql;
67
using Peachpie.Library.PDO.Utilities;
78

89
namespace Peachpie.Library.PDO.MySQL
@@ -54,7 +55,7 @@ protected override string BuildConnectionString(ReadOnlySpan<char> dsn, string u
5455
public override string GetLastInsertId(PDO pdo, string name)
5556
{
5657
var command = pdo.GetCurrentCommand();
57-
var lastid = (command != null) ? MySqlExtensions.LastInsertedId(command) : -1;
58+
var lastid = command != null ? MySqlExtensions.LastInsertedId(command) : -1;
5859

5960
return lastid.ToString();
6061
}

src/PDO/Peachpie.Library.PDO.MySQL/PDOMySqlExtensions.cs

-33
This file was deleted.

src/PDO/Peachpie.Library.PDO.MySQL/Peachpie.Library.PDO.MySQL.csproj

+1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
</PropertyGroup>
1010

1111
<ItemGroup>
12+
<ProjectReference Include="..\..\Peachpie.Library.MySql\Peachpie.Library.MySql.csproj" />
1213
<ProjectReference Include="..\Peachpie.Library.PDO\Peachpie.Library.PDO.csproj" />
1314
</ItemGroup>
1415

src/Peachpie.Library.MySql/MySqlConnectionResource.cs

+10-23
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,15 @@ sealed class MySqlConnectionResource : ConnectionResource
2020
const string ResourceName = "mysql connection";
2121

2222
readonly MySqlConnectionManager _manager;
23+
2324
readonly IDbConnection _connection;
2425

26+
/// <summary>
27+
/// <see cref="_connection"/> lazily cast to <see cref="MySqlConnector.MySqlConnection"/>.
28+
/// <see cref="MySqlExtensions.GetUnderlyingValue"/> for details.
29+
/// </summary>
30+
MySqlConnection _mySqlConnection;
31+
2532
/// <summary>
2633
/// Whether to keep the underlying connection open after disposing this resource.
2734
/// (The owner of the connection is someone else)
@@ -83,22 +90,7 @@ public override void ClosePendingReader()
8390
/// the connection is a wrapped connection such as we get from MiniProfiler, and we look for WrappedConnection to
8491
/// find the native MySqlConnection when we need it.
8592
/// </summary>
86-
internal MySqlConnection MySqlConnection
87-
{
88-
get
89-
{
90-
if (_mySqlConnection != null) return _mySqlConnection;
91-
_mySqlConnection = _connection as MySqlConnection;
92-
if (_mySqlConnection != null) return _mySqlConnection;
93-
if (_innerConnectionMethod == null)
94-
_innerConnectionMethod = _connection.GetType().GetMethod("get_WrappedConnection", BindingFlags.Instance | BindingFlags.Public);
95-
_mySqlConnection = _innerConnectionMethod?.Invoke(_connection, null) as MySqlConnection;
96-
if (_mySqlConnection == null) throw new NullReferenceException("Could not get MySqlConnection for wrapped connection!");
97-
return _mySqlConnection;
98-
}
99-
}
100-
private MySqlConnection _mySqlConnection;
101-
private static MethodInfo _innerConnectionMethod;
93+
internal MySqlConnection MySqlConnection => _mySqlConnection ?? (_mySqlConnection = _connection.AsMySqlConnection());
10294

10395
protected override IDbConnection ActiveConnection => _connection;
10496

@@ -135,10 +127,7 @@ internal ResultResource ExecuteCommandInternal(IDbCommand command, bool convertT
135127
/// <summary>
136128
/// Pings the server.
137129
/// </summary>
138-
internal bool Ping()
139-
{
140-
return MySqlConnection.Ping();
141-
}
130+
internal bool Ping() => MySqlConnection.Ping();
142131

143132
/// <summary>
144133
/// Queries server for a value of a global variable.
@@ -168,9 +157,7 @@ internal long LastInsertedId
168157
get
169158
{
170159
var command = LastResult?.Command;
171-
return command != null
172-
? MySqlExtensions.LastInsertedId(command)
173-
: -1L;
160+
return command != null ? MySqlExtensions.LastInsertedId(command) : -1;
174161
}
175162
}
176163
}

src/Peachpie.Library.MySql/MySqlExtensions.cs

+76-30
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
11
using System;
2+
using System.Collections.Concurrent;
23
using System.Collections.Generic;
34
using System.Collections.ObjectModel;
45
using System.Data;
56
using System.Data.Common;
67
using System.Reflection;
78
using System.Text;
9+
using System.Threading;
810
using MySqlConnector;
911
using Pchp.Core;
1012
using Pchp.Library.Database;
@@ -91,42 +93,86 @@ public static ConnectionResource ValidConnection(Context ctx, PhpResource link =
9193
}
9294

9395
/// <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.
9598
/// </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
100110
{
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}'.");
110140
}
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;
112171

113172
/// <summary>
114173
/// Returns metadata about the columns in the result set.
115174
/// </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();
131177
}
132178
}

0 commit comments

Comments
 (0)