4

My goal is to create a host application able to parse multiple assemblies, detect the contracts and host the services.

In order to load a service, we usually need to hardcode the servicehost instantiation. the following code is working despite not being the behaviour I'm looking for.

ServiceHost wService1Host = new ServiceHost(typeof(Service1));
wService1Host.Open();

ServiceHost wService2Host = new ServiceHost(typeof(Service2));
wService2Host.Open();

However, this mean I know in advance what the services would be. I don't mind having a reference to the assemblies containing the services. I just want the host not knowing about what services are contained within the assemblies. For example, if I add a new Service to one of the assemblies, no changes would be needed on the host side.

This is very similar to this question, but with an added complexity for the reason mentioned above.

Here is the host code I've come with so far. I don't mind managing the services at the moment, I simply want them to be loaded properly.

class Program
  {
    static void Main(string[] args)
    {

      // find currently executing assembly
      Assembly curr = Assembly.GetExecutingAssembly();

      // get the directory where this app is running in
      string currentLocation = Path.GetDirectoryName(curr.Location);

      // find all assemblies inside that directory
      string[] assemblies = Directory.GetFiles(currentLocation, "*.dll");

      // enumerate over those assemblies
      foreach (string assemblyName in assemblies)
      {
        // load assembly just for inspection
        Assembly assemblyToInspect = Assembly.ReflectionOnlyLoadFrom(assemblyName);

        // I've hardcoded the name of the assembly containing the services only to ease debugging
        if (assemblyToInspect != null && assemblyToInspect.GetName().Name == "WcfServices")
        {
          // find all types
          Type[] types = assemblyToInspect.GetTypes();

          // enumerate types and determine if this assembly contains any types of interest
          // you could e.g. put a "marker" interface on those (service implementation)
          // types of interest, or you could use a specific naming convention (all types
          // like "SomeThingOrAnotherService" - ending in "Service" - are your services)
          // or some kind of a lookup table (e.g. the list of types you need to find from
          // parsing the app.config file)
          foreach (Type ty in types)
          {
            Assembly implementationAssembly = Assembly.GetAssembly(ty);
            // When loading the type for the service, load it from the implementing assembly.
            Type implementation = implementationAssembly.GetType(ty.FullName);

            ServiceHost wServiceHost = new ServiceHost(implementation); // FAIL
            wServiceHost.Open();
          }
        }
      }
      Console.WriteLine("Service are up and running.");
      Console.WriteLine("Press <Enter> to stop services...");
      Console.ReadLine();
    }
  }

I get the following error when trying to create the serviceHost :

"It is illegal to reflect on the custom attributes of a Type loaded via ReflectionOnlyGetType (see Assembly.ReflectionOnly) -- use CustomAttributeData instead."

In the link given above, the guy seems to have solved its problem using typeof since he knows in advance what service he wants to expose. Unfortunately, this is not my case.

Note : For the hosting part, I actually have 3 projects. The first one is the host application (see above), the second one, is an assembly containing all my service's contract (the interfaces) and the last assembly contains the services implementation.

Here is the app.config I actually use for hosting the services. The assembly containing the implementation is named "WcfServices" and contains 2 services. One is exposing callbacks and the other only basic services.

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <system.serviceModel>    
    <behaviors>
      <serviceBehaviors>  
        <behavior name="metadataBehavior">
          <serviceMetadata httpGetEnabled="true"/>
        </behavior>        
      </serviceBehaviors>  
    </behaviors>
    <services>
      <service name="WcfServices.Service1"
               behaviorConfiguration="metadataBehavior">

        <endpoint address="Service1Service"
                  binding="basicHttpBinding"
                  contract="WcfServices.IService1"
                  name="basicHttp"/>

        <endpoint binding="mexHttpBinding"
                  contract="IMetadataExchange"
                  name="metadataExchange"/>

        <host>
          <baseAddresses>
            <add baseAddress="http://localhost:8000/Service1"/>
          </baseAddresses>
        </host>        
      </service>

      <service name="WcfServices.Service2"
               behaviorConfiguration="metadataBehavior">

        <endpoint address="Service2Service"
                  binding="wsDualHttpBinding"
                  contract="WcfServices.IService2"/>

        <endpoint address="mex"
                  binding="mexHttpBinding"
                  contract="IMetadataExchange"/>

        <host>
          <baseAddresses>
            <add baseAddress="http://localhost:8000/Service2"/>
          </baseAddresses>
        </host>
      </service>

    </services>    
  </system.serviceModel>
</configuration>

So, to be clear, here's what I'm looking for :
1. Load assemblies in current app directory
2. Looks if there are any contracts implementation in it
3. If there are, instantiate those services (using app.config for the moment)

First of all, is this even possible ? (My guess would be it is since an application named wcfstorm alread seems to do this)
Obviously, How could I make the code above works ?

Thank you!

Community
  • 1
  • 1
Sim
  • 265
  • 1
  • 4
  • 14
  • Horrible solution but... Can you use activator to create an instance of the implementation and then pass the type in to the ServiceHost by calling GetType on that instance. – RobH Oct 23 '12 at 17:40
  • @Munchies By adding the following : `var test = Activator.CreateInstance(implementation);` I receive the error "The requested operation is invalid in the ReflectionOnly context." Also, wouldn't it be rather inefficient instantiating each service twice ? (by creating the ServiceHost object I believe a new instance of the service is created) – Sim Oct 23 '12 at 22:36
  • I've just spotted the problem. You're loading your assembly in a reflection only context. Load your assembly with Assembly.Load or Assembly.LoadFrom and your original code should work. – RobH Oct 24 '12 at 08:32
  • @Munchies Indeed, it worked. Thanks for pointing that out. However, given that implementation, in a performance point of view, would it be better to parse every assembly with the reflection only context and then use LoadFrom only on the assembly I want or use loadFrom for every assembly ? – Sim Oct 24 '12 at 15:07
  • 1
    as far as I know, it doesn't matter, you're still loading the assembly into your app domain. I think reflection only is used for dlls from other versions of the framework/platforms etc. Although it doesn't load dependencies... I guess one solution would be to configure a folder for the service dlls to be dropped, then you should only be loading the assemblies you actually need so you shouldn't need to worry :) – RobH Oct 24 '12 at 15:17
  • Hey @Sim, you can put your solution in an [answer to your own question](http://meta.stackexchange.com/q/17463/162730), which will help others find your solution if they land on this question. – Jeroen Oct 24 '12 at 18:44

1 Answers1

4

Here's what I ended up doing :

private static void LoadServices()
{
  // find currently executing assembly
  Assembly Wcurr = Assembly.GetExecutingAssembly();

  // get the directory where this app is running in
  string wCurrentLocation = Path.GetDirectoryName(Wcurr.Location);

  // enumerate over those assemblies
  foreach (string wAssemblyName in mAssemblies)
  {
    // load assembly just for inspection
    Assembly wAssemblyToInspect = null;
    try
    {
      wAssemblyToInspect = Assembly.LoadFrom(wCurrentLocation + "\\" + wAssemblyName);
    }
    catch (System.Exception ex)
    {
      Console.WriteLine("Unable to load assembly : {0}", wAssemblyName);
    }


    if (wAssemblyToInspect != null)
    {
      // find all types with the HostService attribute
      IEnumerable<Type> wTypes = wAssemblyToInspect.GetTypes().Where(t => Attribute.IsDefined(t, typeof(HostService), false));

      foreach (Type wType in wTypes)
      {
        ServiceHost wServiceHost = new ServiceHost(wType);
        wServiceHost.Open();
        mServices.Add(wServiceHost);
        Console.WriteLine("New Service Hosted : {0}", wType.Name);
      }
    }
  }

  Console.WriteLine("Services are up and running.");
}

Note : This approach requires that the assemblies be referenced by the "host" project.

Note2 : In order to accelerate the assembly parsing, I've hardcoded which assemblies to load in "mAssemblies".

Sim
  • 265
  • 1
  • 4
  • 14