利用了WCF的Socket实现在线聊天室的服务器端

 
文/monkeyboyabc  出处/博客园

1.建立WCF Service,命名为ChatService
添加引用,

 
CODE:
System.ServiceModel.dll
System.ServiceModel.PollingDuplex.dll
System.Runtime.Serialization.dll

2.定义接口IChatService.cs
CODE:
    [ServiceContract(Namespace = “Silverlight”,CallbackContract = typeof(IChatCallback),SessionMode = SessionMode.Required)]
    public interface IChat
    {

        [OperationContract(IsOneWay = true)]
        void Order(Message receivedMessage);

        [OperationContract(IsOneWay = true)]
        void JoinChat(Message receivedMessage);

        [OperationContract(IsOneWay = true)]
        void SayChat(Message receivedMessage);
    }

相对应的callback接口

 
CODE:
    [ServiceContract]
    public interface IChatCallback
    {
        [OperationContract(IsOneWay = true)]
        void Receive(Message returnMessage);
    }

注意:在IChat中的属性 NameSpace一定要是SiliverLight,且在SessionMode中 要标明 SessionMode.Required 否则会出错

3.在Service文件中 定义服务
在文件ChatService.cs中添加以下代码:

 
CODE:
static Dictionary<string, IChatCallback> chatClients = new Dictionary<string, IChatCallback>();
        static List<string> m_NameList = new List<string>();
        static int guestIndex = 1;

        public void JoinChat(Message receivedMessage)
        {
            IChatCallback chatclient;
            bool userAdded = false;
            string test;
            Message returnMessage;

            string name = receivedMessage.GetBody<string>();

            chatclient = OperationContext.Current.GetCallbackChannel<IChatCallback>();

            test = “Service:JoiningPlease Wait”;
            returnMessage = Message.CreateMessage(MessageVersion.Soap11,
                “Silverlight/IChat/Receive”, test);

            chatclient.Receive(returnMessage);

            lock (syncObj)
            {
                if (name == “Guest”)
                {
                    name = “Guest” + guestIndex.ToString();
                    guestIndex = guestIndex + 1;
                }
                if (!chatClients.ContainsKey(name) && name != “” && name != null)
                {
                    test = “Service:Begin Add User To ChattersList”;
                    returnMessage = Message.CreateMessage(MessageVersion.Soap11,
                        “Silverlight/IChat/Receive”, test);

                    chatclient.Receive(returnMessage);

                    this.name = name;
                    m_NameList.Add(name);
                    chatClients.Add(name, chatclient);
                    userAdded = true;
                                     
                }
                else
                {
                    test = “Service:User:” + name + “Join fault.Already has this user or User name is NULL”;

                    returnMessage = Message.CreateMessage(MessageVersion.Soap11,
                        “Silverlight/IChat/Receive”,test );

                    chatclient.Receive(returnMessage);

                    return;
                }
            }

            if (userAdded)
            {
                string text2 = “Service:User:” + name + ” is already Joined”;

                SendMessageToClient(text2);
            }
        }

这个方法是对应接口中的JoinChat,将用户添加到会话列表中。
其中,调用Receive接口将Message返回
然后,遍利用户字典,将信息发送给所有用户

 
CODE:
        public void SayChat(Message receivedMessage)
        {
            IChatCallback chatclient = OperationContext.Current.GetCallbackChannel<IChatCallback>();
            string text = receivedMessage.GetBody<string>();

            if(!chatClients.ContainsValue(chatclient))
            {
                return;
            }

            //TODO: 用更好的方法实现
            for (int i = 0; i < m_NameList.Count; i++)
            {
                IChatCallback clientInList = chatClients[m_NameList];

                if (clientInList.Equals(chatclient))
                {
                    text = m_NameList + ” Says: ” + text;
                    break;
                }
            }

            SendMessageToClient(text);
        }

        private void SendMessageToClient(string test)
        {
            //TODO: 用更好的方法实现
            for (int i = 0; i < m_NameList.Count; i++)
            {
                IChatCallback chatclient = chatClients[m_NameList];

                Message returnMessage = Message.CreateMessage(MessageVersion.Soap11,
                    “Silverlight/IChat/Receive”, test);

                chatclient.Receive(returnMessage);
            }
        }

4.添加宿主文件PollingDuplexServiceHostFactory.cs

 
CODE:
using System;
using System.ServiceModel;
using System.ServiceModel.Activation;
using System.ServiceModel.Channels;

namespace SLChat
{
    public class PollingDuplexServiceHostFactory : ServiceHostFactoryBase
    {
        public override ServiceHostBase CreateServiceHost(string constructorString,
            Uri[] baseAddresses)
        {
            //return new PollingDuplexSimplexServiceHost(baseAddresses);

            PollingDuplexSimplexServiceHost serviceHost = new PollingDuplexSimplexServiceHost(baseAddresses);

            System.ServiceModel.Description.ServiceMetadataBehavior smb = serviceHost.Description.Behaviors.Find<System.ServiceModel.Description.ServiceMetadataBehavior>();
            if (smb != null)
            {
                smb.HttpGetEnabled = true;
               
            }
            else
            {
                smb = new System.ServiceModel.Description.ServiceMetadataBehavior();
                smb.HttpGetEnabled = true;
                serviceHost.Description.Behaviors.Add(smb);
            }

            return serviceHost;
        }
    }

    class PollingDuplexSimplexServiceHost : ServiceHost
    {
        public PollingDuplexSimplexServiceHost(params System.Uri[] addresses)
        {
            base.InitializeDescription(typeof(ChatService), new UriSchemeKeyedCollection(addresses));
        }

        protected override void InitializeRuntime()
        {
            // Define the binding and set time-outs.
            PollingDuplexBindingElement pdbe = new PollingDuplexBindingElement()
            {
                PollTimeout = TimeSpan.FromSeconds(3),
                InactivityTimeout = TimeSpan.FromMinutes(1)
            };

            // Add an endpoint for the given service contract.
            this.AddServiceEndpoint(
                typeof(IChat),
                new CustomBinding(
                    pdbe,
                    new TextMessageEncodingBindingElement(
                    MessageVersion.Soap11,
                    System.Text.Encoding.UTF8),
                    new HttpTransportBindingElement()),
                    “”);

            base.InitializeRuntime();
        }
    }
}

5.一定不要忘记了 添加一个跨域访问的文件**(折磨我尽一天的问题。。。)
clientaccesspolicy.xml

 
CODE:
<?xml version=”1.0″ encoding=”utf-8″?>
<access-policy>
  <cross-domain-access>
    <policy>
      <allow-from http-request-headers=”*”>
        <domain uri=”*”/>
      </allow-from>
      <grant-to>
        <resource path=”/” include-subpaths=”true”/>
      </grant-to>
    </policy>
  </cross-domain-access>
</access-policy>

没有这个文件,在Windows自带的IIS中将发布不了,但是在VS2008的IIS是没有问题的

6.在VS的资源管理器(Solution Explore)右键点击ChatService.svc,点击View MarkUp
将原来的代码修改为:

 
CODE:
<%@ServiceHost language=c# Debug=”true” Service=”SLChat.ChatService” Factory=”SLChat.PollingDuplexServiceHostFactory”%>

使Service用我自己写的Host发布 而不用配置文件