Getting Started with WCF (Windows Communication Foundation) - Running and Debugging Quickly
Overview
This post has an easy goal – to get you up to speed quickly and teaching you the core concepts of WCF. From this post you will be well primed to dive deep into more advanced topics, such as advanced bindings, channel layers, security, transactions, and more.
The web service that we create will be capable of supporting three types of clients. The first client will simply leverage http to send and receive data. The second client may need to take advantage of security and transactions. And the third type of client wants to leverage fast machine to machine connectivity using TCP. We will test for all the clients using the WCF Test Client.
In the next post we will explore hosting this project in IIS or in your own custom application. Then we might delve into moving it into the cloud.
Objectives
In this hands-on lab, you will learn how to:
- Create a WCF Service Library
- Add an interface and a service implementation
- Add 3 endpoints to support wsHttpBinding, basicHttpBinding, netTcpBinding
- How to configure App.config to define the service capabilities and behaviors
- How to debug a WCF Service using the WCF Test Client
Prerequisites
The following is required to complete this hands-on lab:
- Visual Studio 2010 to 2013 - (I will be using Visual Studio 2013)
- A desire to learn the fundamentals of web service programming on the MS stack
Setup
In order to execute the exercises in this hands-on lab you need to set up your environment.
- Begin by starting Visual Studio
Task 1 - Creating a WCF Service Library
This task is about creating the initial WCF project to which we will add functionality.
Start Visual Studio and select File/New Project.
Visual Studio 2013 RC - File New Project
Profile the name, FlipCaseService. It will be used to receive arguments as strings, and then modify them. More details of the app in future slides.
Naming the Service
You will start by deleting 2 files. The reason is to learn about adding classes. You could skip this step (and the next few).
Deleting the Default Classes
Ultimately, we need to add 3 files: (1) Data Contract (2) Interface for Service (3) Service Implementation. Right mouse click on the project and choose Add/New Item.
Adding 1 of 3 files - The Data Contract
Provide a name for what will become the definitions of Data Contracts. Name it DataContracts. Not the best name but just to drive home concepts of what that file contains.
Naming the Data Contracts
Right mouse click on the project and choose Add/New Item.
Adding 2 of 3 files - FlipCaseService
Provide a name for what will become the definitions of the service definition. Name it FlipCaseService.
Naming the WCF Service
Notice the files that we have added. Our next step will be to add code to them.
Solution Explorer
Task 2 - Defining the Data Contracts
A data contract is a formal agreement between a service and a client that abstractly describes the data to be exchanged. That is, to communicate, the client and the service do not have to share the same types, only the same data contracts. A data contract precisely defines, for each parameter or return type, what data is serialized (turned into XML) to be exchanged.
Note the members that make up StringData. OriginalString will be used to pass in data - a string whose case will be reversed, lowercase becoming uppercase, and uppercase becoming lowercase. The resulting string will be placed in the data member called FlippedCaseString. You will see the actual result in future section.
DataContracts.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.Text;
using System.Threading.Tasks;
namespace FlipCaseService
{
[DataContract]
public class StringData
{
[DataMember]
public string OriginalString;
[DataMember]
public string FlippedCaseString;
}
}
Note the implementtation provided with File/New Project.
FlipCaseService.cs - default implementation
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.Text;
namespace FlipCaseService
{
[ServiceContract]
public interface IFlipCaseService
{
[OperationContract]
StringData FlipTheCase(StringData sd);
}
}
Important Notes
An interface defines what must a client know about a class in order to benefit from using it
Developers often put the interfaces into a separate assembly.
This is useful because the interface can be used by any piece of code that needs to know about the interface, but not necessarily about the implementation.
It is common practice to have one class implement more than one interface.
What is great about this is that you can expose the same implemented class in different ways using different interfaces in the WCF endpoints.
In Windows Communication Foundation (WCF) applications, you define the operations by creating a class and marking it with the [ServiceContract] attribute in the interface file. For each method in the class you can mark each method with OperationAttribute.
Interface-based programming is popular among many developers
It makes code better, by increasing reusability, maintainability, and extensibility
Interfaces became popular in languages such as C++ and Smalltalk
The argument is that using distinct interfaces could make software, especially large code-bases, easier to maintain and extend
Interfaces simplify code reuse in object-oriented programming
Interfaces allow you extend the class without breaking the client
If a client does not communicate with an object through an interface, then the client can build inflexible dependencies on a class definition
Although interfaces are not ALWAYS appropriate, many believe that not using interfaces leads to impossibly difficult code improvement better over time
More dependencies are created in the client when a programmer writes code using a class's methods and properties instead of an interface.
Read more:
https://msdn.microsoft.com/en-us/library/aa260635(v=vs.60).aspx
Notice the new implementation, which will flip the case of the string passed in.
FlipCaseService.cs - New implementation
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.Text;
namespace FlipCaseService
{
public class FlipCaseService : IFlipCaseService
{
public StringData FlipTheCase(StringData sd)
{
sd.FlippedCaseString = null;
foreach (char c in sd.OriginalString)
{
sd.FlippedCaseString += char.IsLower(c) ?
c.ToString().ToUpper() : c.ToString().ToLower();
}
return sd;
}
}
}
Task 3 - Implementing Endpoints
All communication with a Windows Communication Foundation (WCF) service occurs through the endpoints of the service. Endpoints provide clients access to the functionality offered by a WCF service.
Each endpoint consists of four properties: - An address that indicates where the endpoint can be found. - A binding that specifies how a client can communicate with the endpoint. - A contract that identifies the operations available. - A set of behaviors that specify local implementation details of the endpoint.
Notice the client can access the service through different endpoints. A WCF client connects to a WCF service by communicating with an Endpoint. At each endpoint is a service contract. Also, each endpoint has an address (which is a URL specifying where the endpoint can be accessed). The endpoint also specifies binding properties which define how the data will be transferred.
Client accessing 3 endpoints
The mnemonic "ABC" is useful to know. It can be used to represent Address / Binding / Contract. The Binding indicates what communication protocols are used to access the service, what type of security mechanisms are to be used. WCF supports predefined bindings for most commonly used communication protocols such as SOAP over HTTP, SOAP over TCP, and SOAP over Message Queues. Interaction between WCF endpoint and client is typically accomplished using a SOAP envelope. SOAP envelopes are defined through XML. An important note when clients connect to services though specific endpoints. When connecting to endpoints, the client not only needs to know the contract, but it also has to adhere to the binding specified by the endpoint. Thus, both client and server must be in sync about their endpoints. Notice endpoint 1 uses wsHttpBinding and endpoint 2 uses basicHttpBinding.
ABC's of Endpoints
The definition of endpoints takes place in the App.config file. You can see the ABC's directly in the endpoint definitions.
App.config | |
1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 |
<?xml version="1.0" encoding="utf-8"?>
<configuration> <appSettings> <add key="aspnet:UseTaskFriendlySynchronizationContext" value="true"/> <!--<add key="ClientSettingsProvider.ServiceUri" value="" />--> </appSettings> <system.web> <compilation debug="true"/> <membership defaultProvider="ClientAuthenticationMembershipProvider"> <providers> <add name="ClientAuthenticationMembershipProvider" type="System.Web.ClientServices.Providers.ClientFormsAuthenticationMembershipProvider, System.Web.Extensions, Version=4.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35" serviceUri=""/> </providers> </membership> <roleManager defaultProvider="ClientRoleProvider" enabled="true"> <providers> <add name="ClientRoleProvider" type="System.Web.ClientServices.Providers.ClientRoleProvider, System.Web.Extensions, Version=4.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35" serviceUri="" cacheTimeout="86400"/> </providers> </roleManager> </system.web> <!--When deploying the service library project, the content of the config file must be added to the host's app.config file. System.Configuration does not support config files for libraries. --> <system.serviceModel> <client/> <services> <service behaviorConfiguration="ServiceBehaviorMetaData" name="FlipCaseService.FlipCaseService"> <endpoint address="flipcase/wsAddress" binding="wsHttpBinding" contract="FlipCaseService.IFlipCaseService" bindingConfiguration="wsHttp"> <identity> <dns value="localhost"/> </identity> </endpoint> <endpoint address="flipcase/basic" binding="basicHttpBinding" contract="FlipCaseService.IFlipCaseService"> <identity> <dns value="localhost"/> </identity> </endpoint> <endpoint address="net.tcp://localhost/FlipCaseNetTcp" binding="netTcpBinding" contract="FlipCaseService.IFlipCaseService"> <identity> <dns value="localhost"/> </identity> </endpoint> <endpoint address="flipcase/mex" binding="mexHttpBinding" contract="IMetadataExchange"> </endpoint> <host> <baseAddresses> <add baseAddress="https://localhost:8080/"/> </baseAddresses> </host> </service> </services> <bindings> <wsHttpBinding> <binding name="wsHttp"> <security mode="None"/> </binding> </wsHttpBinding> </bindings> <behaviors> <serviceBehaviors> <behavior name="FlipCaseService.Service1Behavior"> <serviceMetadata httpGetEnabled="true" httpsGetEnabled="true"/> <serviceDebug includeExceptionDetailInFaults="true"/> </behavior> <behavior name="ServiceBehaviorMetaData"> <serviceMetadata httpGetEnabled="true" httpGetUrl="https://localhost:8080/flipcase/metadata"/> </behavior> </serviceBehaviors> </behaviors> </system.serviceModel> </configuration> |
Task 4 - Some important information about WCF
This next section addresses basic information all WCF devs should know.
The next step is to call into the WCF Service. But before the WCF Service can be called into, it needs to be hosted somewhere.
Hosting WCF
Note A WCF service can be hosted in 3 places. (1) Self-hosting in any managed .NET application (2) Hosting in a Windows service (3) Hosting in different versions of IIS.
Hosting a service in a managed application is the most flexible option because it requires the least infrastructure to deploy. However, it is also the least robust hosting option, because managed applications do not provide the advanced hosting and management features of other hosting options in WCF, such as Internet Information Services (IIS) and Windows services.
Hosting in a Windows Service This scenario is enabled by the managed Windows service hosting option that is a long-running WCF service hosted outside of Internet Information Services (IIS) in a secure environment that is not message activated. The lifetime of the service is controlled instead by the operating system. This hosting option is available in all versions of Windows. - Windows services can be managed with the Microsoft.ManagementConsole.SnapIn in Microsoft Management Console (MMC) and can be configured to start up automatically when the system boots up. This hosting option consists of registering the application domain (AppDomain) that hosts a WCF service as a managed Windows service so that the process lifetime of the service is controlled by the Service Control Manager (SCM) for Windows services.
A WCF service that runs in the IIS environment takes full advantage of IIS features, such as process recycling, idle shutdown, process health monitoring, and message-based activation. This hosting option requires that IIS be properly configured, but it does not require that any hosting code be written as part of the application. You can use IIS hosting only with an HTTP transport.
Every process has at least one application domain, and each application domain can host zero or more WCF ServiceHost instances. WCF requires at least an application domain hosted inside a Windows process.
Processes, Application Domains, ServiceHosts
Task 5 - Running and Debugging Your Code
Hit F5 or choose from the Debug menu.
Starting the debugging process
After running the WCF Service Library, the WCF Test Client will appear. When starting debugging (F5), it will host the service in the WCF Service Host by default. And the WCF Test Client will automatically start for you to test the service. The WCF Test Client is a debugging tool to test WCF service. It is very useful to help developing WCF service. In the next step, we will be able to to debug our running service. You can see that it is configured in your project automatically. See Project Properties -> Debug -> Command line arguments.
The WCF Test Client
Task 6 - Verifying everything works
Notice the 4 steps. On the left pane, you can see the 3 FlipTheCase methods, each representing differnt ABC's, except for the contract, which is shared among all 3 endpoints. Notice in step 2 Bruno in the Original String. In step 3 I hit Invoke, which triggers the code in the WCF service to execute, flipping the case. Notice in step 4 that FlippedCaseString is now flipped (bRUNO) and represents the final result.
The WCF Test Client
Note:
There is a request and a response object.
The client will be able to read "bRUNO" in the FlippedCaseString attribute.
Summary
In this hands-on lab, you will learned how to:
- Create a WCF Service Library
- How to use DataContracts to exchange data between client and WCF Service
- Add an interface and a service implementation
- Add 3 endpoints to support wsHttpBinding, basicHttpBinding, netTcpBinding
- How to configure App.config to define the service capabilities and behaviors
- How to debug a WCF Service using the WCF Test Client
- Further reading at Aaron Skonnard's article (https://msdn.microsoft.com/en-us/magazine/cc163647.aspx)
Comments
Anonymous
October 21, 2013
Where is: blogs.msdn.com/.../how-to-instantly-search-through-gbs-of-code-and-office-documents-in-the-cloud.aspx I need such an indexed data store somewhere and wanted to follow up.Anonymous
November 26, 2014
Thanks very much. Your code was my solution. I have few questions. 1) why do you include mexHttpBinding? 2) why do you include FlipCaseService.Service1Behavior?Anonymous
February 20, 2015
Thank you for this post... I got this error [ Inconsistent accessibility: parameter type 'FlipCaseService1.StringData' is less accessible than method 'FlipCaseService1.FlipCaseService.FlipTheCase(FlipCaseService1.StringData)' ] [ Inconsistent accessibility: return type 'FlipCaseService1.StringData' is less accessible than method 'FlipCaseService1.FlipCaseService.FlipTheCase(FlipCaseService1.StringData)' ]Anonymous
March 07, 2015
Linvex StringData class should be defined public to resolve the issue.