Wednesday 31 July 2013

WCF Transactions in easy 5 Steps with Example

Hi Friends,

In this article, I will first explain what is transaction and what are the properties of transaction and then I will try to explain how we can implement transactions in WCF service. So we will create WCF services which do database transactions and then check the transaction. I will first understand the steps how to enable transactions in WCF services. At the end of the article, I will try to force an error and see how the transaction is rolled back after the error.


What is Transactions?

A transaction is a unit of work that is performed against a database. Transactions are units or sequences of work accomplished in a logical order, whether in a manual fashion by a user or automatically by some sort of a database program.

Properties of Transactions:

Transactions have the following four standard properties, usually referred to by the acronym ACID:
·         Atomicity: ensures that all operations within the work unit are completed successfully; otherwise, the transaction is aborted at the point of failure, and previous operations are rolled back to their former state.
·         Consistency: ensures that the database properly changes states upon a successfully committed transaction.
·         Isolation: enables transactions to operate independently of and transparent to each other.
·         Durability: ensures that the result or effect of a committed transaction persists in case of a system failure.


There are easy 5 steps for setting transactions in WCF

Step 1:  For service operation Setting.

Create a WCF Service for Account and Enable TransactionFlowOption Mandatory
There are 3 modes for transaction flow option

•    Allowed    Transaction may be flowed.
•    Mandatory    Transaction must be flowed.
•    NotAllowed    A transaction should not be flowed. This is the default value.


In the below service I creates one Service contract IAccountService and two Service operations Deposit and Withdraw for particular Account ID a

[ServiceContract]
     public  interface IAccountService
    {
        [OperationContract]
        [TransactionFlow(TransactionFlowOption.Mandatory)]
        [FaultContract(typeof(ErrorInfo))]
         void Deposit(int AccountId, decimal amount);

        [OperationContract]
        [TransactionFlow(TransactionFlowOption.Mandatory)]
        [FaultContract(typeof(ErrorInfo))]
         void WithDraw(int AccountId, decimal amount);

    }


Step 2: In Operational Behavior Setting

In Operation Behavior we have to enable TransactionScopeRequired = true and TransactionAutoComplete=true

·         TransactionScopeRequired: indicates whether the method requires a transaction scope for its execution.
·         TransactionAutoComplete: indicates whether to automatically complete the current transaction scope if no unhandled exceptions occur.

[ServiceBehavior(TransactionIsolationLevel=System.Transactions.IsolationLevel.Serializable)]
   public class AccountService:IAccountService
    {
        [OperationBehavior(TransactionScopeRequired=true,TransactionAutoComplete=true)]
        public void Deposit(int AccountId, decimal amount)
        {
//Your Code Here
 }

[OperationBehavior(TransactionScopeRequired=true,TransactionAutoComplete=true)]
        public void withDraw(int AccountId, decimal amount)
        {
//Your Code Here

}
    }




Step 3: Enable Transaction Flow using WCF Service Config File

Binding - There are only few bindings in WCF that can support transaction, these are NetTcpBinding, NetNamedPipeBinding, WSHttpBinding, WSDualHttpBinding, and WSFederationHttpBinding. So, in order to configure transaction, one has to make a choice out of such bindings. Though these bindings do have transaction support but by default it is disabled, so one has to make them enabled.

We also need to enable transactions for wsHttpBinding by setting the transactionFlow attribute to true.

  <bindings>
            <wsHttpBinding>
                <binding name="WsHttpConfig" transactionFlow="true">
                    <reliableSession enabled="true" />
                </binding>
            </wsHttpBinding>
  </bindings>


Also give this setting to bindingConfiguration

<services>
            <service behaviorConfiguration="MetaBehavior" name="WCFClassLibrary.AccountService">
                <endpoint address="BankService" binding="wsHttpBinding" bindingConfiguration="WsHttpConfig"
                    name="BankService_http" contract="WCFClassLibrary.IAccountService" />
                <host>
                    <baseAddresses>
                        <add baseAddress="http://localhost:8055" />
                    </baseAddresses>
                </host>
            </service>
        </services>


Step 4: On the Client Side Use TrasactionScope

We need to use TrasactionScope Attribute on to the client side for Transaction.
Create the TrasactionScope to execute the commands, guaranteeing that both commands can commit or roll back as a single unit of work.

using (TransactionScope t = new TransactionScope())
                {
                    AccountService.AccountServiceClient proxy = new AccountService.AccountServiceClient();
                    proxy.Deposit(1, 2000);
                    proxy.WithDraw(2, 200);
                    t.Complete();
                    proxy.Close();
                }

Step 5: Check the Transactions is Working or not

In the Withdraw method we are checking that if the Account balance is less than 500 then we are throwing the fault exception and Transaction is rollback

                if (accountBalance - amount < 500)

                    throw new FaultException<ErrorInfo>(new ErrorInfo() { ErrorCode = "200", ErrorDescription = "InSufficient Balance found" }, "InSufficient Balance found");

In this way you can test the Transactions is working or not on the client side.

I have attached Code snippet for that. Please download this.


Happy Programming!!

If you have any query mail me to Sujeet.bhujbal@gmail.com     


Regards

Sujeet Bhujbal

------------------------------------------------------------------------------------------------------------
------------------------------------------------------------------------------------------------------



Tuesday 16 July 2013

WCF Self Hosting with Example and Source Code








Introduction

Windows Communication Foundation (WCF) is a programming interface as part of .NET Framework.

Using 3 types we can host service in WCF
1          1. Self-Hosting
2           2 .IIS Hosting
3          3. WAS


In this article I will tell you how to create WCF Service and how to host using self-hosting and how to consume this service in Web Application. In this article I will discuss about self-hosting your WCF Service, because it’s the most flexible and easiest way to host WCF services. Another way to host WCF Service is using Internet Information Services(IIS). IIS will be explained in my other article.



Download Source Code



Part-1   Create the Service

Open Visual Studio 2010 

Step 1: Select the new Empty Solution and give name WCFProject.


Step 2: Then add new Class Library Project and give name SaleClassLibrary.

Step 3: Then add reference of System.ServiceModel and System.Runtime.Serialization into that project.

Step 4: Then add new Interface ISaleService.
//
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.ServiceModel.Web;
using System.Text;
using System.Collections.Generic;
using System.Runtime.Serialization;
using System.ServiceModel;

 [ServiceContract]
    interface ISaleService
    {
        [OperationContract]
         bool InsertCustomer(Customer obj);

        [OperationContract]
        List<Customer> GetAllCustomer();

        [OperationContract]
        bool DeleteCustomer(int Cid);

        [OperationContract]
        bool UpdateCustomer(Customer obj);    
    }

  [DataContract]
   public class Customer
    {
        [DataMember]
        public int CustomerID;
        [DataMember]
        public string CustomerName;       
        [DataMember]
        public string Address;
        [DataMember]
        public string EmailId;
    }
Step 5: And write the following code in the SaleService.cs file: 

SaleService.cs page:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.ServiceModel.Web;
using System.Text;
using System.Collections.Generic;
using System.Configuration;
using System.Data;
using System.Data.SqlClient;

public class SaleService : ISaleService
    {
        public bool InsertCustomer(Customer obj)
        {
            cutomerList.Add(obj);
            return true;
        }

        public List<Customer> GetAllCustomer()
        {
            return cutomerList;
        }

        public bool DeleteCustomer(int Cid)
        {
            var item = cutomerList.First(x => x.CustomerID == Cid);

            cutomerList.Remove(item);
            return true;
        }

        public bool UpdateCustomer(Customer obj)
        {
            var list = cutomerList;
            cutomerList.Where(p => p.CustomerID ==
            obj.CustomerID).Update(p => p.CustomerName = obj.CustomerName);
            return true;
        }

      public static  List<Customer> cutomerList = new List<Customer>()
         {
        new Customer {CustomerID = 1, CustomerName="Sujeet",
        Address="Pune", EmailId="test@yahoo.com" },
        new Customer {CustomerID = 2, CustomerName="Rahul",
        Address="Pune", EmailId="test@yahoo.com" },
        new Customer {CustomerID = 3, CustomerName="Mayur",
        Address="Pune", EmailId="test@yahoo.com"}
         };

      
   

Step 6: Build your Class library.

Step 7: Add new Console Application in the same solution SaleServiceSelfHosting



Step 8: In this project Add reference of the WCF Class library project and Add reference of System.ReferenceModel Namespace.

Step 9: Add below code on Console Application
class Program
    {
        static void Main(string[] args)
        {

            Uri baseAddress = new Uri("http://localhost:6525/SaleService");

            using (ServiceHost host = new ServiceHost(typeof(SaleClassLibrary.SaleService), baseAddress))
            {
                ServiceMetadataBehavior smb = new ServiceMetadataBehavior();
                smb.HttpGetEnabled = true;
                smb.MetadataExporter.PolicyVersion = PolicyVersion.Policy15;
                host.Description.Behaviors.Add(smb);

                host.Open();

                Console.WriteLine("The service is ready at: {0}", baseAddress);
                Console.WriteLine("Press <Enter> to stop the service");
                Console.ReadKey();
                host.Close();

            }
        }
    }

Step 10: Build your Console application and set as startup project and run the project.

Your self-hosting WCF Service is now running 


WCF Self-Hosting

WCF come with ServiceHost class that makes you can host your services in your own application easily. With WCF what you need to do is just configuring your service endpoint and calling the .open() method of ServiceHost.



Advantages & Disadvantage
To be able to host WCF service you need WCF runtime and .NET application where you want to host your service.

Advantages:
   Easy: This way of hosting don’t need many lines of code to successfully run
   Flexible: You can easily decide when your service active by calling the .open() and .close() method.

Disadvantage
However, this self-hosting method has some disadvantage. This self-hosting service is reachable only when the application that hosts the service is running. So, self-hosting is only suitable when the development and demonstration phase of your system.

Happy Programming!!

If you have any query mail me to Sujeet.bhujbal@gmail.com     


Regards

Sujeet Bhujbal

------------------------------------------------------------------------------------------------------------
------------------------------------------------------------------------------------------------------------