mirror of https://github.com/AMT-Cheif/Stylet.git
Support building abstract factories for two interfaces with the same name
Fixes #123
This commit is contained in:
parent
2b80db4db7
commit
f946b8b9ac
|
@ -0,0 +1,155 @@
|
||||||
|
using StyletIoC;
|
||||||
|
using StyletIoC.Creation;
|
||||||
|
using StyletIoC.Internal;
|
||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Reflection.Emit;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
namespace StyletIoC.Internal
|
||||||
|
{
|
||||||
|
internal class AbstractFactoryBuilder
|
||||||
|
{
|
||||||
|
private readonly ModuleBuilder moduleBuilder;
|
||||||
|
|
||||||
|
public AbstractFactoryBuilder()
|
||||||
|
{
|
||||||
|
var assemblyName = new AssemblyName(StyletIoCContainer.FactoryAssemblyName);
|
||||||
|
var assemblyBuilder = AssemblyBuilder.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run);
|
||||||
|
|
||||||
|
var moduleBuilder = assemblyBuilder.DefineDynamicModule("StyletIoCFactoryModule");
|
||||||
|
this.moduleBuilder = moduleBuilder;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type GetFactoryForType(Type serviceType)
|
||||||
|
{
|
||||||
|
if (serviceType == null)
|
||||||
|
throw new ArgumentNullException(nameof(serviceType));
|
||||||
|
|
||||||
|
// Not thread-safe, as it's only called from the builder
|
||||||
|
if (!serviceType.IsInterface)
|
||||||
|
throw new StyletIoCCreateFactoryException(String.Format("Unable to create a factory implementing type {0}, as it isn't an interface", serviceType.GetDescription()));
|
||||||
|
|
||||||
|
// If the service is 'ISomethingFactory', call our new class 'GeneratedSomethingFactory'
|
||||||
|
var typeBuilder = this.moduleBuilder.DefineType(this.CreateImplementationName(serviceType), TypeAttributes.Public);
|
||||||
|
typeBuilder.AddInterfaceImplementation(serviceType);
|
||||||
|
|
||||||
|
// Define a field which holds a reference to the registration context
|
||||||
|
var registrationContextField = typeBuilder.DefineField("registrationContext", typeof(IRegistrationContext), FieldAttributes.Private);
|
||||||
|
|
||||||
|
// Add a constructor which takes one argument - the container - and sets the field
|
||||||
|
// public Name(IRegistrationContext registrationContext)
|
||||||
|
// {
|
||||||
|
// this.registrationContext = registrationContext;
|
||||||
|
// }
|
||||||
|
var ctorBuilder = typeBuilder.DefineConstructor(MethodAttributes.Public, CallingConventions.Standard, new[] { typeof(IRegistrationContext) });
|
||||||
|
var ilGenerator = ctorBuilder.GetILGenerator();
|
||||||
|
// Load 'this' and the registration context onto the stack
|
||||||
|
ilGenerator.Emit(OpCodes.Ldarg_0);
|
||||||
|
ilGenerator.Emit(OpCodes.Ldarg_1);
|
||||||
|
// Store the registration context in this.registrationContext
|
||||||
|
ilGenerator.Emit(OpCodes.Stfld, registrationContextField);
|
||||||
|
ilGenerator.Emit(OpCodes.Ret);
|
||||||
|
|
||||||
|
// These are needed by all methods, so get them now
|
||||||
|
// IRegistrationContext.GetTypeOrAll(Type, string)
|
||||||
|
// IRegistrationContext extends ICreator, and it's ICreator that actually implements this
|
||||||
|
var containerGetMethod = typeof(IContainer).GetMethod("GetTypeOrAll", new[] { typeof(Type), typeof(string) });
|
||||||
|
// Type.GetTypeFromHandler(RuntimeTypeHandle)
|
||||||
|
var typeFromHandleMethod = typeof(Type).GetMethod("GetTypeFromHandle");
|
||||||
|
|
||||||
|
// Go through each method, emmitting an implementation for each
|
||||||
|
foreach (var methodInfo in serviceType.GetMethods())
|
||||||
|
{
|
||||||
|
var parameters = methodInfo.GetParameters();
|
||||||
|
if (!(parameters.Length == 0 || (parameters.Length == 1 && parameters[0].ParameterType == typeof(string))))
|
||||||
|
throw new StyletIoCCreateFactoryException("Can only implement methods with zero arguments, or a single string argument");
|
||||||
|
|
||||||
|
if (methodInfo.ReturnType == typeof(void))
|
||||||
|
throw new StyletIoCCreateFactoryException("Can only implement methods which return something");
|
||||||
|
|
||||||
|
var attribute = methodInfo.GetCustomAttribute<InjectAttribute>(true);
|
||||||
|
|
||||||
|
var methodBuilder = typeBuilder.DefineMethod(methodInfo.Name, MethodAttributes.Public | MethodAttributes.Virtual, methodInfo.ReturnType, parameters.Select(x => x.ParameterType).ToArray());
|
||||||
|
var methodIlGenerator = methodBuilder.GetILGenerator();
|
||||||
|
// Load 'this' onto stack
|
||||||
|
// Stack: [this]
|
||||||
|
methodIlGenerator.Emit(OpCodes.Ldarg_0);
|
||||||
|
// Load value of 'registrationContext' field of 'this' onto stack
|
||||||
|
// Stack: [this.registrationContext]
|
||||||
|
methodIlGenerator.Emit(OpCodes.Ldfld, registrationContextField);
|
||||||
|
// New local variable which represents type to load
|
||||||
|
LocalBuilder lb = methodIlGenerator.DeclareLocal(methodInfo.ReturnType);
|
||||||
|
// Load this onto the stack. This is a RuntimeTypeHandle
|
||||||
|
// Stack: [this.registrationContext, runtimeTypeHandleOfReturnType]
|
||||||
|
methodIlGenerator.Emit(OpCodes.Ldtoken, lb.LocalType);
|
||||||
|
// Invoke Type.GetTypeFromHandle with this
|
||||||
|
// This is equivalent to calling typeof(T)
|
||||||
|
// Stack: [this.registrationContext, typeof(returnType)]
|
||||||
|
methodIlGenerator.Emit(OpCodes.Call, typeFromHandleMethod);
|
||||||
|
// Load the given key (if it's a parameter), or the key from the attribute if given, or null, onto the stack
|
||||||
|
// Stack: [this.registrationContext, typeof(returnType), key]
|
||||||
|
if (parameters.Length == 0)
|
||||||
|
{
|
||||||
|
if (attribute == null)
|
||||||
|
methodIlGenerator.Emit(OpCodes.Ldnull);
|
||||||
|
else
|
||||||
|
methodIlGenerator.Emit(OpCodes.Ldstr, attribute.Key); // Load null as the key
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
methodIlGenerator.Emit(OpCodes.Ldarg_1); // Load the given string as the key
|
||||||
|
}
|
||||||
|
// Call container.Get(type, key)
|
||||||
|
// Stack: [returnedInstance]
|
||||||
|
methodIlGenerator.Emit(OpCodes.Callvirt, containerGetMethod);
|
||||||
|
methodIlGenerator.Emit(OpCodes.Ret);
|
||||||
|
|
||||||
|
typeBuilder.DefineMethodOverride(methodBuilder, methodInfo);
|
||||||
|
}
|
||||||
|
|
||||||
|
Type constructedType;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
constructedType = typeBuilder.CreateType();
|
||||||
|
}
|
||||||
|
catch (TypeLoadException e)
|
||||||
|
{
|
||||||
|
throw new StyletIoCCreateFactoryException(String.Format("Unable to create factory type for interface {0}. Ensure that the interface is public, or add [assembly: InternalsVisibleTo(StyletIoCContainer.FactoryAssemblyName)] to your AssemblyInfo.cs", serviceType.GetDescription()), e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return constructedType;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void AddFriendlierNameForType(StringBuilder sb, Type type)
|
||||||
|
{
|
||||||
|
var typeInfo = type.GetTypeInfo();
|
||||||
|
if (typeInfo.IsGenericType)
|
||||||
|
{
|
||||||
|
sb.Append(type.GetGenericTypeDefinition().FullName.Replace('.', '+'));
|
||||||
|
sb.Append("<>["); // Just so that they can't fool us with carefully-crafted interface names...
|
||||||
|
foreach (var arg in typeInfo.GetGenericArguments())
|
||||||
|
{
|
||||||
|
this.AddFriendlierNameForType(sb, arg);
|
||||||
|
}
|
||||||
|
sb.Append("]");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sb.Append(type.FullName.Replace('.', '+'));
|
||||||
|
}
|
||||||
|
sb.Append("<>").Append(typeInfo.Assembly.GetName().Name.Replace('.', '+'));
|
||||||
|
}
|
||||||
|
|
||||||
|
private string CreateImplementationName(Type interfaceType)
|
||||||
|
{
|
||||||
|
var sb = new StringBuilder();
|
||||||
|
// Make this unspeakable, just in case...
|
||||||
|
sb.Append("Stylet.AutoGenerated.<>");
|
||||||
|
this.AddFriendlierNameForType(sb, interfaceType);
|
||||||
|
return sb.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -9,7 +9,7 @@ using System.Diagnostics;
|
||||||
using System.Diagnostics.CodeAnalysis;
|
using System.Diagnostics.CodeAnalysis;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using System.Reflection.Emit;
|
|
||||||
|
|
||||||
namespace StyletIoC.Internal
|
namespace StyletIoC.Internal
|
||||||
{
|
{
|
||||||
|
@ -45,9 +45,9 @@ namespace StyletIoC.Internal
|
||||||
private readonly ConcurrentDictionary<RuntimeTypeHandle, BuilderUpper> builderUppers = new ConcurrentDictionary<RuntimeTypeHandle, BuilderUpper>();
|
private readonly ConcurrentDictionary<RuntimeTypeHandle, BuilderUpper> builderUppers = new ConcurrentDictionary<RuntimeTypeHandle, BuilderUpper>();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Cached ModuleBuilder used for building factory implementations
|
/// Builder used to build abstract factories
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private ModuleBuilder factoryBuilder;
|
private AbstractFactoryBuilder abstractFactoryBuilder;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Fired when this container is asked to dispose
|
/// Fired when this container is asked to dispose
|
||||||
|
@ -431,107 +431,10 @@ namespace StyletIoC.Internal
|
||||||
|
|
||||||
internal Type GetFactoryForType(Type serviceType)
|
internal Type GetFactoryForType(Type serviceType)
|
||||||
{
|
{
|
||||||
// Not thread-safe, as it's only called from the builder
|
if (this.abstractFactoryBuilder == null)
|
||||||
if (!serviceType.IsInterface)
|
this.abstractFactoryBuilder = new AbstractFactoryBuilder();
|
||||||
throw new StyletIoCCreateFactoryException(String.Format("Unable to create a factory implementing type {0}, as it isn't an interface", serviceType.GetDescription()));
|
|
||||||
|
|
||||||
if (this.factoryBuilder == null)
|
return this.abstractFactoryBuilder.GetFactoryForType(serviceType);
|
||||||
{
|
|
||||||
var assemblyName = new AssemblyName(StyletIoCContainer.FactoryAssemblyName);
|
|
||||||
var assemblyBuilder = AssemblyBuilder.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run);
|
|
||||||
var moduleBuilder = assemblyBuilder.DefineDynamicModule("StyletIoCFactoryModule");
|
|
||||||
this.factoryBuilder = moduleBuilder;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the service is 'ISomethingFactory', call our new class 'GeneratedSomethingFactory'
|
|
||||||
var typeBuilder = this.factoryBuilder.DefineType("Generated" + serviceType.Name.Substring(1), TypeAttributes.Public);
|
|
||||||
typeBuilder.AddInterfaceImplementation(serviceType);
|
|
||||||
|
|
||||||
// Define a field which holds a reference to the registration context
|
|
||||||
var registrationContextField = typeBuilder.DefineField("registrationContext", typeof(IRegistrationContext), FieldAttributes.Private);
|
|
||||||
|
|
||||||
// Add a constructor which takes one argument - the container - and sets the field
|
|
||||||
// public Name(IRegistrationContext registrationContext)
|
|
||||||
// {
|
|
||||||
// this.registrationContext = registrationContext;
|
|
||||||
// }
|
|
||||||
var ctorBuilder = typeBuilder.DefineConstructor(MethodAttributes.Public, CallingConventions.Standard, new[] { typeof(IRegistrationContext) });
|
|
||||||
var ilGenerator = ctorBuilder.GetILGenerator();
|
|
||||||
// Load 'this' and the registration context onto the stack
|
|
||||||
ilGenerator.Emit(OpCodes.Ldarg_0);
|
|
||||||
ilGenerator.Emit(OpCodes.Ldarg_1);
|
|
||||||
// Store the registration context in this.registrationContext
|
|
||||||
ilGenerator.Emit(OpCodes.Stfld, registrationContextField);
|
|
||||||
ilGenerator.Emit(OpCodes.Ret);
|
|
||||||
|
|
||||||
// These are needed by all methods, so get them now
|
|
||||||
// IRegistrationContext.GetTypeOrAll(Type, string)
|
|
||||||
// IRegistrationContext extends ICreator, and it's ICreator that actually implements this
|
|
||||||
var containerGetMethod = typeof(IContainer).GetMethod("GetTypeOrAll", new[] { typeof(Type), typeof(string) });
|
|
||||||
// Type.GetTypeFromHandler(RuntimeTypeHandle)
|
|
||||||
var typeFromHandleMethod = typeof(Type).GetMethod("GetTypeFromHandle");
|
|
||||||
|
|
||||||
// Go through each method, emmitting an implementation for each
|
|
||||||
foreach (var methodInfo in serviceType.GetMethods())
|
|
||||||
{
|
|
||||||
var parameters = methodInfo.GetParameters();
|
|
||||||
if (!(parameters.Length == 0 || (parameters.Length == 1 && parameters[0].ParameterType == typeof(string))))
|
|
||||||
throw new StyletIoCCreateFactoryException("Can only implement methods with zero arguments, or a single string argument");
|
|
||||||
|
|
||||||
if (methodInfo.ReturnType == typeof(void))
|
|
||||||
throw new StyletIoCCreateFactoryException("Can only implement methods which return something");
|
|
||||||
|
|
||||||
var attribute = methodInfo.GetCustomAttribute<InjectAttribute>(true);
|
|
||||||
|
|
||||||
var methodBuilder = typeBuilder.DefineMethod(methodInfo.Name, MethodAttributes.Public | MethodAttributes.Virtual, methodInfo.ReturnType, parameters.Select(x => x.ParameterType).ToArray());
|
|
||||||
var methodIlGenerator = methodBuilder.GetILGenerator();
|
|
||||||
// Load 'this' onto stack
|
|
||||||
// Stack: [this]
|
|
||||||
methodIlGenerator.Emit(OpCodes.Ldarg_0);
|
|
||||||
// Load value of 'registrationContext' field of 'this' onto stack
|
|
||||||
// Stack: [this.registrationContext]
|
|
||||||
methodIlGenerator.Emit(OpCodes.Ldfld, registrationContextField);
|
|
||||||
// New local variable which represents type to load
|
|
||||||
LocalBuilder lb = methodIlGenerator.DeclareLocal(methodInfo.ReturnType);
|
|
||||||
// Load this onto the stack. This is a RuntimeTypeHandle
|
|
||||||
// Stack: [this.registrationContext, runtimeTypeHandleOfReturnType]
|
|
||||||
methodIlGenerator.Emit(OpCodes.Ldtoken, lb.LocalType);
|
|
||||||
// Invoke Type.GetTypeFromHandle with this
|
|
||||||
// This is equivalent to calling typeof(T)
|
|
||||||
// Stack: [this.registrationContext, typeof(returnType)]
|
|
||||||
methodIlGenerator.Emit(OpCodes.Call, typeFromHandleMethod);
|
|
||||||
// Load the given key (if it's a parameter), or the key from the attribute if given, or null, onto the stack
|
|
||||||
// Stack: [this.registrationContext, typeof(returnType), key]
|
|
||||||
if (parameters.Length == 0)
|
|
||||||
{
|
|
||||||
if (attribute == null)
|
|
||||||
methodIlGenerator.Emit(OpCodes.Ldnull);
|
|
||||||
else
|
|
||||||
methodIlGenerator.Emit(OpCodes.Ldstr, attribute.Key); // Load null as the key
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
methodIlGenerator.Emit(OpCodes.Ldarg_1); // Load the given string as the key
|
|
||||||
}
|
|
||||||
// Call container.Get(type, key)
|
|
||||||
// Stack: [returnedInstance]
|
|
||||||
methodIlGenerator.Emit(OpCodes.Callvirt, containerGetMethod);
|
|
||||||
methodIlGenerator.Emit(OpCodes.Ret);
|
|
||||||
|
|
||||||
typeBuilder.DefineMethodOverride(methodBuilder, methodInfo);
|
|
||||||
}
|
|
||||||
|
|
||||||
Type constructedType;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
constructedType = typeBuilder.CreateType();
|
|
||||||
}
|
|
||||||
catch (TypeLoadException e)
|
|
||||||
{
|
|
||||||
throw new StyletIoCCreateFactoryException(String.Format("Unable to create factory type for interface {0}. Ensure that the interface is public, or add [assembly: InternalsVisibleTo(StyletIoCContainer.FactoryAssemblyName)] to your AssemblyInfo.cs", serviceType.GetDescription()), e);
|
|
||||||
}
|
|
||||||
|
|
||||||
return constructedType;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public BuilderUpper GetBuilderUpper(Type type)
|
public BuilderUpper GetBuilderUpper(Type type)
|
||||||
|
|
|
@ -165,15 +165,18 @@ namespace StyletIoC
|
||||||
var container = new Container(this.autobindAssemblies);
|
var container = new Container(this.autobindAssemblies);
|
||||||
|
|
||||||
// Just in case they want it
|
// Just in case they want it
|
||||||
this.Bind<IContainer>().ToInstance(container).DisposeWithContainer(false).AsWeakBinding();
|
var bindings = this.bindings.ToList();
|
||||||
|
var containerBuilderBindTo = new BuilderBindTo(typeof(IContainer), this.GetAssemblies);
|
||||||
|
containerBuilderBindTo.ToInstance(container).DisposeWithContainer(false).AsWeakBinding();
|
||||||
|
bindings.Add(containerBuilderBindTo);
|
||||||
|
|
||||||
// For each binding which is weak, if another binding exists with any of the same type+key which is strong, we remove this binding
|
// For each binding which is weak, if another binding exists with any of the same type+key which is strong, we remove this binding
|
||||||
var groups = (from binding in this.bindings
|
var groups = (from binding in bindings
|
||||||
from serviceType in binding.ServiceTypes
|
from serviceType in binding.ServiceTypes
|
||||||
select new { ServiceType = serviceType, Binding = binding })
|
select new { ServiceType = serviceType, Binding = binding })
|
||||||
.ToLookup(x => x.ServiceType);
|
.ToLookup(x => x.ServiceType);
|
||||||
|
|
||||||
var filtered = from binding in this.bindings
|
var filtered = from binding in bindings
|
||||||
where !(binding.IsWeak &&
|
where !(binding.IsWeak &&
|
||||||
binding.ServiceTypes.Any(serviceType => groups.Contains(serviceType) && groups[serviceType].Any(groupItem => !groupItem.Binding.IsWeak)))
|
binding.ServiceTypes.Any(serviceType => groups.Contains(serviceType) && groups[serviceType].Any(groupItem => !groupItem.Binding.IsWeak)))
|
||||||
select binding;
|
select binding;
|
||||||
|
|
|
@ -3,6 +3,14 @@ using StyletIoC;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
|
||||||
|
namespace StyletUnitTests.SubNamespace
|
||||||
|
{
|
||||||
|
public interface I1Factory
|
||||||
|
{
|
||||||
|
StyletIoCFactoryTests.I1 GetI1();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
namespace StyletUnitTests
|
namespace StyletUnitTests
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
|
@ -192,5 +200,23 @@ namespace StyletUnitTests
|
||||||
var factory = ioc.Get<IGenericFactory<I1>>();
|
var factory = ioc.Get<IGenericFactory<I1>>();
|
||||||
Assert.IsInstanceOf<C1>(factory.GetI1());
|
Assert.IsInstanceOf<C1>(factory.GetI1());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void SupportsTwoInterfacesWithTheSameNames()
|
||||||
|
{
|
||||||
|
var builder = new StyletIoCBuilder();
|
||||||
|
builder.Bind<I1>().ToAbstractFactory();
|
||||||
|
builder.Bind<SubNamespace.I1Factory>().ToAbstractFactory();
|
||||||
|
Assert.DoesNotThrow(() => builder.BuildContainer());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void SupportsBuildingTheSameBuilderTwice()
|
||||||
|
{
|
||||||
|
var builder = new StyletIoCBuilder();
|
||||||
|
builder.Bind<I1>().ToAbstractFactory();
|
||||||
|
builder.BuildContainer();
|
||||||
|
Assert.DoesNotThrow(() => builder.BuildContainer());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue