Как интегрировать Fitnesse с Autofac/IOC?

Когда Fitnesse создает экземпляр прибора, он ищет открытый конструктор по умолчанию.

Тем не менее, я бы хотел, чтобы конструктор вставлял любые приложения, которые я хочу использовать в этом приспособлении.

Т.е. я хотел бы написать свой прибор это...

public class MyColumnFixture : ColumnFixture {
    private readonly IApplicationService _applicationService;

    public ManualExitSetupFixture(IApplicationService applicationService) {
        _applicationService = applicationService;
    }

    public void DoStuff(string arg1) {
        _applicationService.DoStuff(arg1);
    }
}

Лучшее, что мне удалось придумать, - это представить контейнер как синглтон (см. Ниже). Но должен быть лучший путь. Autofac прекрасно интегрируется со многими другими технологиями, которые мы используем.

public class AutofacIntegration {
    private static IContainer _container;

    public static IContainer Container {
        get {
           if (_container == null) {
              var builder = new ContainerBuilder();
              builder.RegisterModule<MyApplicationModule>();
               _container = builder.Build();                    
           }
           return _container;
         }
    }
}

public class MyColumnFixture : ColumnFixture {
    private readonly IApplicationService _applicationService;

    public ManualExitSetupFixture() {
        _applicationService = AutofacIntegration.Container.Resolve<IApplicationService>();
    }

    public void DoStuff(string arg1) {
        _applicationService.DoStuff(arg1);
    }
}

РЕДАКТИРОВАТЬ: Включая более подробно в моих попытках сделать эту работу на основе помощи Майка...

Я создал класс, который копируется из декомпилированного кода для CreateDefault, декомпилированного из Fitsharp...

public class CreateDefault<T, P> : Operator<T, P>, CreateOperator<T> where P : class, Processor<T>
{
    public bool CanCreate(NameMatcher memberName, Tree<T> parameters)
    {
        return true;
    }

    public TypedValue Create(NameMatcher memberName, Tree<T> parameters)
    {
        ...
    }
}

... и зарегистрировал это в SuiteConfig.xml...

<suiteConfig>
    <ApplicationUnderTest>
        <AddAssembly>..\..\Services\Win\MyProj.Fitnesse\MyProj.Fitnesse\bin\Debug\MyProj.Fitnesse.dll</AddAssembly>
     </ApplicationUnderTest>
     <Fit.Operators>
         <Add>MyProj.Fitnesse.CreateDefault`2</Add>
     </Fit.Operators>
</suiteConfig>

... это дает следующее исключение при попытке загрузки в мой класс CreateDefault<,>.

System.Reflection.TargetInvocationException: Exception has been thrown by the target of an invocation. ---> System.Reflection.TargetInvocationException: Exception has been thrown by the target of an invocation. ---> fitSharp.Machine.Exception.CreateException: Constructor with 0 parameter(s) failed for type 'MyProj.Fitnesse.CreateDefault`2'. ---> System.ArgumentException: Cannot create an instance of MyProj.Fitnesse.CreateDefault`2[T,P] because Type.ContainsGenericParameters is true.
   at System.RuntimeType.CreateInstanceCheckThis()
   at System.RuntimeType.CreateInstanceImpl(BindingFlags bindingAttr, Binder binder, Object[] args, CultureInfo culture, Object[] activationAttributes, StackCrawlMark& stackMark)
   at System.Activator.CreateInstance(Type type, BindingFlags bindingAttr, Binder binder, Object[] args, CultureInfo culture, Object[] activationAttributes)
   at System.Reflection.Assembly.CreateInstance(String typeName, Boolean ignoreCase, BindingFlags bindingAttr, Binder binder, Object[] args, CultureInfo culture, Object[] activationAttributes)
   at fitSharp.Machine.Engine.RuntimeType.CreateInstance()
   at fitSharp.Machine.Engine.CreateDefault`2.CreateWithoutParameters(RuntimeType runtimeType)
   --- End of inner exception stack trace ---
   at fitSharp.Machine.Engine.CreateDefault`2.CreateWithoutParameters(RuntimeType runtimeType)
   at fitSharp.Machine.Engine.CreateDefault`2.Create(NameMatcher memberName, Tree`1 parameters)
   --- End of inner exception stack trace ---
   at System.RuntimeMethodHandle.InvokeMethod(Object target, Object[] arguments, Signature sig, Boolean constructor)
   at System.Reflection.RuntimeMethodInfo.UnsafeInvokeInternal(Object obj, Object[] parameters, Object[] arguments)
   at System.Reflection.RuntimeMethodInfo.Invoke(Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture)
   at System.RuntimeType.InvokeMember(String name, BindingFlags bindingFlags, Binder binder, Object target, Object[] providedArgs, ParameterModifier[] modifiers, CultureInfo culture, String[] namedParams)
   at fitSharp.Machine.Engine.MethodMember.TryInvoke(Object[] parameters)
   at fitSharp.Machine.Engine.ReflectionMember.Invoke(Object[] parameters)
   at fitSharp.Machine.Engine.ProcessorBase`2.Operate[O](Object[] parameters)
   at fitSharp.Machine.Engine.Operators`2.Add(String operatorName)
   --- End of inner exception stack trace ---
   at fitSharp.Machine.Engine.ProcessorExtension.InvokeWithThrow[T](Processor`1 processor, TypedValue instance, MemberName memberName, Tree`1 parameters)
   at fitSharp.Machine.Application.SuiteConfiguration.LoadNode(String typeName, XmlNode methodNode)
   at fitSharp.Machine.Application.SuiteConfiguration.LoadXml(String configurationXml)
   at fitSharp.Machine.Application.ArgumentParser.InvokeArgumentHandler(String switch, String argumentValue)
   at fitSharp.Machine.Application.ArgumentParser.Parse(IList`1 commandLineArguments)
   at fitSharp.Machine.Application.Shell.Run(IList`1 commandLineArguments)

РЕДАКТИРОВАТЬ: Большое спасибо, Майк, теперь это работает удовольствие.

Реализация действительно довольно проста, и я уверен, что она может быть улучшена, но вот быстрые инструкции...

  1. Определить пользовательский CreateOperator
  2. Создайте SuiteConfig.xml для указания на dll и загрузите пользовательский оператор создания
  3. Укажите SuiteConfig при запуске бегуна
  4. Зарегистрировать светильники

Мой код CreateOperator...

public class AutofacCreateOperator : CellOperator, CreateOperator<Cell>
{
    private static IContainer _container;

    public AutofacCreateOperator()
    {
        var builder = new ContainerBuilder();
        builder.RegisterModule<FitnesseModule>();
        _container = builder.Build();
    }

    public bool CanCreate(NameMatcher memberName, Tree<Cell> parameters)
    {
        return _container.ComponentRegistry.IsRegistered(new TypedService(Type.GetType(memberName.MatchName)));
    }

    public TypedValue Create(NameMatcher memberName, Tree<Cell> parameters)
    {
        return new TypedValue(_container.Resolve(Type.GetType(memberName.MatchName)));
    }
}

1 ответ

Решение

Это можно сделать - вам нужно написать собственный класс для обработки операции создания для fitSharp. Он реализует этот интерфейс:

public interface CreateOperator<T> {
    bool CanCreate(NameMatcher memberName, Tree<T> parameters);
    TypedValue Create(NameMatcher memberName, Tree<T> parameters);
}

Затем вы указываете fitSharp использовать этот класс в файле конфигурации пакета:

<suiteConfig>
    <Fit.Operators>
        <Add>My.Create.Operator</Add>
    </Fit.Operators>
    ...
</suiteConfig>

В качестве пояснения приведу очень простой оператор создания. Вы бы изменили это, чтобы сделать инъекцию AutoFac.

public class TestCreateOperator: CellOperator, CreateOperator<Cell> {
  public bool CanCreate(NameMatcher memberName, Tree<Cell> parameters) {
    return memberName.Matches("testname");
  }
  public TypedValue Create(NameMatcher memberName, Tree<Cell> parameters) {
    return new TypedValue("mytestname");
  }
}

К вашему сведению, вот встроенный класс, который делает создание прибора.

public class CreateDefault<T,P>: Operator<T, P>, CreateOperator<T> where P: class, Processor<T> {
    public bool CanCreate(NameMatcher memberName, Tree<T> parameters) {
        return true;
    }

public TypedValue Create(NameMatcher memberName, Tree<T> parameters) {
    var runtimeType = Processor.ApplicationUnderTest.FindType(memberName);
    return parameters.Branches.Count == 0
                 ? CreateWithoutParameters(runtimeType)
                 : CreateWithParameters(parameters, runtimeType);
}

static TypedValue CreateWithoutParameters(RuntimeType runtimeType) {
    try {
        return runtimeType.CreateInstance();
    }
    catch (System.Exception e) {
        throw new CreateException(runtimeType.Type, 0, e.InnerException ?? e);
    }
}

TypedValue CreateWithParameters(Tree<T> parameters, RuntimeType runtimeType) {
    RuntimeMember member = runtimeType.GetConstructor(parameters.Branches.Count);
    object[] parameterList = new ParameterList<T>(Processor).GetParameterList(TypedValue.Void, parameters, member);
        try {
            return member.Invoke(parameterList);
        }
        catch (System.Exception e) {
            throw new CreateException(runtimeType.Type, parameterList.Length, e.InnerException ?? e);
        }
    }
}
Другие вопросы по тегам