using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Text.RegularExpressions; 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; using MD5 = System.Security.Cryptography.MD5; namespace TLSharp.Core { public class TelegramClient { private MtProtoSender _sender; private AuthKey _key; private TcpTransport _transport; private string _apiHash = ""; private int _apiId = 0; private Session _session; private List dcOptions; public enum sms_type { numeric_code_via_sms = 0, numeric_code_via_telegram = 5 } public TelegramClient(ISessionStore store, string sessionUserId, int apiId, string apiHash) { _apiHash = apiHash; _apiId = apiId; 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"); _session = Session.TryLoadOrCreateNew(store, sessionUserId); _transport = new TcpTransport(_session.ServerAddress, _session.Port); } public async Task Connect(bool reconnect = false) { if (_session.AuthKey == null || reconnect) { var result = await Authenticator.DoAuthentication(_transport); _session.AuthKey = result.AuthKey; _session.TimeOffset = result.TimeOffset; } _sender = new MtProtoSender(_transport, _session); if (!reconnect) { var request = new InitConnectionRequest(_apiId); await _sender.Send(request); await _sender.Receive(request); dcOptions = request.ConfigConstructor.dc_options; } return true; } private async Task ReconnectToDc(int dcId) { if (dcOptions == null || !dcOptions.Any()) throw new InvalidOperationException($"Can't reconnect. Establish initial connection first."); var dc = dcOptions.Cast().First(d => d.id == dcId); _transport = new TcpTransport(dc.ip_address, dc.port); _session.ServerAddress = dc.ip_address; _session.Port = dc.port; await Connect(true); } public bool IsUserAuthorized() { return _session.User != null; } public async Task IsPhoneRegistered(string phoneNumber) { if (_sender == null) throw new InvalidOperationException("Not connected!"); var authCheckPhoneRequest = new AuthCheckPhoneRequest(phoneNumber); await _sender.Send(authCheckPhoneRequest); await _sender.Receive(authCheckPhoneRequest); return authCheckPhoneRequest._phoneRegistered; } public async Task SendCodeRequest(string phoneNumber, sms_type tokenDestination = sms_type.numeric_code_via_telegram) { var completed = false; AuthSendCodeRequest request = null; while (!completed) { request = new AuthSendCodeRequest(phoneNumber, (int)tokenDestination, _apiId, _apiHash, "en"); try { await _sender.Send(request); await _sender.Receive(request); completed = true; } catch (InvalidOperationException ex) { if (ex.Message.StartsWith("Your phone number registered to") && ex.Data["dcId"] != null) { await ReconnectToDc((int)ex.Data["dcId"]); } else { throw; } } } return request._phoneCodeHash; } public async Task MakeAuth(string phoneNumber, string phoneHash, string code) { var request = new AuthSignInRequest(phoneNumber, phoneHash, code); await _sender.Send(request); await _sender.Receive(request); _session.SessionExpires = request.SessionExpires; _session.User = request.user; _session.Save(); return request.user; } public async Task UploadFile(string name, byte[] data) { var partSize = 65536; var file_id = DateTime.Now.Ticks; var partedData = new Dictionary(); 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.Receive(saveFilePartRequest); if (saveFilePartRequest.Done == false) throw new InvalidOperationException($"File part {i} does not uploaded"); } string md5_checksum; using (var md5 = 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 SendMediaMessage(int contactId, InputFile file) { var request = new Message_SendMediaRequest( new InputPeerContactConstructor(contactId), new InputMediaUploadedPhotoConstructor(file)); await _sender.Send(request); await _sender.Receive(request); return true; } public async Task ImportContactByPhoneNumber(string phoneNumber) { if (!validateNumber(phoneNumber)) throw new InvalidOperationException("Invalid phone number. It should be only digit string, from 5 to 20 digits."); var request = new ImportContactRequest(new InputPhoneContactConstructor(0, phoneNumber, "My Test Name", String.Empty)); await _sender.Send(request); await _sender.Receive(request); var importedUser = (ImportedContactConstructor)request.imported.FirstOrDefault(); return importedUser?.user_id; } public async Task ImportByUserName(string username) { if (string.IsNullOrEmpty(username)) throw new InvalidOperationException("Username can't be null"); var request = new ImportByUserName(username); await _sender.Send(request); await _sender.Receive(request); return request.id; } public async Task SendMessage(int id, string message) { var request = new SendMessageRequest(new InputPeerContactConstructor(id), message); await _sender.Send(request); await _sender.Receive(request); } public async Task> GetMessagesHistoryForContact(int user_id, int offset, int limit, int max_id = -1) { var request = new GetHistoryRequest(new InputPeerContactConstructor(user_id), offset, max_id, limit); await _sender.Send(request); await _sender.Receive(request); return request.messages; } public async Task> GetFile(long volume_id, int local_id, long secret, int offset, int limit) { var request = new GetFileRequest(new InputFileLocationConstructor(volume_id, local_id, secret), offset, limit); await _sender.Send(request); await _sender.Receive(request); return Tuple.Create(request.type, request.bytes); } public async Task GetDialogs(int offset, int limit, int max_id = 0) { var request = new GetDialogsRequest(offset, max_id, limit); await _sender.Send(request); await _sender.Receive(request); return new MessageDialogs { Dialogs = request.dialogs, Messages = request.messages, Chats = request.chats, Users = request.users, }; } public async Task GetUserFull(int user_id) { var request = new GetUserFullRequest(user_id); await _sender.Send(request); await _sender.Receive(request); return request._userFull; } private bool validateNumber(string number) { var regex = new Regex("^\\d{7,20}$"); return regex.IsMatch(number); } } }