ARTICLE

A Simple Duplex Service in WCF

Posted by Dhananjay Kumar Articles | WCF with C# September 06, 2010
This article will give step by step explanation on how to create a simple duplex service in WCF.
Reader Level:
Download Files:
 

Objective

This article will give step by step explanation on how to create a simple duplex service in WCF.

In call back operation or duplex service, Service can also call some function at the client .

Duplex1.gif

  1. Duplex service allows calling back some operation (function) on the client.
     
  2. Duplex service also knows as Call Backs. 
     
  3. All Binding does not support duplex service. 
     
  4. Duplex communication is not standard. They are absolute Microsoft feature. 
     
  5. wsDualHttpBinding supports duplex communication over HTTP binding. 
     
  6. Duplex communication is possible over netTcpBinding and netNamedPipeBinding

Steps involved in creating a duplex service

  1. Create a WCF Service
     
  2. Create a call back contract. 
     
  3. Create service contract 
     
  4. Implement service and configure the behavior for duplex communication and call the client function at the service. 
     
  5. Configure the endpoint for duplex service 
     
  6. Create the client. 
     
  7. Call the WCF Service

Create a WCF Service

  1. Open visual studio and create a new project. Select WCF Service application from WCF project template tab.
     
  2. Delete all the default generated code. 
     
  3. If using VS2010, open web.config and comment serviceHostEnvironment as below. 

    Duplex2.gif
     
  4. If using VS2008 then delete the default End Point generated. 
     
  5. Now you are left with empty IService1 and Service1.svc.cs file.

Create a call back contract

  1. Open IService1.cs.
     
  2. Add an interface in the same file. Give any name of the interface. I am giving name here IMyContractCallBack 

    Duplex3.gif
     
  3. Create an operation contract in the Interface.
     
  4. Make sure return type is Void. 
     
  5. Make sure IsOneWayProperty is set to true. 
     
  6. There is no need to mark call back interface as ServiceContract.

Explanation

At the client side, this call back interface will get implemented and on a duplex channel Service will call the CallBackFunction from the client.

Create service contract

  1. Open IService1.
     
  2. Declare operation contract called from the client. Make the isoneway to property to true from the service operation contract also. 

    Duplex4.gif
     
  3. Set the CallBackContract in Service contract attribute. Set the call back contract as the interface created for the call back.

    Duplex5.gif

Here IMyContractCallBack is name of the interface user for duplex communication.

So Service contract and call back contract will be as below,

IService1.cs

using System;

using System.Collections.Generic;

using System.Linq;

using System.Runtime.Serialization;

using System.ServiceModel;

using System.ServiceModel.Web;

using System.Text;


namespace
WcfService3

{

    [ServiceContract(CallbackContract=typeof(IMyContractCallBack))]

    public interface IService1

    {

        [OperationContract(IsOneWay=true)]

        void   NormalFunction();

    } 

    public interface IMyContractCallBack

    {

        [OperationContract(IsOneWay=true)]

        void    CallBackFunction(string str);

    }

}

Implement service and configure for duplex communication and call client function

  1. In service behavior set the instance mode of the service to percall.

    Duplex6.gif
     
  2. Create a call back channel

    Duplex7.gif

    Here I am using OperationConetxt to get current call back channel.
     
  3. Now instance of IMyContractCallBack can be used to make call at the function in call back interface contract at the client side. Like below ,

    Duplex8.gif
Here we are going to call function at the client side from the operation contract of the service.

Service1.svc.cs

using System;

using System.Collections.Generic;

using System.Linq;

using System.Runtime.Serialization;

using System.ServiceModel;

using System.ServiceModel.Web;

using System.Text;


namespace
WcfService3

{  

    [ServiceBehavior(InstanceContextMode= InstanceContextMode.PerCall)]

    public class Service1 : IService1

    {  

        public void     NormalFunction()

        {

            IMyContractCallBack callback = OperationContext.Current.GetCallbackChannel<IMyContractCallBack>();       

            callback.CallBackFunction("Calling from Call Back");

        }

    }

}

Configure the endpoint for duplex service

As we discussed earlier that duplex service is not supported by all type of bindings. So while configuring the end point for the service we need to choose the binding supports duplex communication. Here we are going to use wsDualHttpBinding

  1. Declare the end point with wsDualHttpBinding.

    Duplex9.gif
     
  2. Declare the Meta data exchange end point. 

    Duplex10.gif
     
  3. Declare the host address 

    Duplex11.gif
     
  4. Declare the service behavior

    Duplex12.gif

So the service in configuration will look like

Duplex13.gif

Web.Config

<?xml version="1.0"?>

<configuration>

  <system.web>

    <compilation debug="true" targetFramework="4.0" />

  </system.web>

  <system.serviceModel>

    <behaviors>

      <serviceBehaviors>

        <behavior name ="svcbh">        

          <serviceMetadata httpGetEnabled="False"/>        

          <serviceDebug includeExceptionDetailInFaults="False"/>

        </behavior>

      </serviceBehaviors>

    </behaviors>

    <!--<serviceHostingEnvironment multipleSiteBindingsEnabled="true" />-->

    <services>

      <service name ="WcfService3.Service1" behaviorConfiguration ="svcbh" >

        <host>

          <baseAddresses>

            <add baseAddress = "http//localhost:9000/Service1/" />

          </baseAddresses>

        </host>

        <endpoint name ="duplexendpoint"

                  address =""

                  binding ="wsDualHttpBinding"

                  contract ="WcfService3.IService1"/>

        <endpoint name ="MetaDataTcpEndpoint"

                  address="mex"

                  binding="mexHttpBinding"

                  contract="IMetadataExchange"/>

      </service>

    </services>

  </system.serviceModel>

 <system.webServer>

    <modules runAllManagedModulesForAllRequests="true"/>

  </system.webServer

</configuration>

Create the client

  1. Create a console application to consume this service.
     
  2. Add the reference of System.ServiceModel. In console application project. 
     
  3. Add the service reference of the service, we created in previous steps.

Create a duplex proxy class to implement call back contract

  1. Right click and add a class in console application. Give any name; I am giving the name MyCallBack.
     
  2. Add the namespace 

    Duplex14.gif
     
  3. Implement call back contract. In our case it is IService1Callbackand IDisposable.

    Duplex15.gif

    Note: Since name of the service contract is IService1, so the call back interface name will be IService1Callback . It is name of service contract suffixes by keyword CallBack.
     
  4. Now implement the call back contract method in this class. 

    Duplex16.gif
     
  5. Now make a function in this class to create duplex proxy.

    a. Create an instance of InstanceContext and pass the reference of current class in the constructor of that.

    Duplex17.gif

    b. Pass this context as parameter of service1client

    Duplex18.gif

    c. Call the service on this proxy

    Duplex18.1.gif

So, the function will be

Duplex19.gif


For your reference MyCallback class will be as below,

MyCallback.cs

using System;

using System.Collections.Generic;

using System.Linq;

using System.Text;

using ConsoleApplication1.ServiceReference1;

using System.ServiceModel;


namespace
ConsoleApplication1

{

    class MyCallBack :IService1Callback ,IDisposable

    {

        Service1Client proxy;    

        public void   CallBackFunction(string str)

        {

            Console.WriteLine(str);

        }

        public void callService()

        {

           InstanceContext context = new InstanceContext(this);

           proxy = new Service1Client(context);

           proxy.NormalFunction();

        }

        public void Dispose()

        {

            proxy.Close();

        }

    }

}

Call the WCF Service

Create the instance of MyCallBack class and call the callService() method to call the service.

Program.cs

using System;

using System.Collections.Generic;

using System.Linq;

using System.Text;

using ConsoleApplication1.ServiceReference1;

using System.ServiceModel;


namespace
ConsoleApplication1

{

    class Program

    {

        static void Main(string[] args)

        {

            MyCallBack obj = new MyCallBack();

            obj.callService();

            Console.Read();

            obj.Dispose();

        }

    }

}

Output

Duplex20.gif 

Login to add your contents and source code to this article
post comment
     

I am getting the follwoing error.HTTP could not register URL http://+:80/Temporary_Listen_Addresses/ebcf34bf-68d2-403c-948a-8bd10fa76d09/ because TCP port 80 is being used by another application.Please help me.

Posted by ramana murthy May 09, 2013

Thanks Dhananjay, but this gives error: There was no endpoint listening at http://localhost:4471/Service1.svc that could accept the message. This is often caused by an incorrect address or SOAP action. See InnerException, if present, for more details.

Posted by Rashid Quamar Jan 12, 2013

Thanks Dhananjay Kumar.It was vey good. 1)How can I set service on my website? 2)How can I send data from client to service?

Posted by sajad beto Apr 25, 2012

Sir I have executed above Program CallBackContract (Which should be same name in the client and server side) Since it is Declared in the Server side and Client Side Implementation your using in Server side IMyContractCallBack and Your implementing at client side as IService1CallBack

Posted by sANKATI mallesham Jan 12, 2012

i have face problem in this implementation while running this console application.it give me a AddressAlreadyUseException...i dont know what is actual problem....please help me..

Posted by hitesh hitesh Dec 21, 2010
COMMENT USING
PREMIUM SPONSORS
DynamicPDF™ product line allows you to dynamically generate PDF documents, merge PDF documents and add new content to existing PDF documents from within your applications.
SPONSORED BY
  • PDF reports have never been easier to create. With our included WYSIWYG Designer, you can layout your reports, set up your data source and let DynamicPDF ReportWriter do the rest.