2015-09-28 04:01:17 +02:00
|
|
|
|
using System;
|
2015-10-14 18:16:27 +02:00
|
|
|
|
using System.Collections.Generic;
|
2015-09-28 04:01:17 +02:00
|
|
|
|
using System.Linq;
|
2016-01-17 10:16:44 +01:00
|
|
|
|
using System.Text;
|
2015-09-28 04:01:17 +02:00
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
using TLSharp.Core.Auth;
|
|
|
|
|
|
using TLSharp.Core.MTProto;
|
|
|
|
|
|
using TLSharp.Core.MTProto.Crypto;
|
|
|
|
|
|
using TLSharp.Core.Network;
|
|
|
|
|
|
using TLSharp.Core.Requests;
|
|
|
|
|
|
|
|
|
|
|
|
namespace TLSharp.Core
|
|
|
|
|
|
{
|
|
|
|
|
|
public class TelegramClient
|
|
|
|
|
|
{
|
|
|
|
|
|
private MtProtoSender _sender;
|
|
|
|
|
|
private AuthKey _key;
|
|
|
|
|
|
private readonly TcpTransport _transport;
|
2015-10-01 14:55:02 +02:00
|
|
|
|
private string _apiHash = "a2514f96431a228e4b9ee473f6c51945";
|
|
|
|
|
|
private int _apiId = 19474;
|
2015-09-28 04:01:17 +02:00
|
|
|
|
private Session _session;
|
|
|
|
|
|
|
2015-10-14 18:16:27 +02:00
|
|
|
|
public User loggedUser { get { return _session.User; } }
|
|
|
|
|
|
|
|
|
|
|
|
public List<Chat> chats;
|
|
|
|
|
|
public List<User> users;
|
|
|
|
|
|
|
|
|
|
|
|
public TelegramClient(ISessionStore store, string sessionUserId)
|
2015-09-28 04:01:17 +02:00
|
|
|
|
{
|
2015-10-01 14:55:02 +02:00
|
|
|
|
if (_apiId == 0)
|
|
|
|
|
|
throw new InvalidOperationException("Your API_ID is invalid. Do a configuration first https://github.com/sochix/TLSharp#quick-configuration");
|
|
|
|
|
|
|
|
|
|
|
|
if (string.IsNullOrEmpty(_apiHash))
|
|
|
|
|
|
throw new InvalidOperationException("Your API_ID is invalid. Do a configuration first https://github.com/sochix/TLSharp#quick-configuration");
|
|
|
|
|
|
|
2015-09-28 04:01:17 +02:00
|
|
|
|
_transport = new TcpTransport();
|
|
|
|
|
|
_session = Session.TryLoadOrCreateNew(store, sessionUserId);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<bool> Connect()
|
|
|
|
|
|
{
|
|
|
|
|
|
if (_session.AuthKey == null)
|
|
|
|
|
|
{
|
|
|
|
|
|
var result = await Authenticator.DoAuthentication(_transport);
|
|
|
|
|
|
_session.AuthKey = result.AuthKey;
|
|
|
|
|
|
_session.TimeOffset = result.TimeOffset;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_sender = new MtProtoSender(_transport, _session);
|
|
|
|
|
|
|
|
|
|
|
|
var request = new InitConnectionRequest(_apiId);
|
|
|
|
|
|
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public bool IsUserAuthorized()
|
|
|
|
|
|
{
|
|
|
|
|
|
return _session.User != null;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<bool> IsPhoneRegistered(string phoneNumber)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (_sender == null)
|
|
|
|
|
|
throw new InvalidOperationException("Not connected!");
|
|
|
|
|
|
|
|
|
|
|
|
var authCheckPhoneRequest = new AuthCheckPhoneRequest(phoneNumber);
|
|
|
|
|
|
await _sender.Send(authCheckPhoneRequest);
|
|
|
|
|
|
await _sender.Recieve(authCheckPhoneRequest);
|
|
|
|
|
|
|
|
|
|
|
|
return authCheckPhoneRequest._phoneRegistered;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<string> SendCodeRequest(string phoneNumber)
|
|
|
|
|
|
{
|
|
|
|
|
|
var request = new AuthSendCodeRequest(phoneNumber, 5, _apiId, _apiHash, "en");
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
|
|
|
|
|
|
return request._phoneCodeHash;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<User> MakeAuth(string phoneNumber, string phoneHash, string code)
|
|
|
|
|
|
{
|
|
|
|
|
|
var request = new AuthSignInRequest(phoneNumber, phoneHash, code);
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
|
|
|
|
|
|
_session.SessionExpires = request.SessionExpires;
|
|
|
|
|
|
_session.User = request.user;
|
|
|
|
|
|
|
|
|
|
|
|
_session.Save();
|
|
|
|
|
|
|
|
|
|
|
|
return request.user;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2016-01-17 10:16:44 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<InputFile> UploadFile(string name, byte[] data)
|
|
|
|
|
|
{
|
|
|
|
|
|
var partSize = 65536;
|
|
|
|
|
|
|
|
|
|
|
|
var file_id = DateTime.Now.Ticks;
|
|
|
|
|
|
|
|
|
|
|
|
var partedData = new Dictionary<int, byte[]>();
|
|
|
|
|
|
var parts = Convert.ToInt32(Math.Ceiling(Convert.ToDouble(data.Length) / Convert.ToDouble(partSize)));
|
|
|
|
|
|
var remainBytes = data.Length;
|
|
|
|
|
|
for (int i = 0; i < parts; i++)
|
|
|
|
|
|
{
|
|
|
|
|
|
partedData.Add(i, data
|
|
|
|
|
|
.Skip(i * partSize)
|
|
|
|
|
|
.Take(remainBytes < partSize ? remainBytes : partSize)
|
|
|
|
|
|
.ToArray());
|
|
|
|
|
|
|
|
|
|
|
|
remainBytes -= partSize;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for(int i =0;i<parts;i++)
|
|
|
|
|
|
{
|
|
|
|
|
|
var saveFilePartRequest = new Upload_SaveFilePartRequest(file_id, i, partedData[i]);
|
|
|
|
|
|
await _sender.Send(saveFilePartRequest);
|
|
|
|
|
|
await _sender.Recieve(saveFilePartRequest);
|
|
|
|
|
|
|
|
|
|
|
|
if (saveFilePartRequest.Done == false)
|
|
|
|
|
|
throw new InvalidOperationException($"File part {i} does not uploaded");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string md5_checksum;
|
|
|
|
|
|
using (var md5 = System.Security.Cryptography.MD5.Create())
|
|
|
|
|
|
{
|
|
|
|
|
|
var hash = md5.ComputeHash(data);
|
|
|
|
|
|
var hashResult = new StringBuilder(hash.Length * 2);
|
|
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < hash.Length; i++)
|
|
|
|
|
|
hashResult.Append(hash[i].ToString("x2"));
|
|
|
|
|
|
|
|
|
|
|
|
md5_checksum = hashResult.ToString();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var inputFile = new InputFileConstructor(file_id, parts, name, md5_checksum);
|
|
|
|
|
|
|
|
|
|
|
|
return inputFile;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<messages_StatedMessage> SendMediaMessage(InputPeer inputPeer, InputMedia inputMedia)
|
|
|
|
|
|
{
|
|
|
|
|
|
var request = new Message_SendMediaRequest(inputPeer, inputMedia);
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
|
|
|
|
|
|
return request.StatedMessage;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2015-09-28 04:01:17 +02:00
|
|
|
|
public async Task<int?> ImportContact(string phoneNumber)
|
|
|
|
|
|
{
|
|
|
|
|
|
var request = new ImportContactRequest(new InputPhoneContactConstructor(0, phoneNumber, "My Test Name", string.Empty));
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
|
|
|
|
|
|
var importedUser = request.users.FirstOrDefault();
|
|
|
|
|
|
|
|
|
|
|
|
return importedUser == null ? (int?) null : ((UserContactConstructor) importedUser).id;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task SendMessage(int id, string message)
|
|
|
|
|
|
{
|
|
|
|
|
|
var request = new SendMessageRequest(new InputPeerContactConstructor(id), message );
|
|
|
|
|
|
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
}
|
2015-10-14 18:16:27 +02:00
|
|
|
|
|
|
|
|
|
|
public async Task LoadChatsAndUsers(int offset, int max_id, int limit)
|
|
|
|
|
|
{
|
|
|
|
|
|
// GetDialogs
|
|
|
|
|
|
var request = new GetDialogsRequest(offset, max_id, limit);
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
|
|
|
|
|
|
chats = request.chats;
|
|
|
|
|
|
users = request.users;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<List<Message>> GetHistory(int user_id, int offset, int max_id, int limit)
|
|
|
|
|
|
{
|
|
|
|
|
|
var request = new GetHistoryRequest(new InputPeerContactConstructor(user_id), offset, max_id, limit);
|
|
|
|
|
|
await _sender.Send(request);
|
|
|
|
|
|
await _sender.Recieve(request);
|
|
|
|
|
|
|
|
|
|
|
|
return request.messages;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2015-09-28 04:01:17 +02:00
|
|
|
|
}
|