using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Net.Sockets;
using System.Net;
namespace TCPService
{
class PolicyServer
{
Socket m_listenerIPv4;
Socket m_listenerIPv6;
byte[] m_policy;
// pass in the path of an XML file containing the socket policy
public PolicyServer(string policyContents)
{
m_policy = Encoding.UTF8.GetBytes(policyContents);
// Create the Listening Sockets
m_listenerIPv4 = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
m_listenerIPv4.Bind(new IPEndPoint(IPAddress.Any, 943));
m_listenerIPv4.Listen(10);
m_listenerIPv4.BeginAccept(new AsyncCallback(OnConnection), m_listenerIPv4);
//Console.WriteLine("Listenting on IPv4 port 943.");
if (System.Net.Sockets.Socket.OSSupportsIPv6)
{
m_listenerIPv6 = new Socket(AddressFamily.InterNetworkV6, SocketType.Stream, ProtocolType.Tcp);
m_listenerIPv6.Bind(new IPEndPoint(IPAddress.IPv6Any, 943));
m_listenerIPv6.Listen(10);
m_listenerIPv6.BeginAccept(new AsyncCallback(OnConnection), m_listenerIPv6);
//Console.WriteLine("Listenting on IPv6 port 943.");
}
else
{
// Console.WriteLine("IPv6 is not supported by the system.");
}
}
// Called when we receive a connection from a client
public void OnConnection(IAsyncResult res)
{
Socket listener = (Socket)res.AsyncState;
Socket client = null;
try
{
client = listener.EndAccept(res);
}
catch (SocketException)
{
return;
}
// handle this policy request with a PolicyConnection
PolicyConnection pc = new PolicyConnection(client, m_policy);
// look for more connections
listener.BeginAccept(new AsyncCallback(OnConnection), listener);
}
public void Close()
{
m_listenerIPv4.Close();
if (m_listenerIPv6 != null)
{
m_listenerIPv6.Close();
}
}
}
class PolicyConnection
{
Socket m_connection;
// buffer to receive the request from the client
byte[] m_buffer;
int m_received;
// the policy to return to the client
byte[] m_policy;
// the request that we're expecting from the client
static string s_policyRequestString = "<policy-file-request/>";
public PolicyConnection(Socket client, byte[] policy)
{
m_connection = client;
m_policy = policy;
m_buffer = new byte[s_policyRequestString.Length];
m_received = 0;
try
{
// receive the request from the client
m_connection.BeginReceive(m_buffer, 0, s_policyRequestString.Length, SocketFlags.None, new AsyncCallback(OnReceive), null);
}
catch (SocketException)
{
m_connection.Close();
}
}
// Called when we receive data from the client
private void OnReceive(IAsyncResult res)
{
try
{
m_received += m_connection.EndReceive(res);
// if we haven't gotten enough for a full request yet, receive again
if (m_received < s_policyRequestString.Length)
{
m_connection.BeginReceive(m_buffer, m_received, s_policyRequestString.Length - m_received, SocketFlags.None, new AsyncCallback(OnReceive), null);
return;
}
// make sure the request is valid
string request = System.Text.Encoding.UTF8.GetString(m_buffer, 0, m_received);
if (StringComparer.InvariantCultureIgnoreCase.Compare(request, s_policyRequestString) != 0)
{
m_connection.Close();
return;
}
// send the policy
m_connection.BeginSend(m_policy, 0, m_policy.Length, SocketFlags.None, new AsyncCallback(OnSend), null);
}
catch (SocketException)
{
m_connection.Close();
}
}
// called after sending the policy to the client; close the connection.
public void OnSend(IAsyncResult res)
{
try
{
m_connection.EndSend(res);
}
finally
{
m_connection.Close();
}
}
}
//跨域用的xml文件,以代码的方式传入。
public static class SocketPolicy
{
public const string Policy = @"
<xml version=""1.0"" encoding =""utf-8"">
<access-policy>
<cross-domain-access>
<policy>
<allow-from>
<domain uri=""*"" />
</allow-from>
<grant-to>
<socket-resource port=""4502-4530"" protocol=""tcp"" />
</grant-to>
</policy>
</cross-domain-access>
</access-policy>
";
}
}