You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

241 lines
6.0 KiB

using PUtility;
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
namespace PComm
{
public class PClient
{
private readonly int BUFF_SIZE = 1000 * 50; // 50kB
private readonly int LARGE_DATA_SIZE_THRESHOLD = 1000 * 50; // 50kB
public delegate void ClientReceivedHandler(PClient sender, PDataType dataType, byte[] data);
public event ClientReceivedHandler OnReceived;
public delegate void ClientDisconnectedHandler(PClient sender);
public event ClientDisconnectedHandler Disconnected;
public string ID { get; set; }
public IPEndPoint EndPoint { get; private set; }
public bool Connected { get { return socket.Connected; } }
public bool DataCompression { get; set; } = true;
private Socket socket;
public PClient(string ip, int port, string id = null)
{
if (string.IsNullOrEmpty(id))
id = Guid.NewGuid().ToString();
socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
this.ID = id;
this.EndPoint = new IPEndPoint(IPAddress.Parse(ip), port);
Connect();
}
public PClient(Socket accepted)
{
if (string.IsNullOrEmpty(this.ID))
this.ID = Guid.NewGuid().ToString();
socket = accepted;
this.EndPoint = (IPEndPoint)socket.RemoteEndPoint;
socket.BeginReceive(new byte[] { 0 }, 0, 0, 0, AcceptCallback, null);
}
public void Connect()
{
if (socket == null || socket.Connected)
return;
socket.Connect(this.EndPoint);
// send ID
this.Send(PDataType.ClientID, Encoding.UTF8.GetBytes(this.ID));
socket.BeginReceive(new byte[] { 0 }, 0, 0, 0, AcceptCallback, null);
}
private int SendToSocket(byte[] buffer)
{
try
{
if (!socket.Connected)
Connect();
return socket.Send(buffer, SocketFlags.None);
}
catch (Exception ex)
{
Debug.WriteLine($"[ERROR] Socket send: {ex.Message}");
PFileManager.Instance.WriteLog($"[ERROR] Socket send: {ex.Message}");
if (CheckSocketConnection(socket))
{
socket.BeginReceive(new byte[] { 0 }, 0, 0, 0, AcceptCallback, null);
}
else
{
Debug.WriteLine($"[ERROR] Socket close {socket.RemoteEndPoint.ToString()}");
PFileManager.Instance.WriteLog($"[ERROR] Socket close {socket.RemoteEndPoint.ToString()}");
Close();
if (Disconnected != null)
Disconnected(this);
}
return 0;
}
}
private int SendByProtocol(PDataType dataType, byte[] data)
{
byte[] type = BitConverter.GetBytes((int)dataType);
SendToSocket(type);
byte[] isCompressed;
byte[] toSendData;
// 데이터 크기 100B 이하 또는 50MB 이상 압축하지 않음 (CPU 부하시간 더 큼)
// 압축한 경우 1 전송, 아닌경우 0 전송
if (DataCompression == false || data.Length <= 100 || data.Length >= 50000000)
{
isCompressed = BitConverter.GetBytes(false);
toSendData = data;
}
else
{
isCompressed = BitConverter.GetBytes(true);
toSendData = PUtil.CompressBytes(data, CompressionLevel.Optimal);
}
SendToSocket(isCompressed);
byte[] length = BitConverter.GetBytes(toSendData.Length);
SendToSocket(length);
return SendToSocket(toSendData);
}
public int Send(string msg)
{
byte[] data = Encoding.UTF8.GetBytes(msg);
return Send(PDataType.SimpleString, data);
}
public int Send(PDataType dataType, byte[] data)
{
return SendByProtocol(dataType, data);
}
private void AcceptCallback(IAsyncResult ar)
{
try
{
socket.EndReceive(ar);
// get data type
byte[] dataTypeBuff = new byte[4];
socket.Receive(dataTypeBuff, dataTypeBuff.Length, SocketFlags.None);
PDataType dataType = (PDataType)BitConverter.ToInt32(dataTypeBuff, 0);
// check compressed
byte[] isCompressedBuff = new byte[1];
socket.Receive(isCompressedBuff, isCompressedBuff.Length, SocketFlags.None);
bool isCompressed = BitConverter.ToBoolean(isCompressedBuff, 0);
// get data size
byte[] sizeBuff = new byte[4];
socket.Receive(sizeBuff, sizeBuff.Length, SocketFlags.None);
int dataSize = BitConverter.ToInt32(sizeBuff, 0);
// if data size is large, ready to receive data;
if (dataSize >= LARGE_DATA_SIZE_THRESHOLD)
{
int factor = dataSize / LARGE_DATA_SIZE_THRESHOLD;
int gain = factor > 1000 ? 1000 : factor;
Thread.Sleep(100 * gain);
}
using (MemoryStream ms = new MemoryStream())
{
while (dataSize > 0)
{
byte[] buff;
if (dataSize < BUFF_SIZE)
buff = new byte[dataSize];
else
buff = new byte[BUFF_SIZE];
int receiveSize = socket.Receive(buff, buff.Length, SocketFlags.None);
ms.Write(buff, 0, buff.Length);
dataSize -= receiveSize;
}
byte[] receivedData = ms.ToArray();
byte[] data;
if (isCompressed)
data = PUtil.DecompressBytes(receivedData);
else
data = receivedData;
if (OnReceived != null)
OnReceived(this, dataType, data);
socket.BeginReceive(new byte[] { 0 }, 0, 0, 0, AcceptCallback, null);
}
}
catch (Exception ex)
{
Debug.WriteLine($"[ERROR] Socket receive: {ex.Message}");
PFileManager.Instance.WriteLog($"[ERROR] Socket receive: {ex.Message}");
if (CheckSocketConnection(socket))
{
socket.BeginReceive(new byte[] { 0 }, 0, 0, 0, AcceptCallback, null);
}
else
{
Debug.WriteLine($"[ERROR] Socket close {socket.RemoteEndPoint.ToString()}");
PFileManager.Instance.WriteLog($"[ERROR] Socket close {socket.RemoteEndPoint.ToString()}");
Close();
if (Disconnected != null)
Disconnected(this);
}
}
}
private bool CheckSocketConnection(Socket socket)
{
bool availability = socket.Available == 0;
bool poll = socket.Poll(1000, SelectMode.SelectRead);
if (availability && poll)
return false;
else
return true;
}
public void Close()
{
socket.Close();
socket.Dispose();
}
}
}