#import "TON.h" #import "MTLogging.h" #import "tonlib/Client.h" #import "MTQueue.h" #import "MTSignal.h" static td::SecureString makeSecureString(NSData * _Nonnull data) { if (data == nil || data.length == 0) { return td::SecureString(); } else { return td::SecureString((const char *)data.bytes, (size_t)data.length); } } static std::string makeString(NSData * _Nonnull data) { if (data == nil || data.length == 0) { return std::string(); } else { return std::string((const char *)data.bytes, ((const char *)data.bytes) + data.length); } } static NSData * _Nonnull makeData(std::string &string) { if (string.size() == 0) { return [NSData data]; } else { return [[NSData alloc] initWithBytes:string.data() length:string.size()]; } } static NSString * _Nullable readString(std::string &string) { if (string.size() == 0) { return @""; } else { return [[NSString alloc] initWithBytes:string.data() length:string.size() encoding:NSUTF8StringEncoding]; } } static TONTransactionMessage * _Nullable parseTransactionMessage(tonlib_api::object_ptr &message) { if (message == nullptr) { return nil; } NSString *source = readString(message->source_); NSString *destination = readString(message->destination_); NSString *textMessage = readString(message->message_); if (source == nil || destination == nil) { return nil; } if (textMessage == nil) { textMessage = @""; } return [[TONTransactionMessage alloc] initWithValue:message->value_ source:source destination:destination textMessage:textMessage]; } @implementation TONKey - (instancetype)initWithPublicKey:(NSString *)publicKey secret:(NSData *)secret { self = [super init]; if (self != nil) { _publicKey = publicKey; _secret = secret; } return self; } @end @implementation TONAccountState - (instancetype)initWithIsInitialized:(bool)isInitialized balance:(int64_t)balance seqno:(int32_t)seqno lastTransactionId:(TONTransactionId * _Nullable)lastTransactionId syncUtime:(int64_t)syncUtime { self = [super init]; if (self != nil) { _isInitialized = isInitialized; _balance = balance; _seqno = seqno; _lastTransactionId = lastTransactionId; _syncUtime = syncUtime; } return self; } @end @implementation TONTransactionId - (instancetype)initWithLt:(int64_t)lt transactionHash:(NSData *)transactionHash { self = [super init]; if (self != nil) { _lt = lt; _transactionHash = transactionHash; } return self; } @end @implementation TONTransactionMessage - (instancetype)initWithValue:(int64_t)value source:(NSString * _Nonnull)source destination:(NSString * _Nonnull)destination textMessage:(NSString * _Nonnull)textMessage { self = [super init]; if (self != nil) { _value = value; _source = source; _destination = destination; _textMessage = textMessage; } return self; } @end @implementation TONTransaction - (instancetype)initWithData:(NSData * _Nonnull)data transactionId:(TONTransactionId * _Nonnull)transactionId timestamp:(int64_t)timestamp fee:(int64_t)fee inMessage:(TONTransactionMessage * _Nullable)inMessage outMessages:(NSArray * _Nonnull)outMessages { self = [super init]; if (self != nil) { _data = data; _transactionId = transactionId; _timestamp = timestamp; _fee = fee; _inMessage = inMessage; _outMessages = outMessages; } return self; } @end @implementation TONExternalRequest - (instancetype)initWithData:(NSData * _Nonnull)data onResult:(void (^)(NSData * _Nullable, NSString * _Nullable))onResult { self = [super init]; if (self != nil) { _data = data; _onResult = [onResult copy]; } return self; } @end @implementation TONSendGramsResult - (instancetype)initWithSentUntil:(int64_t)sentUntil { self = [super init]; if (self != nil) { _sentUntil = sentUntil; } return self; } @end using tonlib_api::make_object; @interface TONReceiveThreadParams : NSObject @property (nonatomic, readonly) std::shared_ptr client; @property (nonatomic, copy, readonly) void (^received)(tonlib::Client::Response &); @end @implementation TONReceiveThreadParams - (instancetype)initWithClient:(std::shared_ptr)client received:(void (^)(tonlib::Client::Response &))received { self = [super init]; if (self != nil) { _client = client; _received = [received copy]; } return self; } @end @interface TONRequestHandler : NSObject @property (nonatomic, copy, readonly) void (^completion)(tonlib_api::object_ptr &); @end @implementation TONRequestHandler - (instancetype)initWithCompletion:(void (^)(tonlib_api::object_ptr &))completion { self = [super init]; if (self != nil) { _completion = [completion copy]; } return self; } @end @implementation TONError - (instancetype)initWithText:(NSString *)text { self = [super init]; if (self != nil) { _text = text; } return self; } @end typedef enum { TONInitializationStatusInitializing, TONInitializationStatusReady, TONInitializationStatusError } TONInitializationStatus; @interface TON () { std::shared_ptr _client; uint64_t _nextRequestId; NSLock *_requestHandlersLock; NSMutableDictionary *_requestHandlers; MTPipe *_initializedStatus; NSMutableSet *_sendGramRandomIds; MTQueue *_queue; } @end @implementation TON + (void)receiveThread:(TONReceiveThreadParams *)params { while (true) { auto response = params.client->receive(1000); if (response.object) { params.received(response); } } } - (instancetype)initWithKeystoreDirectory:(NSString *)keystoreDirectory config:(NSString *)config blockchainName:(NSString *)blockchainName performExternalRequest:(void (^)(TONExternalRequest * _Nonnull))performExternalRequest { self = [super init]; if (self != nil) { _queue = [MTQueue mainQueue]; _requestHandlersLock = [[NSLock alloc] init]; _requestHandlers = [[NSMutableDictionary alloc] init]; _initializedStatus = [[MTPipe alloc] initWithReplay:true]; _initializedStatus.sink(@(TONInitializationStatusInitializing)); _nextRequestId = 1; _sendGramRandomIds = [[NSMutableSet alloc] init]; _client = std::make_shared(); std::weak_ptr weakClient = _client; NSLock *requestHandlersLock = _requestHandlersLock; NSMutableDictionary *requestHandlers = _requestHandlers; NSThread *thread = [[NSThread alloc] initWithTarget:[self class] selector:@selector(receiveThread:) object:[[TONReceiveThreadParams alloc] initWithClient:_client received:^(tonlib::Client::Response &response) { if (response.object->get_id() == tonlib_api::updateSendLiteServerQuery::ID) { auto result = tonlib_api::move_object_as(response.object); int64_t requestId = result->id_; NSData *data = makeData(result->data_); if (performExternalRequest) { performExternalRequest([[TONExternalRequest alloc] initWithData:data onResult:^(NSData * _Nullable result, NSString * _Nullable error) { auto strongClient = weakClient.lock(); if (strongClient != nullptr) { if (result != nil) { auto query = make_object( requestId, makeString(result) ); strongClient->send({ 1, std::move(query) }); } else if (error != nil) { auto query = make_object( requestId, make_object( 400, error.UTF8String ) ); strongClient->send({ 1, std::move(query) }); } } }]); } return; } NSNumber *requestId = @(response.id); [requestHandlersLock lock]; TONRequestHandler *handler = requestHandlers[requestId]; [requestHandlers removeObjectForKey:requestId]; [requestHandlersLock unlock]; if (handler != nil) { handler.completion(response.object); } }]]; [thread start]; [[NSFileManager defaultManager] createDirectoryAtPath:keystoreDirectory withIntermediateDirectories:true attributes:nil error:nil]; MTPipe *initializedStatus = _initializedStatus; [[self requestInitWithConfigString:config blockchainName:blockchainName keystoreDirectory:keystoreDirectory] startWithNext:nil error:^(id error) { NSString *errorText = @"Unknown error"; if ([error isKindOfClass:[TONError class]]) { errorText = ((TONError *)error).text; } initializedStatus.sink(@(TONInitializationStatusError)); } completed:^{ initializedStatus.sink(@(TONInitializationStatusReady)); }]; } return self; } - (MTSignal *)requestInitWithConfigString:(NSString *)configString blockchainName:(NSString *)blockchainName keystoreDirectory:(NSString *)keystoreDirectory { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else { [subscriber putCompletion]; } }]; auto query = make_object(make_object( make_object( configString.UTF8String, blockchainName.UTF8String, true, false ), make_object( keystoreDirectory.UTF8String ) )); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)createKeyWithLocalPassword:(NSData *)localPassword mnemonicPassword:(NSData *)mnemonicPassword { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else if (object->get_id() == tonlib_api::key::ID) { auto result = tonlib_api::move_object_as(object); NSString *publicKey = [[NSString alloc] initWithData:[[NSData alloc] initWithBytes:result->public_key_.data() length:result->public_key_.length()] encoding:NSUTF8StringEncoding]; if (publicKey == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error decoding UTF8 string in createKeyWithLocalPassword"]]; return; } NSData *secret = [[NSData alloc] initWithBytes:result->secret_.data() length:result->secret_.length()]; [subscriber putNext:[[TONKey alloc] initWithPublicKey:publicKey secret:secret]]; [subscriber putCompletion]; } else { assert(false); } }]; auto query = make_object( makeSecureString(localPassword), makeSecureString(mnemonicPassword), td::SecureString() ); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)getWalletAccountAddressWithPublicKey:(NSString *)publicKey { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { NSData *publicKeyData = [publicKey dataUsingEncoding:NSUTF8StringEncoding]; if (publicKeyData == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error encoding UTF8 string in getWalletAccountAddressWithPublicKey"]]; return [[MTBlockDisposable alloc] initWithBlock:^{}]; } uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else if (object->get_id() == tonlib_api::accountAddress::ID) { auto result = tonlib_api::move_object_as(object); [subscriber putNext:[[NSString alloc] initWithUTF8String:result->account_address_.c_str()]]; [subscriber putCompletion]; } else { assert(false); } }]; auto query = make_object( make_object( makeString(publicKeyData) ) ); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)getAccountStateWithAddress:(NSString *)accountAddress { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else if (object->get_id() == tonlib_api::generic_accountStateUninited::ID) { auto result = tonlib_api::move_object_as(object); TONTransactionId *lastTransactionId = nil; if (result->account_state_->last_transaction_id_ != nullptr) { lastTransactionId = [[TONTransactionId alloc] initWithLt:result->account_state_->last_transaction_id_->lt_ transactionHash:makeData(result->account_state_->last_transaction_id_->hash_)]; } [subscriber putNext:[[TONAccountState alloc] initWithIsInitialized:false balance:result->account_state_->balance_ seqno:-1 lastTransactionId:lastTransactionId syncUtime:result->account_state_->sync_utime_]]; [subscriber putCompletion]; } else if (object->get_id() == tonlib_api::generic_accountStateWallet::ID) { auto result = tonlib_api::move_object_as(object); TONTransactionId *lastTransactionId = nil; if (result->account_state_->last_transaction_id_ != nullptr) { lastTransactionId = [[TONTransactionId alloc] initWithLt:result->account_state_->last_transaction_id_->lt_ transactionHash:makeData(result->account_state_->last_transaction_id_->hash_)]; } [subscriber putNext:[[TONAccountState alloc] initWithIsInitialized:true balance:result->account_state_->balance_ seqno:result->account_state_->seqno_ lastTransactionId:lastTransactionId syncUtime:result->account_state_->sync_utime_]]; [subscriber putCompletion]; } else { assert(false); } }]; auto query = make_object(make_object(accountAddress.UTF8String)); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)sendGramsFromKey:(TONKey *)key localPassword:(NSData *)localPassword fromAddress:(NSString *)fromAddress toAddress:(NSString *)address amount:(int64_t)amount textMessage:(NSString * _Nonnull)textMessage forceIfDestinationNotInitialized:(bool)forceIfDestinationNotInitialized timeout:(int32_t)timeout randomId:(int64_t)randomId { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { if ([_sendGramRandomIds containsObject:@(randomId)]) { [_sendGramRandomIds addObject:@(randomId)]; return [[MTBlockDisposable alloc] initWithBlock:^{ }]; } NSData *publicKeyData = [key.publicKey dataUsingEncoding:NSUTF8StringEncoding]; if (publicKeyData == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error encoding UTF8 string in sendGramsFromKey"]]; return [[MTBlockDisposable alloc] initWithBlock:^{}]; } NSData *textMessageData = [textMessage dataUsingEncoding:NSUTF8StringEncoding]; if (textMessageData == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error encoding UTF8 string in sendGramsFromKey"]]; return [[MTBlockDisposable alloc] initWithBlock:^{}]; } uint64_t requestId = _nextRequestId; _nextRequestId += 1; __weak TON *weakSelf = self; MTQueue *queue = _queue; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { [queue dispatchOnQueue:^{ __strong TON *strongSelf = weakSelf; if (strongSelf != nil) { [_sendGramRandomIds removeObject:@(randomId)]; } }]; auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else if (object->get_id() == tonlib_api::sendGramsResult::ID) { auto result = tonlib_api::move_object_as(object); TONSendGramsResult *sendResult = [[TONSendGramsResult alloc] initWithSentUntil:result->sent_until_]; [subscriber putNext:sendResult]; [subscriber putCompletion]; } else { [subscriber putCompletion]; } }]; auto query = make_object( make_object( make_object( makeString(publicKeyData), makeSecureString(key.secret) ), makeSecureString(localPassword) ), make_object(fromAddress.UTF8String), make_object(address.UTF8String), amount, timeout, forceIfDestinationNotInitialized, makeString(textMessageData) ); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)exportKey:(TONKey *)key localPassword:(NSData *)localPassword { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { NSData *publicKeyData = [key.publicKey dataUsingEncoding:NSUTF8StringEncoding]; if (publicKeyData == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error encoding UTF8 string in exportKey"]]; return [[MTBlockDisposable alloc] initWithBlock:^{}]; } uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else if (object->get_id() == tonlib_api::exportedKey::ID) { auto result = tonlib_api::move_object_as(object); NSMutableArray *wordList = [[NSMutableArray alloc] init]; for (auto &it : result->word_list_) { NSString *string = [[NSString alloc] initWithData:[[NSData alloc] initWithBytes:it.data() length:it.size()] encoding:NSUTF8StringEncoding]; if (string == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error decoding UTF8 string in exportedKey::word_list"]]; return; } [wordList addObject:string]; } [subscriber putNext:wordList]; [subscriber putCompletion]; } else { assert(false); } }]; auto query = make_object( make_object( make_object( makeString(publicKeyData), makeSecureString(key.secret) ), makeSecureString(localPassword) ) ); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)importKeyWithLocalPassword:(NSData *)localPassword mnemonicPassword:(NSData *)mnemonicPassword wordList:(NSArray *)wordList { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else if (object->get_id() == tonlib_api::key::ID) { auto result = tonlib_api::move_object_as(object); NSString *publicKey = [[NSString alloc] initWithData:[[NSData alloc] initWithBytes:result->public_key_.data() length:result->public_key_.length()] encoding:NSUTF8StringEncoding]; if (publicKey == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error decoding UTF8 string in importKeyWithLocalPassword"]]; return; } NSData *secret = [[NSData alloc] initWithBytes:result->secret_.data() length:result->secret_.length()]; [subscriber putNext:[[TONKey alloc] initWithPublicKey:publicKey secret:secret]]; [subscriber putCompletion]; } else { assert(false); } }]; std::vector wordVector; for (NSString *word in wordList) { NSData *wordData = [word dataUsingEncoding:NSUTF8StringEncoding]; wordVector.push_back(makeSecureString(wordData)); } auto query = make_object( makeSecureString(localPassword), makeSecureString(mnemonicPassword), make_object(std::move(wordVector))); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)deleteKey:(TONKey *)key { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { NSData *publicKeyData = [key.publicKey dataUsingEncoding:NSUTF8StringEncoding]; if (publicKeyData == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error encoding UTF8 string in deleteKey"]]; return [[MTBlockDisposable alloc] initWithBlock:^{}]; } uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else { [subscriber putCompletion]; } }]; auto query = make_object( make_object( makeString(publicKeyData), makeSecureString(key.secret) ) ); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)deleteAllKeys { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else { [subscriber putCompletion]; } }]; auto query = make_object(); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } - (MTSignal *)getTransactionListWithAddress:(NSString * _Nonnull)address lt:(int64_t)lt hash:(NSData * _Nonnull)hash { return [[[[MTSignal alloc] initWithGenerator:^id(MTSubscriber *subscriber) { NSData *addressData = [address dataUsingEncoding:NSUTF8StringEncoding]; if (addressData == nil) { [subscriber putError:[[TONError alloc] initWithText:@"Error encoding UTF8 string in getTransactionListWithAddress"]]; return [[MTBlockDisposable alloc] initWithBlock:^{}]; } uint64_t requestId = _nextRequestId; _nextRequestId += 1; _requestHandlers[@(requestId)] = [[TONRequestHandler alloc] initWithCompletion:^(tonlib_api::object_ptr &object) { if (object->get_id() == tonlib_api::error::ID) { auto error = tonlib_api::move_object_as(object); [subscriber putError:[[TONError alloc] initWithText:[[NSString alloc] initWithUTF8String:error->message_.c_str()]]]; } else if (object->get_id() == tonlib_api::raw_transactions::ID) { auto result = tonlib_api::move_object_as(object); NSMutableArray *transactions = [[NSMutableArray alloc] init]; for (auto &it : result->transactions_) { TONTransactionId *transactionId = [[TONTransactionId alloc] initWithLt:it->transaction_id_->lt_ transactionHash:makeData(it->transaction_id_->hash_)]; TONTransactionMessage *inMessage = parseTransactionMessage(it->in_msg_); NSMutableArray * outMessages = [[NSMutableArray alloc] init]; for (auto &messageIt : it->out_msgs_) { TONTransactionMessage *outMessage = parseTransactionMessage(messageIt); if (outMessage != nil) { [outMessages addObject:outMessage]; } } [transactions addObject:[[TONTransaction alloc] initWithData:makeData(it->data_) transactionId:transactionId timestamp:it->utime_ fee:it->fee_ inMessage:inMessage outMessages:outMessages]]; } [subscriber putNext:transactions]; [subscriber putCompletion]; } else { assert(false); } }]; auto query = make_object( make_object( makeString(addressData) ), make_object( lt, makeString(hash) ) ); _client->send({ requestId, std::move(query) }); return [[MTBlockDisposable alloc] initWithBlock:^{ }]; }] startOn:[MTQueue mainQueue]] deliverOn:[MTQueue mainQueue]]; } @end