Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.
Articles / Languages / C#

WCF Killer

4.92/5 (22 votes)
11 Feb 2011CPOL4 min read 76.2K   1.1K  
A simple high performance drop-in TCP protocol handler for client server communications with POCO objects and no messy proxy classes like WCF.

Image 1

Introduction

This is a piece of code that I have been using for a long time, which is simple to use, doesn't impose any additional work, and gets the job done for client server communications using the TCP protocol. You just include a single file in your project and get all the benefits.

Kudos to Yuen Chiu So for writing the original article found here (dotnettcp.aspx). The original article only handled strings; my article extends that to a level which you can use in your own client server application which must transfer the detailed objects of data. I have also tweaked and tested the code to get the maximum performance and stability out of it (see the performance test section).

Background

This code came about after using named pipe communication for a long time and having the following problems, although arguably named pipes are faster at data transfer:

  • Named pipes require authentication on connection to a computer, which is very limiting in non-domain computers and requires the user to enter a valid system user and password on the destination computer to even work.
  • The named pipe implementation I was using was very large, code wise.

So for the above reasons, I set about creating a replacement which would fit the following requirements:

  • Simple to use in code, with minimal configuration.
  • Multi-threaded so it can scale to hundreds of simultaneous users.
  • Does not impose additional code "proxies" and "contracts" and should work with normal serializable objects.
  • Does not require computer level authentication, which is implicit in using the TCP protocol as it is handled at a lower level than the OS (unlike named pipes).
  • Flexibility to implement your own authentication on top of it.
  • Handle large data transfer objects, e.g., 3 MB data packets.

The test application

testserver.png

Run the solution, and you will get a command line application like in the picture above. Press the 'S' key for server mode, 'C' key for client mode, and anything else for automatic mode which will start sending requests to the server. The application is hard coded to send and receive on the 127.0.0.1 IP address which is the local system; you can change this to test on a real network with different computers.

The data.cs file contains the sample POCO data object which is hard coded to a 3 MB size; you can change this to test smaller and larger data packets (see the performance test section).

Using the code

To use the code, first you must create your data packets like the ones below:

C#
[Serializable()]
public class Packet
{
    public Packet()
    {
        data = new byte[3*1024*1024]; // 3 mega byte
    }
    public byte[] data { get; set; }

    public string Message { get; set; }
    public string Username { get; set; }
    public string Password { get; set; }
    public Guid SessionGuid { get; set; }

    public new string ToString()
    {
        return Message;
    }
}

[Serializable()]
public class ReturnPacket
{
    public bool OK { get; set; }
    public string Message { get; set; }
}

As you can see, the data structures are normal classes and not proxies; you also must put the Serializble attribute on the class so the .NET serializer can work with it.

For the server code, you do the following:

C#
NetworkServer ns = new NetworkServer(99, new ProcessPayload(serverprocess));
// will listen on port 99

ns.Start();

private static object serverprocess(object data)
{
    Packet dp = data as Packet;
    if (dp != null)
        return HandlePacket(dp);
   
    Console.WriteLine("message not recognized");
    return new ReturnPacket();
}

private static object HandlePacket(Packet dp)
{
    ReturnPacket ret = new ReturnPacket();
    if (dp.SessionGuid == Guid.Empty)
    {
        // Authenticate username and password possibly with LDAP server
    }
    else
    {
        // check sessionguid valid -> if not return failed
    }
    ret.OK = true;
    ret.Message = "your msg : " + dp.Message + "\r\nreturn from server " + DateTime.Now;
    return ret;
}

As you can see, it is pretty straightforward and simple.

For the client side, you do the following:

C#
NetworkClient nc = new NetworkClient("127.0.0.1", 99);
// send to local host on port 99

nc.Connect();
Packet p = new Packet();
ReturnPacket ret = nc.Send(p) as ReturnPacket;

What is up to you

The following is up to you to implement yourself as you see fit for your own application:

  • Data Packet Definitions: what you want to send over the wire
  • Authentication: handling authentication is up to you if you need it
  • Session Management: related to authentication is session management

Performance tests

Here is the performance test results done on an AMD K625 1.5ghz, 4GB RAM, Windows 7 Home, win rating of 3.9 Notebook (CPU usage above 88%):

Number of simultaneous clientsData packet sizeChunk sizeRequest in 10 secs
5~12kb32kb12681
5~12kb16kb12291
5~12kb4kb11089
5~3mb4kb141
5~3mb16kb207
5~3mb32kb220

As you can see, the best performance is with a CHUNK_SIZE of 32 KB for both small and large data packets. You can go higher, e.g., 64 KB, but you get diminishing returns, and possibly (I don't know for sure), you may have problems with some switches and routers in the network as they might block large packet sizes.

Points of interest

I have put the network server code in a NETSERVER conditional compilation block so you will have to add that to your project definition. There is a Config class in the NetWorkClient.cs file which has some predefined options that I have tweaked to maximum performance; one note worthy option is NUM_OF_THREADS which is how many threads the server creates to handle a request. The default is 10, which should be enough for most applications; I have used it for handling 50+ clients applications in real world circumstances.

What you can do at home or further directions

Below is a list of possibilities you can do at home if you feel adventurous:

  • Progress Events: events to publish the progress of data for the client application to hook on to and display UI for.
  • Encrypted Communication: using SslStream instead of NetworkStream for encrypted data transfer.
  • LDAP Authentication of User: check the username and password embedded in the packet with an LDAP server (see the comments in the test app).
  • Session Management: handle client sessions and authenticated connections (see the comments in the test app).
  • Maybe replace BinarySerializer: for greater performance, you could look at the protocol serializer from Google which is incredibly fast and compact.

History

  • Initial release: Feb. 12, 2011.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)