2021-09-15 15:41:20 +02:00
|
|
|
using System;
|
2014-03-13 20:14:56 +11:00
|
|
|
using System.Data;
|
|
|
|
|
using System.Data.Common;
|
2020-09-16 13:08:27 +02:00
|
|
|
using Microsoft.Extensions.Logging;
|
2018-05-30 16:40:25 +02:00
|
|
|
using StackExchange.Profiling.Data;
|
2021-02-09 10:22:42 +01:00
|
|
|
using Umbraco.Cms.Core;
|
2021-02-12 13:36:50 +01:00
|
|
|
using Umbraco.Cms.Infrastructure.Persistence.FaultHandling;
|
2014-03-13 20:14:56 +11:00
|
|
|
|
2021-02-12 13:36:50 +01:00
|
|
|
namespace Umbraco.Extensions
|
2014-03-13 20:14:56 +11:00
|
|
|
{
|
2019-12-12 12:55:17 +01:00
|
|
|
public static class DbConnectionExtensions
|
2014-03-13 20:14:56 +11:00
|
|
|
{
|
2021-09-15 15:41:20 +02:00
|
|
|
public static bool IsConnectionAvailable(string connectionString, DbProviderFactory factory)
|
2014-03-13 20:14:56 +11:00
|
|
|
{
|
2021-09-09 13:08:27 +02:00
|
|
|
var connection = factory?.CreateConnection();
|
2016-04-12 19:55:50 +02:00
|
|
|
|
2016-09-01 19:06:08 +02:00
|
|
|
if (connection == null)
|
2019-12-12 08:11:23 +01:00
|
|
|
throw new InvalidOperationException($"Could not create a connection for provider \"{factory}\".");
|
2016-04-12 19:55:50 +02:00
|
|
|
|
2016-09-01 19:06:08 +02:00
|
|
|
connection.ConnectionString = connectionString;
|
|
|
|
|
using (connection)
|
2014-03-13 20:14:56 +11:00
|
|
|
{
|
|
|
|
|
return connection.IsAvailable();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static bool IsAvailable(this IDbConnection connection)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
connection.Open();
|
|
|
|
|
connection.Close();
|
|
|
|
|
}
|
2016-04-12 19:55:50 +02:00
|
|
|
catch (DbException e)
|
2014-03-13 20:14:56 +11:00
|
|
|
{
|
2015-07-30 15:36:33 +01:00
|
|
|
// Don't swallow this error, the exception is super handy for knowing "why" its not available
|
2020-11-10 08:50:47 +00:00
|
|
|
StaticApplicationLogging.Logger.LogWarning(e, "Configured database is reporting as not being available.");
|
2014-03-13 20:14:56 +11:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-14 14:06:30 +01:00
|
|
|
/// <summary>
|
|
|
|
|
/// Unwraps a database connection.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>UmbracoDatabase wraps the original database connection in various layers (see
|
|
|
|
|
/// OnConnectionOpened); this unwraps and returns the original database connection.</remarks>
|
|
|
|
|
internal static IDbConnection UnwrapUmbraco(this IDbConnection connection)
|
|
|
|
|
{
|
2018-05-30 16:40:25 +02:00
|
|
|
var unwrapped = connection;
|
2021-09-15 15:41:20 +02:00
|
|
|
|
2018-05-30 16:40:25 +02:00
|
|
|
IDbConnection c;
|
2016-12-14 14:06:30 +01:00
|
|
|
do
|
|
|
|
|
{
|
2018-05-30 16:40:25 +02:00
|
|
|
c = unwrapped;
|
2016-12-14 14:06:30 +01:00
|
|
|
|
2021-09-15 15:41:20 +02:00
|
|
|
if (unwrapped is ProfiledDbConnection profiled)
|
|
|
|
|
{
|
|
|
|
|
unwrapped = profiled.WrappedConnection;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (unwrapped is RetryDbConnection retrying)
|
|
|
|
|
{
|
|
|
|
|
unwrapped = retrying.Inner;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
while (c != unwrapped);
|
2016-12-14 14:06:30 +01:00
|
|
|
|
|
|
|
|
return unwrapped;
|
|
|
|
|
}
|
2014-03-13 20:14:56 +11:00
|
|
|
}
|
|
|
|
|
}
|