Подтвердить что ты не робот

Внедрение данных в службу WCF

У меня есть службы WCF, структурированные, как предложено Мигель Кастро. Это означает, что я настроил все вручную и имею консольное приложение, на котором размещаются мои службы, используя объекты ServiceHost.

Я хочу, чтобы мои классы обслуживания были худшими, и в настоящее время они просто передают вызовы классам поведения. Теперь моя проблема заключается в модульном тестировании классов обслуживания. Я хочу что-то добавить к классам в качестве параметра конструктора, чтобы я мог отмахиваться от этого и писать собственные изолированные тестовые тесты. Класс ServiceHost, похоже, не принимает аргументы, поэтому мой вопрос заключается в том, как я могу вводить данные в классы обслуживания - или не могу?

4b9b3361

Ответ 1

WCF поддерживает Injection конструктора, но вам нужно перепрыгнуть через несколько обручей, чтобы добраться туда. Ключ заключается в написании пользовательского сервиса ServiceHostFactory. Хотя это тоже должно иметь конструктор по умолчанию, вы можете использовать его для установления правильного поведения.

В качестве примера я недавно написал тот, который использует Castle Windsor для привязки зависимостей для реализации сервиса. Реализация CreateServiceHost просто делает это:

return new WindsorServiceHost(this.container, serviceType, baseAddresses);

где this.container - сконфигурированный IWindsorContainer.

WindsorServiceHost выглядит следующим образом:

public class WindsorServiceHost : ServiceHost
{
    public WindsorServiceHost(IWindsorContainer container, Type serviceType, params Uri[] baseAddresses)
        : base(serviceType, baseAddresses)
    {
        if (container == null)
        {
            throw new ArgumentNullException("container");
        }

        foreach (var cd in this.ImplementedContracts.Values)
        {
            cd.Behaviors.Add(new WindsorInstanceProvider(container));
        }
    }
}

и WindsorInstanceProvider выглядит следующим образом:

public class WindsorInstanceProvider : IInstanceProvider, IContractBehavior
{
    private readonly IWindsorContainer container;

    public WindsorInstanceProvider(IWindsorContainer container)
    {
        if (container == null)
        {
            throw new ArgumentNullException("container");
        }

        this.container = container;
    }

    #region IInstanceProvider Members

    public object GetInstance(InstanceContext instanceContext, Message message)
    {
        return this.GetInstance(instanceContext);
    }

    public object GetInstance(InstanceContext instanceContext)
    {
        var serviceType = instanceContext.Host.Description.ServiceType;
        return this.container.Resolve(serviceType);
    }

    public void ReleaseInstance(InstanceContext instanceContext, object instance)
    {
        this.container.Release(instance);
    }

    #endregion

    #region IContractBehavior Members

    public void AddBindingParameters(ContractDescription contractDescription, ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
    {
    }

    public void ApplyClientBehavior(ContractDescription contractDescription, ServiceEndpoint endpoint, ClientRuntime clientRuntime)
    {
    }

    public void ApplyDispatchBehavior(ContractDescription contractDescription, ServiceEndpoint endpoint, DispatchRuntime dispatchRuntime)
    {
        dispatchRuntime.InstanceProvider = this;
    }

    public void Validate(ContractDescription contractDescription, ServiceEndpoint endpoint)
    {
    }

    #endregion
}

Это может показаться много, но обратите внимание, что это многоразовый код общего назначения, который имеет довольно низкую циклическую сложность.

Вы можете выполнить ту же самую икону кодирования, чтобы внедрить инъекцию зависимостей с другим контейнером DI или с помощью программы "Бедный человек".

Здесь более старая запись этой идиомы, в которой используется Bad Man DI.

Ответ 2

Если вы использовали Castle Windsor, у него есть отличное средство интеграции WCF, которое позволяет вам делать это и намного более легко.

Ответ 3

Вы настроили свой сервис как Singleton? Я обнаружил, что реализации IInstanceProvider могут быть проблематичными при использовании контейнера DI для создания экземпляров службы.

Ответ 4

Статья Хостинг макета как службы WCF  содержит статический метод, который будет генерировать хост службы WCF на основе объекта, который передается в метод с одной конечной точкой.

Этот метод был также отправлен в для Рекомендуемых шаблонов для веб-сервисов для модульного тестирования.

Пример использования вызовов NSubstitute, но можно использовать другие фальшивые произведения.