Greetings to everyone who cares! One boring day, I decided to try my hand at creating a project using the Telegram API, not expecting it to turn into what it did. I explored several integration approaches and came to some interesting conclusions along the way. Eventually, I began developing an open-source library called react-native-tdlib.
\ In this series of articles, I’ll share the challenges I faced and the new discoveries I made throughout this journey.
MTProto or the Story of Big CrutchAt first, I decided to take the easy route and simply used libraries designed for the browser, specifically ”@mtproto/core” together with “react-native-webview-crypto”. I know, it already sounds a bit odd, but I wanted to give it a shot. I even managed to implement the entire authorization process using this stack.
\ However, it didn’t take long to realize that this approach wasn’t cutting it. Since it relies on a browser running in the background, the performance takes a significant hit — responses become very slow, and some functions are outright impossible to implement.
\ Because of these limitations, I quickly abandoned this approach and decided it was time to dive into native code.
TDLib Pre-built LibraryI’ll be honest — this was my first experience working with a prebuilt library. To use it, you need to build the library separately for each platform (iOS and Android). The process involves running .sh scripts provided in their examples, along with meeting specific system requirements for building.
\ Once built, you end up with a library for each platform, which you then need to manually import into your project (in my case, into my own library).
\ If you’re curious, you can check out my repository to see how I organized the library files and integrated them.
\ Problem: The library is quite large — around 400 MB for both platforms — which makes storing it in the repository impractical. For now, the pre-built library is included in the repo since I haven’t found a better solution yet.
\ In the future, I plan to upload it to an external storage service and set it up to import automatically during installation. Honestly, I’m still exploring the best way to handle this, as it’s the first time I’ve encountered a problem like this.
Native ModuleIn the first step, I decided to try to wrap basic TDLib methods without additional logic and try to implement something this way.
Let’s break this down using the tdjsonclient_receive example method. Below, I provide the code for this function in Java and Objective-C.
\
@ReactMethod public void td_json_client_receive(Promise promise) { try { if (client == null) { promise.reject("CLIENT_NOT_INITIALIZED", "TDLib client is not initialized"); return; } CountDownLatch latch = new CountDownLatch(1); AtomicReference\
RCT_EXPORT_METHOD(td_json_client_receive:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { if (_client == NULL) { reject(@"CLIENT_NOT_INITIALIZED", @"TDLib client not initialized", nil); return; } const char *response = td_json_client_receive(_client, 10.0); if (response) { NSString *responseString = [NSString stringWithUTF8String:response]; resolve(responseString); } else { reject(@"RECEIVE_ERROR", @"No response from TDLib", nil); } }\ Problem: The receive function returns all events from TDLib, and to catch the specific event we need, we have to use a loop. This approach isn’t very efficient at the JavaScript layer. I’ll attach the implementation code in JS below, but I wouldn’t recommend using this method.
\
/** * Fetches the list of supported languages from TDLib. */ const fetchSupportedLanguages = async () => { try { await setLocalizationTargetOption(); const request = { '@type': 'getLocalizationTargetInfo', only_locales: true, }; TdLib.td_json_client_send(request); while (true) { const response = await TdLib.td_json_client_receive(); if (response) { const parsedResponse = JSON.parse(response); if (parsedResponse['@type'] === 'localizationTargetInfo') { return parsedResponse; } if (parsedResponse['@type'] === 'error') { throw new Error( `Error fetching supported languages: ${parsedResponse.message}`, ); } } else { throw new Error('No response from TDLib'); } } } catch (error) { console.error('Error in fetchSupportedLanguages:', error); throw error; } };\ Now, we’ve finally reached the final solution: the complex logic needs to be moved into native code. This involves creating a Client and handling everything there. I understand that this approach requires implementing a lot of logic and methods, but I don’t see any better alternatives.
\ Below is an example implementation of the getAuthorizationState function, where we loop through events to find the one we need.
\
RCT_EXPORT_METHOD(getAuthorizationState:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) { @try { if (_client == NULL) { reject(@"TDLIB_NOT_STARTED", @"TDLib client is not initialized. Call startTdLibService first.", nil); return; } NSString *request = @"{\"@type\":\"getAuthorizationState\"}"; td_json_client_send(_client, [request UTF8String]); while (true) { const char *response = td_json_client_receive(_client, 10.0); if (response != NULL) { NSString *responseString = [NSString stringWithUTF8String:response]; NSLog(@"TDLib response: %@", responseString); NSDictionary *responseDict = [NSJSONSerialization JSONObjectWithData:[responseString dataUsingEncoding:NSUTF8StringEncoding] options:0 error:nil]; NSString *type = responseDict[@"@type"]; if ([type isEqualToString:@"authorizationStateWaitPhoneNumber"] || [type isEqualToString:@"authorizationStateWaitCode"] || [type isEqualToString:@"authorizationStateReady"] || [type isEqualToString:@"authorizationStateWaitOtherDeviceConfirmation"] || [type isEqualToString:@"authorizationStateClosed"]) { resolve(responseString); return; } if ([type containsString:@"update"]) { NSLog(@"Ignoring update: %@", type); continue; } } else { reject(@"NO_RESPONSE", @"No response from TDLib", nil); return; } } } @catch (NSException *exception) { reject(@"GET_AUTH_STATE_EXCEPTION", exception.reason, nil); } }\
@ReactMethod public void getAuthorizationState(Promise promise) { try { if (client == null) { promise.reject("CLIENT_NOT_INITIALIZED", "TDLib client is not initialized"); return; } client.send(new TdApi.GetAuthorizationState(), object -> { if (object instanceof TdApi.AuthorizationState) { try { Map\ You can find the rest of the implemented functions in the repository. Below is an example implementation at the JS layer:
\
useEffect(() => { // Initializes TDLib with the provided parameters and checks the authorization state TdLib.startTdLib(parameters).then(r => { TdLib.getAuthorizationState().then(r => { if (JSON.parse(r)['@type'] === 'authorizationStateReady') { TdLib.getProfile(); // Fetches the user's profile if authorization is ready } }); }); }, []);\ At this point, I’ve completed the development of methods for authorization and retrieving user profiles. I’ll dive deeper into these topics in the next article. Thank you for your attention!
https://github.com/vladlenskiy/react-native-tdlib?embedable=true
\
All Rights Reserved. Copyright , Central Coast Communications, Inc.