ExpressionBuilder
class ExpressionBuilder
using Stashbox.Exceptions;
using Stashbox.Lifetime;
using Stashbox.Registration;
using Stashbox.Resolution;
using Stashbox.Utils;
using Stashbox.Utils.Data;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Reflection;
namespace Stashbox.Expressions
{
internal class ExpressionBuilder
{
private readonly ExpressionFactory expressionFactory;
private readonly ServiceRegistrator serviceRegistrator;
internal ExpressionBuilder(ExpressionFactory expressionFactory, ServiceRegistrator serviceRegistrator)
{
this.expressionFactory = expressionFactory;
this.serviceRegistrator = serviceRegistrator;
}
internal Expression BuildExpressionAndApplyLifetime(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext, Type resolveType)
{
if (!IsOutputLifetimeManageable(serviceRegistration))
return BuildExpressionForRegistration(serviceRegistration, resolutionContext, resolveType);
return serviceRegistration.RegistrationContext.Lifetime.ApplyLifetime(this, serviceRegistration, resolutionContext, resolveType);
}
internal Expression BuildExpressionForRegistration(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext, Type resolveType)
{
if (serviceRegistration.RegistrationType == RegistrationType.OpenGeneric) {
Type implementationType = serviceRegistration.ImplementationType.MakeGenericType(resolveType.GetGenericArguments());
ServiceRegistration serviceRegistration2 = serviceRegistration.Clone(implementationType, RegistrationType.Default);
serviceRegistration2.RegistrationContext.Name = null;
serviceRegistrator.Register(resolutionContext.CurrentContainerContext, serviceRegistration2, resolveType, false);
return BuildExpressionAndApplyLifetime(serviceRegistration2, resolutionContext, resolveType);
}
if (serviceRegistration.IsDecorator || resolutionContext.DecoratingService.Key == resolveType)
return BuildDisposalTrackingAndFinalizerExpression(serviceRegistration, resolutionContext, resolveType);
IEnumerable<ServiceRegistration> decoratorsOrDefault = resolutionContext.CurrentContainerContext.DecoratorRepository.GetDecoratorsOrDefault(resolveType);
if (decoratorsOrDefault == null) {
if (!resolveType.IsClosedGenericType())
return BuildDisposalTrackingAndFinalizerExpression(serviceRegistration, resolutionContext, resolveType);
decoratorsOrDefault = resolutionContext.CurrentContainerContext.DecoratorRepository.GetDecoratorsOrDefault(resolveType.GetGenericTypeDefinition());
if (decoratorsOrDefault == null)
return BuildDisposalTrackingAndFinalizerExpression(serviceRegistration, resolutionContext, resolveType);
}
Expression expression = BuildDisposalTrackingAndFinalizerExpression(serviceRegistration, resolutionContext, resolveType);
if (expression == null)
return null;
foreach (ServiceRegistration item in decoratorsOrDefault) {
ResolutionContext resolutionContext2 = resolutionContext.BeginDecoratingContext(resolveType, expression);
expression = BuildExpressionForRegistration(item, resolutionContext2, resolveType);
if (expression == null)
return null;
}
return expression;
}
private Expression BuildDisposalTrackingAndFinalizerExpression(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext, Type resolveType)
{
Expression expression = BuildExpressionByRegistrationType(serviceRegistration, resolutionContext, resolveType);
if (expression == null)
return null;
if (serviceRegistration.RegistrationContext.ExistingInstance == null && (object)serviceRegistration.RegistrationContext.Finalizer != null)
expression = BuildFinalizerExpression(expression, serviceRegistration, resolutionContext.CurrentScopeParameter);
if (!RegistrationHoldsDisposable(resolutionContext.CurrentContainerContext, serviceRegistration) || !expression.Type.IsDisposable())
return CheckRuntimeCircularDependencyExpression(expression, serviceRegistration, resolutionContext, resolveType);
MethodInfo methodInfo = Constants.AddDisposalMethod.MakeGenericMethod(expression.Type);
return CheckRuntimeCircularDependencyExpression(resolutionContext.CurrentScopeParameter.CallMethod(methodInfo, expression), serviceRegistration, resolutionContext, resolveType);
}
private Expression BuildExpressionByRegistrationType(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext, Type resolveType)
{
resolutionContext = (resolutionContext.ShouldFallBackToRequestInitiatorContext ? resolutionContext.BeginCrossContainerContext(resolutionContext.RequestInitiatorContainerContext) : resolutionContext);
switch (serviceRegistration.RegistrationType) {
case RegistrationType.Factory:
return GetExpressionForFactory(serviceRegistration, resolutionContext, resolveType);
case RegistrationType.Instance:
return serviceRegistration.RegistrationContext.ExistingInstance.AsConstant();
case RegistrationType.WireUp:
return expressionFactory.ConstructBuildUpExpression(serviceRegistration, resolutionContext, serviceRegistration.RegistrationContext.ExistingInstance.AsConstant(), serviceRegistration.ImplementationType);
case RegistrationType.Func:
return GetExpressionForFunc(serviceRegistration, resolutionContext);
default:
return GetExpressionForDefault(serviceRegistration, resolutionContext);
}
}
private Expression CheckRuntimeCircularDependencyExpression(Expression expression, ServiceRegistration serviceRegistration, ResolutionContext resolutionContext, Type resolveType)
{
if (!resolutionContext.CurrentContainerContext.ContainerConfiguration.RuntimeCircularDependencyTrackingEnabled)
return expression;
ParameterExpression parameterExpression = resolveType.AsVariable(null);
return new Expression[4] {
resolutionContext.CurrentScopeParameter.CallMethod(Constants.CheckRuntimeCircularDependencyBarrierMethod, serviceRegistration.RegistrationId.AsConstant(), resolveType.AsConstant()),
parameterExpression.AssignTo(expression),
resolutionContext.CurrentScopeParameter.CallMethod(Constants.ResetRuntimeCircularDependencyBarrierMethod, serviceRegistration.RegistrationId.AsConstant()),
parameterExpression
}.AsBlock(parameterExpression);
}
private static Expression BuildFinalizerExpression(Expression instanceExpression, ServiceRegistration serviceRegistration, Expression scopeExpression)
{
MethodInfo methodInfo = Constants.AddWithFinalizerMethod.MakeGenericMethod(instanceExpression.Type);
return scopeExpression.CallMethod(methodInfo, instanceExpression, serviceRegistration.RegistrationContext.Finalizer.AsConstant());
}
private static bool IsOutputLifetimeManageable(ServiceRegistration serviceRegistration)
{
if (serviceRegistration.RegistrationType != RegistrationType.OpenGeneric && serviceRegistration.RegistrationType != RegistrationType.Instance)
return serviceRegistration.RegistrationContext.Lifetime != null;
return false;
}
private static bool RegistrationHoldsDisposable(IContainerContext containerContext, ServiceRegistration serviceRegistration)
{
if (serviceRegistration.RegistrationContext.IsLifetimeExternallyOwned || serviceRegistration.RegistrationContext.ExistingInstance != null)
return false;
if (!containerContext.ContainerConfiguration.TrackTransientsForDisposalEnabled)
return !(serviceRegistration.RegistrationContext.Lifetime is TransientLifetime);
return true;
}
private Expression GetExpressionForDefault(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext)
{
if (resolutionContext.WeAreInCircle(serviceRegistration.RegistrationId))
throw new CircularDependencyException(serviceRegistration.ImplementationType, null);
resolutionContext.PullOutCircularDependencyBarrier(serviceRegistration.RegistrationId);
Expression result = PrepareDefaultExpression(serviceRegistration, resolutionContext);
resolutionContext.LetDownCircularDependencyBarrier();
return result;
}
private Expression PrepareDefaultExpression(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext)
{
if (serviceRegistration.RegistrationContext.DefinedScopeName == null)
return expressionFactory.ConstructExpression(serviceRegistration, resolutionContext);
ParameterExpression parameterExpression = Constants.ResolutionScopeType.AsVariable(null);
MethodCallExpression expression = resolutionContext.CurrentScopeParameter.ConvertTo(Constants.ResolverType).CallMethod(Constants.BeginScopeMethod, serviceRegistration.RegistrationContext.DefinedScopeName.AsConstant(), true.AsConstant());
ResolutionContext resolutionContext2 = resolutionContext.BeginNewScopeContext(new KeyValue<object, ParameterExpression>(serviceRegistration.RegistrationContext.DefinedScopeName, parameterExpression));
resolutionContext.AddDefinedVariable(parameterExpression);
resolutionContext.AddInstruction(parameterExpression.AssignTo(expression.ConvertTo(Constants.ResolutionScopeType)));
Expression result = expressionFactory.ConstructExpression(serviceRegistration, resolutionContext2);
foreach (ParameterExpression item in resolutionContext2.DefinedVariables.Walk()) {
resolutionContext.AddDefinedVariable(item);
}
foreach (Expression singleInstruction in resolutionContext2.SingleInstructions) {
resolutionContext.AddInstruction(singleInstruction);
}
return result;
}
private Expression GetExpressionForFactory(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext, Type resolveType)
{
Expression instance = ((object)serviceRegistration.RegistrationContext.ContainerFactory != null) ? ConstructFactoryExpression(serviceRegistration.RegistrationContext.ContainerFactory, serviceRegistration, resolutionContext.CurrentScopeParameter.ConvertTo(Constants.ResolverType)) : ConstructFactoryExpression(serviceRegistration.RegistrationContext.SingleFactory, serviceRegistration);
return expressionFactory.ConstructBuildUpExpression(serviceRegistration, resolutionContext, instance, resolveType);
}
private static Expression ConstructFactoryExpression(Delegate delegate, ServiceRegistration serviceRegistration, params Expression[] parameters)
{
if (serviceRegistration.RegistrationContext.IsFactoryDelegateACompiledLambda || delegate.IsCompiledLambda())
return delegate.InvokeDelegate(parameters);
MethodInfo method = delegate.GetMethod();
if (!method.IsStatic)
return method.CallMethod(delegate.Target.AsConstant(), parameters);
return method.CallStaticMethod(parameters);
}
private Expression GetExpressionForFunc(ServiceRegistration serviceRegistration, ResolutionContext resolutionContext)
{
MethodInfo method = serviceRegistration.RegistrationContext.FuncDelegate.GetMethod();
Expression[] funcParametersWithScope = GetFuncParametersWithScope(serviceRegistration.ImplementationType.GetSingleMethod("Invoke").GetParameters(), resolutionContext);
if (serviceRegistration.RegistrationContext.FuncDelegate.IsCompiledLambda())
return serviceRegistration.RegistrationContext.FuncDelegate.InvokeDelegate(funcParametersWithScope).AsLambda(funcParametersWithScope.Take(funcParametersWithScope.Length - 1).Cast<ParameterExpression>());
return (method.IsStatic ? method.CallStaticMethod(funcParametersWithScope) : serviceRegistration.RegistrationContext.FuncDelegate.Target.AsConstant().CallMethod(method, funcParametersWithScope)).AsLambda(funcParametersWithScope.Take(funcParametersWithScope.Length - 1).Cast<ParameterExpression>());
}
private static Expression[] GetFuncParametersWithScope(IList<ParameterInfo> parameterInfos, ResolutionContext resolutionContext)
{
int count = parameterInfos.Count;
Expression[] array = new Expression[count + 1];
for (int i = 0; i < count; i++) {
array[i] = parameterInfos[i].ParameterType.AsParameter(null);
}
array[array.Length - 1] = resolutionContext.CurrentScopeParameter.ConvertTo(Constants.ResolverType);
return array;
}
}
}