mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-06-28 12:29:51 +00:00
* tor wip * Enable tor on iOS * Prevent app lag when node is exceptionally slow (usually over tor) * fix: logic in daemonBlockchainHeight refresh fix: storing tor state * Pin ledger_flutter_plus dependency to fix builds * bump arti version * wip * add single httpclient * route everything I was able to catch trough the built-in tor node * Enable proxy for http.Client [run tests] * add tor proxy support to cw_evm, cw_tron and cw_polygon [run tests] * remove log pollution, cleanup [skip slack] * fix tests not working in latest main [skip slack] [run tests] * remove cw_wownero import * fix build issues * migrate all remaining calls to use ProxyWrapper add a CI action to enforce using ProxyWrapper instead of http/http.dart to prevent leaks * fix tor background sync (will work on test builds after #2142 is merged and this PR is rebased on top) * wip [skip ci] * relicense to GPLv3 add socks5 license, build fixes * use ProxyWrapper instead of http in robinhood * Revert "relicense to GPLv3" * feat(cw_bitcoin): support socks proxy and CakeTor * fix(tor): migrate OCP and EVM over to ProxyWrapper() * chore: cleanup fix: show tor loading screen when app is starting * fix: tor switch properly dismisses fullscreen loading dialog fix: connectToNode after tor startup on app start * fix(tor): status check for xmr/wow/zano * fix(tor): onramper request fix * fix(api): ServicesResponse is now being cached and doesn't fetch data everytime DashboardViewModel is being rebuilt fix(tor): do not fallback to clearnet when tor failed. fix(tor): do not leak connections during app startup chore: refactor bootstrap() function to be separated into bootstrapOffline and bootstrapOnline fix(cw_bitcoin): migrate payjoin to use ProxyWrapper * [skip ci] remove print * address comments from review * fix: derusting tor implementation Instead of rust-based Arti I've moved back to the OG C++ tor implementation. This fixed all issues we had with Tor. - onion services now work - all requests are going through without random errors - we don't have to navigate a maze of multiple forks of multiple packages - fully working `torrc` config file (probably will be needed for Tari). - logging for Tor client - and so on. feat: network logging tab feat: use built-in proxy on Tails - this should resolve all issues for Tails users (needs testing though) * fix conflicts with main bump https to fix build issue relax store() call * fix(cw_wownero): tor connection fix(tor): connection issues * fix(cw_evm): add missing chainId fix(cw_core): solana rpc fix * feat: mark tor as experimental fix: drop anonpay onion authority fix: drop fiatapi onion authority fix: drop trocador onion authority fix: disable networkimage when tor is enabled fix: handle cakepay errors gracefully * fix re-formatting [skip ci] * changes from review * Delete android/.kotlin/sessions/kotlin-compiler-2468481326039681181.salive * fix missing imports * Update pubspec_base.yaml --------- Co-authored-by: OmarHatem <omarh.ismail1@gmail.com>
200 lines
6.1 KiB
Dart
200 lines
6.1 KiB
Dart
import 'dart:convert';
|
|
import 'dart:developer';
|
|
|
|
import 'package:cake_wallet/entities/solana_nft_asset_model.dart';
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
|
import 'package:cake_wallet/reactions/wallet_connect.dart';
|
|
import 'package:cake_wallet/src/screens/wallet_connect/services/bottom_sheet_service.dart';
|
|
import 'package:cake_wallet/src/screens/wallet_connect/widgets/bottom_sheet/bottom_sheet_message_display_widget.dart';
|
|
import 'package:cw_core/wallet_type.dart';
|
|
import 'package:cw_core/utils/proxy_wrapper.dart';
|
|
import 'package:mobx/mobx.dart';
|
|
import 'package:cake_wallet/.secrets.g.dart' as secrets;
|
|
|
|
import 'package:cake_wallet/entities/wallet_nft_response.dart';
|
|
import 'package:cake_wallet/store/app_store.dart';
|
|
|
|
part 'nft_view_model.g.dart';
|
|
|
|
class NFTViewModel = NFTViewModelBase with _$NFTViewModel;
|
|
|
|
abstract class NFTViewModelBase with Store {
|
|
NFTViewModelBase(this.appStore, this.bottomSheetService)
|
|
: isLoading = false,
|
|
isImportNFTLoading = false,
|
|
nftAssetByWalletModels = ObservableList(),
|
|
solanaNftAssetModels = ObservableList();
|
|
|
|
final AppStore appStore;
|
|
final BottomSheetService bottomSheetService;
|
|
|
|
@observable
|
|
bool isLoading;
|
|
|
|
@observable
|
|
bool isImportNFTLoading;
|
|
|
|
ObservableList<NFTAssetModel> nftAssetByWalletModels;
|
|
|
|
ObservableList<SolanaNFTAssetModel> solanaNftAssetModels;
|
|
|
|
@action
|
|
Future<void> getNFTAssetByWallet() async {
|
|
final walletType = appStore.wallet!.type;
|
|
|
|
if (!isNFTACtivatedChain(walletType)) return;
|
|
|
|
final walletAddress = appStore.wallet!.walletInfo.address;
|
|
log('Fetching wallet NFTs for $walletAddress');
|
|
|
|
final chainName = getChainNameBasedOnWalletType(walletType);
|
|
// the [chain] refers to the chain network that the nft is on
|
|
// the [format] refers to the number format type of the responses
|
|
// the [normalizedMetadata] field is a boolean that determines if
|
|
// the response would include a json string of the NFT Metadata that can be decoded
|
|
// and used within the wallet
|
|
// the [excludeSpam] field is a boolean that determines if spam nfts be excluded from the response.
|
|
|
|
Uri uri;
|
|
if (walletType == WalletType.solana) {
|
|
uri = Uri.https(
|
|
'solana-gateway.moralis.io',
|
|
'/account/$chainName/$walletAddress/nft',
|
|
);
|
|
} else {
|
|
uri = Uri.https(
|
|
'deep-index.moralis.io',
|
|
'/api/v2.2/$walletAddress/nft',
|
|
{
|
|
"chain": chainName,
|
|
"format": "decimal",
|
|
"media_items": "false",
|
|
"exclude_spam": "true",
|
|
"normalizeMetadata": "true",
|
|
},
|
|
);
|
|
}
|
|
|
|
try {
|
|
if (isLoading) return;
|
|
|
|
isLoading = true;
|
|
|
|
final response = await ProxyWrapper().get(
|
|
clearnetUri: uri,
|
|
headers: {
|
|
"Accept": "application/json",
|
|
"X-API-Key": secrets.moralisApiKey,
|
|
},
|
|
);
|
|
|
|
|
|
final decodedResponse = jsonDecode(response.body) as Map<String, dynamic>;
|
|
|
|
if (walletType == WalletType.solana) {
|
|
final results = await Future.wait(
|
|
(decodedResponse as List<dynamic>).map(
|
|
(x) {
|
|
final data = x as Map<String, dynamic>;
|
|
final mint = data['mint'] as String? ?? '';
|
|
return getSolanaNFTDetails(mint, chainName);
|
|
},
|
|
).toList(),
|
|
);
|
|
|
|
solanaNftAssetModels.clear();
|
|
|
|
solanaNftAssetModels.addAll(results);
|
|
} else {
|
|
final result =
|
|
WalletNFTsResponseModel.fromJson(decodedResponse as Map<String, dynamic>).result ?? [];
|
|
|
|
nftAssetByWalletModels.clear();
|
|
|
|
nftAssetByWalletModels.addAll(result);
|
|
}
|
|
} catch (e) {
|
|
log(e.toString());
|
|
bottomSheetService.queueBottomSheet(
|
|
isModalDismissible: true,
|
|
widget: BottomSheetMessageDisplayWidget(
|
|
message: S.current.moralis_nft_error,
|
|
),
|
|
);
|
|
} finally {
|
|
isLoading = false;
|
|
}
|
|
}
|
|
|
|
Future<SolanaNFTAssetModel> getSolanaNFTDetails(String address, String chainName) async {
|
|
final uri = Uri.https(
|
|
'solana-gateway.moralis.io',
|
|
'/nft/$chainName/$address/metadata',
|
|
);
|
|
|
|
final response = await ProxyWrapper().get(
|
|
clearnetUri: uri,
|
|
headers: {
|
|
"Accept": "application/json",
|
|
"X-API-Key": secrets.moralisApiKey,
|
|
},
|
|
);
|
|
|
|
final decodedResponse = jsonDecode(response.body) as Map<String, dynamic>;
|
|
|
|
return SolanaNFTAssetModel.fromJson(decodedResponse);
|
|
}
|
|
|
|
@action
|
|
Future<void> importNFT(String tokenAddress, String? tokenId) async {
|
|
final chainName = getChainNameBasedOnWalletType(appStore.wallet!.type);
|
|
// the [chain] refers to the chain network that the nft is on
|
|
// the [format] refers to the number format type of the responses
|
|
// the [normalizedMetadata] field is a boolean that determines if
|
|
// the response would include a json string of the NFT Metadata that can be decoded
|
|
// and used within the wallet
|
|
|
|
try {
|
|
isImportNFTLoading = true;
|
|
|
|
if (appStore.wallet!.type == WalletType.solana) {
|
|
final result = await getSolanaNFTDetails(tokenAddress, chainName);
|
|
|
|
solanaNftAssetModels.add(result);
|
|
} else {
|
|
final uri = Uri.https(
|
|
'deep-index.moralis.io',
|
|
'/api/v2.2/nft/$tokenAddress/$tokenId',
|
|
{
|
|
"chain": chainName,
|
|
"format": "decimal",
|
|
"media_items": "false",
|
|
"normalizeMetadata": "true",
|
|
},
|
|
);
|
|
final response = await ProxyWrapper().get(
|
|
clearnetUri: uri,
|
|
headers: {
|
|
"Accept": "application/json",
|
|
"X-API-Key": secrets.moralisApiKey,
|
|
},
|
|
);
|
|
|
|
final decodedResponse = jsonDecode(response.body) as Map<String, dynamic>;
|
|
|
|
final nftAsset = NFTAssetModel.fromJson(decodedResponse);
|
|
|
|
nftAssetByWalletModels.add(nftAsset);
|
|
}
|
|
} catch (e) {
|
|
bottomSheetService.queueBottomSheet(
|
|
isModalDismissible: true,
|
|
widget: BottomSheetMessageDisplayWidget(
|
|
message: e.toString(),
|
|
),
|
|
);
|
|
} finally {
|
|
isImportNFTLoading = false;
|
|
}
|
|
}
|
|
}
|