Search code examples
c#.netwcfmultiple-instancesself-hosting

Multiple instances of a self hosted WCF service


We have a "worker" service running from console application in c#, for development we were always running a single instance of this service, which fetches chunks of data and performs some calculations, these chunks of data are provided by another service (which keeps track of how much data is left etc.)

Now in QA we want to run multiple instances of the "worker" service simultaneously (on the same machine).However we are get an exception as soon as the second instance is launched:

The TransportManager failed to listen on the supplied URI using the NetTcpPortSharing service: the URI is already registered with the service.

We are using netTcpBinding and the endpoint address is hardcoded into the app.config and remains the same and because of that I assume we are getting this error.

<services>
    <service behaviorConfiguration="CoreBehavior" name="WorkerService">
        <endpoint address="net.tcp://localhost:8001/WorkerAssignment" binding="netTcpBinding" contract="IWorkerService" bindingConfiguration="CoreTcpBinding"/>
    </service>
</services>
<bindings>
    <netTcpBinding>
        <binding name="CoreTcpBinding" portSharingEnabled="true">
            <security mode="None"/>
        </binding>
    </netTcpBinding>
</bindings> 

Application code :

var host = new ServiceHost(typeof(WorkerService));
host.Open();

How do we provide a different URI for each instance so that atleast the port will remain the same ?

OR If there is a different way to run multiple instances of the same service?


Solution

  • If you want to have multiple instances of the service than it is enough to have single service host - just decorate you WorkerService with ServiceBehaviorAttribute

    [ServiceBehavior(InstanceContextMode = InstanceContextMode.Percall)] 
    public class WorkerService : IWorkerService 
    {
      //...service data
    }
    

    This will make sure that every call to the service will first create new instance of the service. Other ways of creating service class can be found here

    If, however you would like to have multiple service hosts than it is impossible to have two service hosts that will host same service on completely the same url.

    Another case would be if you want to have one service host hosting the same service on multiple endpoints with the same base address and custom uri's. In this case you can make use of overloaded ServiceHost constructor or investigate methods AddBaseAddress , AddServiceEndpoint. Or if you want to do it from configuration file than here's simple example with your code slightly modified

    <service behaviorConfiguration="CoreBehavior" name="WorkerService">
        <endpoint address="WorkerAssignment" binding="netTcpBinding" contract="IWorkerService"/>
        <endpoint address="QAWorkerAssignment" binding="netTcpBinding" contract="IWorkerService"/>
      <host>
        <baseAddresses>
          <add baseAddress="net.tcp://localhost:8001/" />
        </baseAddresses>
      </host>
    </service>
    

    with this configuration you will have two endpoints for your service

    net.tcp://localhost:8001/WorkerAssignment

    net.tcp://localhost:8001/QAWorkerAssignment