2010-01-11 4 views
18

Ich habe WCF-Dienste strukturiert wie von Miguel Castro vorgeschlagen. Das bedeutet, dass ich alles manuell eingerichtet habe und eine Konsolenanwendung meine Dienste unter Verwendung von ServiceHost-Objekten hostet.Einfügen von Daten in einen WCF-Dienst

Ich möchte meine Serviceklassen dünn halten und sie geben gerade Aufrufe an Verhaltensklassen weiter. Mein Problem ist jetzt das Testen der Serviceklassen. Ich möchte etwas in die Klassen als Konstruktorparameter einfügen, so dass ich das nachahmen kann und geeignete isolierte Komponententests schreiben kann. Die ServiceHost-Klasse scheint keine Argumente zu akzeptieren. Daher lautet meine Frage, wie ich Daten in die Serviceklassen einspeisen kann - oder nicht?

+0

verwenden Sie einen IoC-Container? und wenn ja welche? – Fabiano

+0

Ich verwende noch keinen IoC-Container auf der Serverseite. Ich plane jetzt eine einzuführen. Ich verwende Unity auf der Client-Seite (wie es mit Prism kam), aber erwäge, StructureMap auf dem Server zu verwenden. Offen für jeden wirklich. – stiank81

Antwort

29

WCF unterstützt Konstruktor Injection, aber Sie müssen durch ein paar Reifen springen, um dorthin zu gelangen. Der Schlüssel liegt darin, eine benutzerdefinierte ServiceHostFactory zu schreiben. Auch wenn dies einen Standardkonstruktor haben muss, können Sie damit alle korrekten Verhaltensweisen verbinden.

Als Beispiel habe ich kürzlich eine geschrieben, die Castle Windsor verwendet, um Abhängigkeiten für die Service-Implementierung zu verkabeln. Die Implementierung von CreateServiceHost einfach tut dies:

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

wo this.container ist eine konfigurierte IWindsorContainer.

WindsorServiceHost sieht wie folgt aus:

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)); 
     } 
    } 
} 

und WindsorInstanceProvider sieht wie folgt aus:

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 
} 

Dieses wie viel aussehen, aber feststellen, dass es wiederverwendbar, Allzweck-Code ist, dass ein eher hat geringe zyklomatische Komplexität.

Sie können dieselbe Codierungssprache verwenden, um Dependency Injection mit einem anderen DI-Container zu implementieren, oder indem Sie DI von Poor Man verwenden.

Hier ist ein older writeup dieser Idiom, die Armer Menschen DI verwendet.

+0

Danke für Ihre Antwort. Sieht vielversprechend aus! Will morgen hinein graben! – stiank81

+0

Sehr hilfreiche Antwort. –

+0

Was war der Vorteil dieses Ansatzes, anstatt die Wcf-Funktion zu benutzen ??? – CrazyDart

1

Haben Sie Ihren Dienst als Singleton konfiguriert? Ich habe herausgefunden, dass IInstanceProvider-Implementierungen problematisch sein können, wenn Sie einen DI-Container zum Erstellen der Dienstinstanzen verwenden.

5

Wenn Sie Castle Windsor verwenden, hat es eine große WCF-Integrationsmöglichkeit, mit der Sie dies und vieles mehr sehr einfach tun können.