Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Initial version #1

Merged
merged 1 commit into from
Nov 6, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
30 changes: 30 additions & 0 deletions MetallicBlueDev.EntityGateCore/MetallicBlueDev.EntityGateCore.sln
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 16
VisualStudioVersion = 16.0.29418.71
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "MetallicBlueDev.EntityGateCore", "MetallicBlueDev.EntityGateCore\MetallicBlueDev.EntityGateCore.csproj", "{26EE1D92-1913-4205-BE53-25AE3AD32DB2}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{3EF1955C-1888-48C3-B438-09104D10B1EE}"
ProjectSection(SolutionItems) = preProject
MetallicBlueDev.EntityGateCore.sln.vsspell = MetallicBlueDev.EntityGateCore.sln.vsspell
EndProjectSection
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{26EE1D92-1913-4205-BE53-25AE3AD32DB2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{26EE1D92-1913-4205-BE53-25AE3AD32DB2}.Debug|Any CPU.Build.0 = Debug|Any CPU
{26EE1D92-1913-4205-BE53-25AE3AD32DB2}.Release|Any CPU.ActiveCfg = Release|Any CPU
{26EE1D92-1913-4205-BE53-25AE3AD32DB2}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {37E596CC-9314-473A-9F60-9D038788BDE2}
EndGlobalSection
EndGlobal
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
<?xml version="1.0" encoding="utf-8"?>
<!-- Visual Studio Spell Checker configuration file - [https://github.com/EWSoftware/VSSpellChecker]
Do not edit the XML. Use the configuration file editor in Visual Studio to modify the settings. -->
<SpellCheckerConfiguration Format="2018.8.16.0">
<InheritAdditionalDictionaryFolders>True</InheritAdditionalDictionaryFolders>
<SelectedLanguages>
<LanguageName>en-US</LanguageName>
</SelectedLanguages>
<InheritIgnoredClassifications>True</InheritIgnoredClassifications>
<IgnoredClassifications />
<InheritIgnoredWords>True</InheritIgnoredWords>
<InheritExclusionExpressions>True</InheritExclusionExpressions>
<InheritIgnoredFilePatterns>True</InheritIgnoredFilePatterns>
<InheritXmlSettings>True</InheritXmlSettings>
</SpellCheckerConfiguration>
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
<?xml version="1.0" encoding="utf-8"?>
<configuration>

<configSections>
<section name="EntityGateCoreConfigs" type="MetallicBlueDev.EntityGate.Configuration.EntityGateSectionHandler, MetallicBlueDev.EntityGate" />
</configSections>

<connectionStrings>
<add name="SampleDbInstance" connectionString="DATABASE=MyDbName;SERVER=my-db-host;Integrated Security=False;USER ID=MyUserName;PASSWORD=MyPassword;" providerName="System.Data.SqlClient" />
<add name="OtherDbInstance" connectionString="DATABASE=OtherDbName;SERVER=other-db-host;Integrated Security=False;USER ID=OtherUserName;PASSWORD=OtherPassword;" providerName="System.Data.SqlClient" />
<add name="DbInstance1" connectionString="DATABASE=OtherDbName;SERVER=other-db-host;Integrated Security=False;USER ID=OtherUserName;PASSWORD=OtherPassword;" providerName="System.Data.SqlClient" />
<add name="DbInstance2" connectionString="DATABASE=OtherDbName;SERVER=other-db-host;Integrated Security=False;USER ID=OtherUserName;PASSWORD=OtherPassword;" providerName="System.Data.SqlClient" />
</connectionStrings>

<EntityGateCoreConfigs>
<EntityGateCoreConfig>
<!--
Name of the connection string to use.
It must exist in the connectionStrings section.
-->
<ConnectionName>SampleDbInstance</ConnectionName>

<!-- Optional: Maximum number of attempts after a failure. -->
<MaximumNumberOfAttempts>5</MaximumNumberOfAttempts>

<!-- Optional: Waiting time after a failure. -->
<AttemptDelay>1000</AttemptDelay>

<!-- Optional: Timeout. -->
<Timeout>30</Timeout>

<!-- Optional: EF LazyLoading. -->
<LazyLoading>False</LazyLoading>

<!-- Optional: Determines if the backup of the original values is performed automatically. -->
<AutomaticCheckOfOriginalValues>False</AutomaticCheckOfOriginalValues>

<!-- Full type name for extending context options. -->
<ContextOptionsExtension>Microsoft.EntityFrameworkCore.SqlServer.Infrastructure.Internal.SqlServerOptionsExtension</ContextOptionsExtension>
</EntityGateCoreConfig>

<!-- Additional example. -->
<EntityGateCoreConfig>
<!-- Minimum configuration -->
<ConnectionName>DbInstance2</ConnectionName>
<ContextOptionsExtension>Microsoft.EntityFrameworkCore.SqlServer.Infrastructure.Internal.SqlServerOptionsExtension</ContextOptionsExtension>
</EntityGateCoreConfig>
<!-- In-line (attribute style). -->
<EntityGateCoreConfig
ConnectionName="OtherDbInstance"
MaximumNumberOfAttempts="3"
AttemptDelay="900"
Timeout="40"/>
<EntityGateCoreConfig ConnectionName="DbInstance1" LazyLoading="False" />
<EntityGateCoreConfig ConnectionName="DbInstance2" Timeout="1000" AutomaticCheckOfOriginalValues="False" />
</EntityGateCoreConfigs>

</configuration>
Original file line number Diff line number Diff line change
@@ -0,0 +1,283 @@
using System;
using System.Globalization;
using MetallicBlueDev.EntityGate.Extensions;
using MetallicBlueDev.EntityGate.GateException;
using MetallicBlueDev.EntityGate.Helpers;
using MetallicBlueDev.EntityGateCore.Properties;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.Extensions.Logging;

namespace MetallicBlueDev.EntityGate.Configuration
{
/// <summary>
/// Client configuration.
/// </summary>
[Serializable()]
public sealed class ClientConfiguration
{
private int maximumNumberOfAttempts = 0;
private int attemptDelay = 0;
private string connectionString = null;
private int timeout = 0;

/// <summary>
/// Default value of LazyLoading.
/// </summary>
public bool LazyLoading { get; internal set; } = true;

/// <summary>
/// Determines whether the configuration has been updated.
/// </summary>
public bool Changed { get; private set; } = false;

/// <summary>
/// Maximum number of attempts.
/// </summary>
public int MaximumNumberOfAttempts
{
get => maximumNumberOfAttempts;
set
{
if (value > 0)
{
maximumNumberOfAttempts = value;
ConfigurationChanged();
}
}
}

/// <summary>
/// Waiting time (in milliseconds) before the next attempt.
/// </summary>
public int AttemptDelay
{
get => attemptDelay;
set
{
if (value > 0)
{
attemptDelay = value;
ConfigurationChanged();
}
}
}

/// <summary>
/// The raw connection string.
/// </summary>
public string ConnectionString
{
get => connectionString;
set
{
if (value.IsNotNullOrEmpty())
{
connectionString = value;
ConfigurationChanged();
}
}
}

/// <summary>
/// Maximum time (in seconds) to run a query.
/// </summary>
public int Timeout
{
get => timeout;
set
{
if (value > 3)
{
timeout = value;
ConfigurationChanged();
}
}
}

/// <summary>
/// Get or set the status of the notification.
/// </summary>
public bool CanUseNotification { get; set; }

/// <summary>
/// Determines if the backup of the original values is performed automatically.
///
/// If the main entity implements the <see cref="InterfacedObject.IEntityObjectArchival"/> interface, the backup is automatically enabled.
/// </summary>
public bool AutomaticCheckOfOriginalValues { get; set; }

/// <summary>
/// Returns the logging system.
/// Use <see cref="SetLoggerFactory(ILoggerFactory)(string)"/> to define it.
/// </summary>
public ILogger Logger { get; private set; } = null;

/// <summary>
/// Determines whether it is possible to use the event log.
/// The <see cref="Logger"/> method must be defined.
/// </summary>
public bool CanUseLogging => Logger != null;

/// <summary>
/// Returns the <see cref="IDbContextOptionsExtension"/> used for configuration.
/// Use <see cref="SetContextOptionsExtension(string)"/> to define it.
/// </summary>
public IDbContextOptionsExtension ContextOptionsExtension { get; private set; } = null;

/// <summary>
/// The system provider of logging.
/// Use <see cref="SetLoggerFactory(ILoggerFactory)(string)"/> to define it.
/// </summary>
internal ILoggerFactory GateLoggerFactory { get; private set; } = null;

/// <summary>
/// New configuration.
/// </summary>
internal ClientConfiguration()
{
var defaultConfig = EntityGateCoreConfigLoader.GetFirstConfig();

if (defaultConfig != null)
{
ChangeConnectionString(defaultConfig.ConnectionName);
}
}

/// <summary>
/// Configuration of the logging system.
/// </summary>
/// <param name="loggerFactory">The system provider of logging.</param>
public void SetLoggerFactory(ILoggerFactory loggerFactory)
{
GateLoggerFactory = loggerFactory;

if (GateLoggerFactory != null)
{
Logger = GateLoggerFactory.CreateLogger("EntityGateCore");
}
}

/// <summary>
/// Create the context option extension with the full name of the <see cref="IDbContextOptionsExtension"/> type.
/// </summary>
/// <param name="fullTypeName">Full name of the <see cref="IDbContextOptionsExtension"/> type.</param>
public void SetContextOptionsExtension(string fullTypeName)
{
var optionExType = ReflectionHelper.MakeType(fullTypeName);
SetContextOptionsExtension(optionExType);
}

/// <summary>
/// Creating the context option extension with the requested <see cref="IDbContextOptionsExtension"/> type.
/// </summary>
/// <param name="contextOptionType"><see cref="IDbContextOptionsExtension"/> type.</param>
public void SetContextOptionsExtension(Type contextOptionType)
{
ContextOptionsExtension = ReflectionHelper.MakeInstance<IDbContextOptionsExtension>(contextOptionType);
}

/// <summary>
/// Change the connection string.
/// </summary>
/// <param name="connectionName">Name of the connection.</param>
public void ChangeConnectionString(string connectionName)
{
var currentConfig = EntityGateCoreConfigLoader.GetConfig(connectionName);

if (currentConfig != null)
{
CopyConfiguration(currentConfig);

if (CanUseLogging && Logger.IsEnabled(LogLevel.Information))
{
Logger.LogInformation(string.Format(CultureInfo.InvariantCulture, Resources.ConfigurationHasBeenLoaded, connectionName));
}
}
else if (CanUseLogging && Logger.IsEnabled(LogLevel.Error))
{
Logger.LogError(string.Format(CultureInfo.InvariantCulture, Resources.UnableToFindConnectionString, connectionName));
}
}

/// <summary>
/// Update the connection configuration.
/// </summary>
/// <param name="optionsBuilder"><see cref="IDbContextOptionsExtension"/> builder.</param>
internal void Update(DbContextOptionsBuilder optionsBuilder)
{
if (CanUseLogging)
{
optionsBuilder.UseLoggerFactory(GateLoggerFactory);
}

if (ContextOptionsExtension == null)
{
throw new ConfigurationEntityGateCoreException(Resources.ContextOptionsExtensionUndefined);
}

if (ContextOptionsExtension is RelationalOptionsExtension optionExt)
{
optionExt.WithConnectionString(ConnectionString);
optionExt.WithCommandTimeout(Timeout);
}

optionsBuilder.Options.WithExtension(ContextOptionsExtension);

ConfigurationSynchronized();
}

/// <summary>
/// Signal that the configuration has changed.
/// </summary>
private void ConfigurationChanged()
{
if (!Changed)
{
Changed = true;

if (CanUseLogging && Logger.IsEnabled(LogLevel.Information))
{
Logger.LogInformation(Resources.ConfigurationChanged);
}
}
}

/// <summary>
/// Copy of the configuration.
/// </summary>
/// <param name="config"></param>
private void CopyConfiguration(EntityGateCoreConfig config)
{
ConnectionString = EntityGateCoreConfigLoader.GetConnectionString(config.ConnectionName);

if (config.ContextOptionsExtension.IsNotNullOrEmpty())
{
SetContextOptionsExtension(config.ContextOptionsExtension);
}

MaximumNumberOfAttempts = config.MaximumNumberOfAttempts;
AttemptDelay = config.AttemptDelay;
LazyLoading = config.LazyLoading;
Timeout = config.Timeout;
AutomaticCheckOfOriginalValues = config.AutomaticCheckOfOriginalValues;
}

/// <summary>
/// Indicates that the context is synchronized to this configuration.
/// </summary>
private void ConfigurationSynchronized()
{
if (Changed)
{
Changed = false;

if (CanUseLogging && Logger.IsEnabled(LogLevel.Information))
{
Logger.LogInformation(Resources.ConfigurationSynchronized);
}
}
}
}
}

Loading