TLSharp/TLSharp.Core/Network/TcpTransport.cs

185 lines
7 KiB
C#
Raw Normal View History

2015-09-28 04:01:17 +02:00
using System;
2020-04-05 13:33:18 +02:00
using System.Linq;
2015-09-28 04:01:17 +02:00
using System.Net;
using System.Net.Sockets;
using System.Threading;
2015-09-28 04:01:17 +02:00
using System.Threading.Tasks;
2020-04-05 13:33:18 +02:00
using TLSharp.Core.MTProto.Crypto;
2015-09-28 04:01:17 +02:00
namespace TLSharp.Core.Network
{
public delegate TcpClient TcpClientConnectionHandler(string address, int port);
2016-04-18 12:50:57 +02:00
public class TcpTransport : IDisposable
{
private readonly TcpClient tcpClient;
private readonly NetworkStream stream;
2016-04-18 12:50:57 +02:00
private int sendCounter = 0;
private CancellationTokenSource tokenSource = new CancellationTokenSource();
2016-04-18 12:50:57 +02:00
public TcpTransport(string address, int port, TcpClientConnectionHandler handler = null)
2016-04-18 12:50:57 +02:00
{
if (handler == null)
{
var ipAddress = IPAddress.Parse(address);
var endpoint = new IPEndPoint(ipAddress, port);
tcpClient = new TcpClient(ipAddress.AddressFamily);
try {
tcpClient.Connect (endpoint);
} catch (Exception ex) {
throw new Exception ($"Problem when trying to connect to {endpoint}; either there's no internet connection or the IP address version is not compatible (if the latter, consider using DataCenterIPVersion enum)",
ex);
}
}
else
tcpClient = handler(address, port);
if (tcpClient.Connected)
{
stream = tcpClient.GetStream();
}
2016-04-18 12:50:57 +02:00
}
public async Task Send(byte[] packet, CancellationToken token = default(CancellationToken))
2016-04-18 12:50:57 +02:00
{
if (!tcpClient.Connected)
2016-04-18 12:50:57 +02:00
throw new InvalidOperationException("Client not connected to server.");
var tcpMessage = new TcpMessage(sendCounter, packet);
await stream.WriteAsync(tcpMessage.Encode(), 0, tcpMessage.Encode().Length, token).ConfigureAwait(false);
2016-04-18 12:50:57 +02:00
sendCounter++;
}
public async Task<TcpMessage> Receive(CancellationToken token = default(CancellationToken))
2016-01-09 12:05:00 +01:00
{
var stream = tcpClient.GetStream();
2016-01-09 12:05:00 +01:00
var packetLengthBytes = new byte[4];
if (await stream.ReadAsync(packetLengthBytes, 0, 4, token).ConfigureAwait(false) != 4)
2016-01-09 12:05:00 +01:00
throw new InvalidOperationException("Couldn't read the packet length");
int packetLength = BitConverter.ToInt32(packetLengthBytes, 0);
2015-09-28 04:01:17 +02:00
var seqBytes = new byte[4];
if (await stream.ReadAsync(seqBytes, 0, 4) != 4)
throw new InvalidOperationException("Couldn't read the sequence");
int seq = BitConverter.ToInt32(seqBytes, 0);
int readBytes = 0;
var body = new byte[packetLength - 12];
int neededToRead = packetLength - 12;
do
{
var bodyByte = new byte[packetLength - 12];
var availableBytes = await stream.ReadAsync(bodyByte, 0, neededToRead);
neededToRead -= availableBytes;
Buffer.BlockCopy(bodyByte, 0, body, readBytes, availableBytes);
readBytes += availableBytes;
}
while (readBytes != packetLength - 12);
var crcBytes = new byte[4];
if (await stream.ReadAsync(crcBytes, 0, 4) != 4)
throw new InvalidOperationException("Couldn't read the crc");
int checksum = BitConverter.ToInt32(crcBytes, 0);
byte[] rv = new byte[packetLengthBytes.Length + seqBytes.Length + body.Length];
Buffer.BlockCopy(packetLengthBytes, 0, rv, 0, packetLengthBytes.Length);
Buffer.BlockCopy(seqBytes, 0, rv, packetLengthBytes.Length, seqBytes.Length);
Buffer.BlockCopy(body, 0, rv, packetLengthBytes.Length + seqBytes.Length, body.Length);
var crc32 = new Crc32();
var computedChecksum = crc32.ComputeHash(rv).Reverse();
if (!crcBytes.SequenceEqual(computedChecksum))
{
throw new InvalidOperationException("invalid checksum! skip");
}
return new TcpMessage(seq, body);
}
2020-04-26 18:29:26 +02:00
public async Task<TcpMessage> Receive(TimeSpan timeToWait)
{
var stream = tcpClient.GetStream();
var packetLengthBytes = new byte[4];
var token = tokenSource.Token;
2020-04-26 18:29:26 +02:00
stream.ReadTimeout = (int)timeToWait.TotalMilliseconds;
int bytes = 0;
try
{
bytes = stream.Read(packetLengthBytes, 0, 4);
2020-05-10 19:19:52 +02:00
}
catch (System.IO.IOException io)
{
var socketError = io.InnerException as SocketException;
if (socketError != null && socketError.SocketErrorCode == SocketError.TimedOut)
throw new OperationCanceledException();
2020-04-26 18:29:26 +02:00
throw;
}
if (bytes != 4)
throw new InvalidOperationException("Couldn't read the packet length");
int packetLength = BitConverter.ToInt32(packetLengthBytes, 0);
2016-01-09 12:05:00 +01:00
var seqBytes = new byte[4];
if (await stream.ReadAsync(seqBytes, 0, 4, token).ConfigureAwait(false) != 4)
2016-01-09 12:05:00 +01:00
throw new InvalidOperationException("Couldn't read the sequence");
int seq = BitConverter.ToInt32(seqBytes, 0);
2015-09-28 04:01:17 +02:00
2016-01-09 12:05:00 +01:00
int readBytes = 0;
var body = new byte[packetLength - 12];
int neededToRead = packetLength - 12;
do
{
var bodyByte = new byte[packetLength - 12];
var availableBytes = await stream.ReadAsync(bodyByte, 0, neededToRead, token).ConfigureAwait(false);
2016-01-09 12:05:00 +01:00
neededToRead -= availableBytes;
Buffer.BlockCopy(bodyByte, 0, body, readBytes, availableBytes);
readBytes += availableBytes;
}
while (readBytes != packetLength - 12);
var crcBytes = new byte[4];
if (await stream.ReadAsync(crcBytes, 0, 4, token).ConfigureAwait(false) != 4)
2016-01-09 12:05:00 +01:00
throw new InvalidOperationException("Couldn't read the crc");
byte[] rv = new byte[packetLengthBytes.Length + seqBytes.Length + body.Length];
Buffer.BlockCopy(packetLengthBytes, 0, rv, 0, packetLengthBytes.Length);
Buffer.BlockCopy(seqBytes, 0, rv, packetLengthBytes.Length, seqBytes.Length);
Buffer.BlockCopy(body, 0, rv, packetLengthBytes.Length + seqBytes.Length, body.Length);
2020-04-05 13:33:18 +02:00
var crc32 = new Crc32();
var computedChecksum = crc32.ComputeHash(rv).Reverse();
2016-01-09 12:05:00 +01:00
2020-04-05 13:33:18 +02:00
if (!crcBytes.SequenceEqual(computedChecksum))
2016-01-09 12:05:00 +01:00
{
throw new InvalidOperationException("invalid checksum! skip");
}
return new TcpMessage(seq, body);
2018-03-03 17:38:51 +01:00
}
public bool IsConnected
{
get
{
return this.tcpClient.Connected;
2018-03-03 17:38:51 +01:00
}
}
2016-04-18 12:50:57 +02:00
public void Dispose()
{
if (tcpClient.Connected)
{
stream.Close();
tcpClient.Close();
}
2018-03-03 17:38:51 +01:00
}
2016-04-18 12:50:57 +02:00
}
2015-09-28 04:01:17 +02:00
}