Skip to content
Snippets Groups Projects
substrate_sdk.dart 14.9 KiB
Newer Older
// ignore_for_file: avoid_print

import 'package:flutter/material.dart';
poka's avatar
poka committed
import 'package:flutter/services.dart';
import 'package:gecko/globals.dart';
poka's avatar
poka committed
import 'package:gecko/models/chest_data.dart';
import 'package:gecko/models/wallet_data.dart';
poka's avatar
poka committed
import 'package:polkawallet_sdk/api/apiKeyring.dart';
import 'package:polkawallet_sdk/api/types/networkParams.dart';
poka's avatar
poka committed
import 'package:polkawallet_sdk/api/types/txInfoData.dart';
import 'package:polkawallet_sdk/polkawallet_sdk.dart';
import 'package:polkawallet_sdk/storage/keyring.dart';
poka's avatar
poka committed
import 'package:polkawallet_sdk/storage/types/keyPairData.dart';
import 'package:provider/provider.dart';
poka's avatar
poka committed
import 'package:truncate/truncate.dart';
// import 'package:web_socket_channel/io.dart';

class SubstrateSdk with ChangeNotifier {
poka's avatar
poka committed
  final int ss58 = 42;

  final WalletSDK sdk = WalletSDK();
  final Keyring keyring = Keyring();
poka's avatar
poka committed
  String generatedMnemonic = '';
poka's avatar
poka committed
  bool sdkLoading = false;
poka's avatar
poka committed
  bool importIsLoading = false;
  bool isLoadingEndpoint = false;
poka's avatar
poka committed
  String debugConnection = '';
  String transactionStatus = '';
poka's avatar
poka committed
  TextEditingController jsonKeystore = TextEditingController();
  TextEditingController keystorePassword = TextEditingController();

poka's avatar
poka committed
    sdkLoading = true;
poka's avatar
poka committed
    await keyring.init([ss58]);
    keyring.setSS58(ss58);

    await sdk.init(keyring);
    sdkReady = true;
poka's avatar
poka committed
    sdkLoading = false;
  Future<void> connectNode(BuildContext ctx) async {
poka's avatar
poka committed
    List<NetworkParams> node = [];

    for (String _endpoint in configBox.get('endpoint')) {
      final n = NetworkParams();
      n.name = currencyName;
      n.endpoint = _endpoint;
      n.ss58 = ss58;
      node.add(n);
    }
poka's avatar
poka committed
    int timeout = 10000;

    // if (n.endpoint!.startsWith('ws://')) {
    //   timeout = 5000;
    // }

    //// Check websocket conenction - only for wss
    // final channel = IOWebSocketChannel.connect(
    //   Uri.parse('wss://192.168.1.72:9944'),
    // );

    // channel.stream.listen(
    //   (dynamic message) {
    //     log.d('message $message');
    //   },
    //   onDone: () {
    //     log.d('ws channel closed');
    //   },
    //   onError: (error) {
    //     log.d('ws error $error');
    //   },
    // );

    if (sdk.api.connectedNode?.endpoint != null) {
      await sdk.api.setting.unsubscribeBestNumber();
poka's avatar
poka committed
    }

    isLoadingEndpoint = true;
    notifyListeners();
poka's avatar
poka committed
    final res = await sdk.api.connectNode(keyring, node).timeout(
          Duration(milliseconds: timeout),
poka's avatar
poka committed
          onTimeout: () => null,
        );
    isLoadingEndpoint = false;
    notifyListeners();
    if (res != null) {
      nodeConnected = true;
poka's avatar
poka committed

      // Subscribe bloc number
      sdk.api.setting.subscribeBestNumber((res) {
        blocNumber = int.parse(res.toString());
        notifyListeners();
      });
      snackNode(ctx, true);
    } else {
      nodeConnected = false;
poka's avatar
poka committed
      debugConnection = res.toString();
      notifyListeners();
      snackNode(ctx, false);
    log.d(sdk.api.connectedNode?.endpoint);
poka's avatar
poka committed

poka's avatar
poka committed
  Future<String> importAccount(
      {String mnemonic = '',
      bool fromMnemonic = false,
      String derivePath = '',
      String password = ''}) async {
poka's avatar
poka committed
    // toy exercise immense month enter answer table prefer speed cycle gold phone
    final clipboardData = await Clipboard.getData(Clipboard.kTextPlain);
poka's avatar
poka committed
    if (mnemonic != '') {
      fromMnemonic = true;
      generatedMnemonic = mnemonic;
    } else if (clipboardData!.text!.split(' ').length == 12) {
poka's avatar
poka committed
      fromMnemonic = true;
      generatedMnemonic = clipboardData.text!;
    }

    if (password == '') {
      password = keystorePassword.text;
    }

poka's avatar
poka committed
    final KeyType keytype;
    final String keyToImport;
    if (fromMnemonic) {
      keytype = KeyType.mnemonic;
      keyToImport = generatedMnemonic;
    } else {
      keytype = KeyType.keystore;
      keyToImport = jsonKeystore.text.replaceAll("'", "\\'");
    }

poka's avatar
poka committed
    importIsLoading = true;
    notifyListeners();
poka's avatar
poka committed
    if (clipboardData?.text != null) jsonKeystore.text = clipboardData!.text!;
    var json = await sdk.api.keyring
        .importAccount(keyring,
            keyType: keytype,
            key: keyToImport,
            name: derivePath,
            password: password,
poka's avatar
poka committed
            derivePath: derivePath,
            cryptoType: CryptoType.sr25519)
poka's avatar
poka committed
        .catchError((e) {
      importIsLoading = false;
      notifyListeners();
    });
poka's avatar
poka committed
    if (json == null) return '';
poka's avatar
poka committed
    try {
      await sdk.api.keyring.addAccount(
poka's avatar
poka committed
        keyring,
poka's avatar
poka committed
        keyType: keytype,
poka's avatar
poka committed
        acc: json,
        password: password,
poka's avatar
poka committed
      );
poka's avatar
poka committed
      // Clipboard.setData(ClipboardData(text: jsonEncode(acc.toJson())));
poka's avatar
poka committed
    } catch (e) {
poka's avatar
poka committed
      importIsLoading = false;
      notifyListeners();
poka's avatar
poka committed
    }
poka's avatar
poka committed

poka's avatar
poka committed
    importIsLoading = false;
poka's avatar
poka committed
    await Future.delayed(const Duration(milliseconds: 20));
poka's avatar
poka committed
    notifyListeners();
    final bakedAddress = keyring.allAccounts.last.address;
    return bakedAddress!;
poka's avatar
poka committed
  }

  void reload() {
    notifyListeners();
  }
poka's avatar
poka committed

poka's avatar
poka committed
  Future<List<AddressInfo>> getKeyStoreAddress() async {
    List<AddressInfo> result = [];
poka's avatar
poka committed

poka's avatar
poka committed
    // sdk.api.account.unsubscribeBalance();
    for (var element in keyring.allAccounts) {
poka's avatar
poka committed
      // Clipboard.setData(ClipboardData(text: jsonEncode(element)));
poka's avatar
poka committed
      final account = AddressInfo(address: element.address);
      // await sdk.api.account.subscribeBalance(element.address, (p0) {
      //   account.balance = int.parse(p0.freeBalance) / 100;
      // });
      // sdk.api.setting.unsubscribeBestNumber();
      account.balance = await getBalance(element.address!);
poka's avatar
poka committed
      result.add(account);
poka's avatar
poka committed
    }

    return result;
  }

  Future<double> getBalance(String address, {bool isUd = false}) async {
    double balance = 0.0;
    if (nodeConnected) {
      final brutBalance = await sdk.api.account.queryBalance(address);
      balance = int.parse(brutBalance!.freeBalance) / 100;
    }
    return balance;
  }

poka's avatar
poka committed
  Future<double> subscribeBalance(String address, {bool isUd = false}) async {
    double balance = 0.0;
    if (nodeConnected) {
      await sdk.api.account.subscribeBalance(address, (_balance) {
        balance = int.parse(_balance.freeBalance) / 100;
        notifyListeners();
      });
    }

    return balance;
  }

  KeyPairData getKeypair(String address) {
    return keyring.keyPairs.firstWhere((kp) => kp.address == address,
        orElse: (() => KeyPairData()));
  }

  Future<bool> checkPassword(String address, String pass) async {
    final account = getKeypair(address);
    return await sdk.api.keyring.checkPassword(account, pass);
  }

  int getDerivationNumber(String address) {
    final account = getKeypair(address);
    final deriveNbr = account.name!.split('//')[1];
    return int.parse(deriveNbr);
  }

  Future<KeyPairData?> changePassword(
      String address, String passOld, String? passNew) async {
    final account = getKeypair(address);
    keyring.setCurrent(account);

    return await sdk.api.keyring.changePassword(keyring, passOld, passNew);
  }

poka's avatar
poka committed
  Future<void> deleteAllAccounts() async {
    for (var account in keyring.allAccounts) {
      await sdk.api.keyring.deleteAccount(keyring, account);
    }
  }

poka's avatar
poka committed
  Future<void> deleteAccounts(List<String> address) async {
    for (var a in address) {
      final account = getKeypair(a);
      await sdk.api.keyring.deleteAccount(keyring, account);
    }
  }

  Future<String> generateMnemonic({String lang = appLang}) async {
poka's avatar
poka committed
    final gen = await sdk.api.keyring.generateMnemonic(ss58);
poka's avatar
poka committed
    generatedMnemonic = gen.mnemonic!;
poka's avatar
poka committed

poka's avatar
poka committed
    // final res = await importAccount(fromMnemonic: true);
poka's avatar
poka committed
    // await Clipboard.setData(ClipboardData(text: generatedMnemonic));
poka's avatar
poka committed
    return gen.mnemonic!;
poka's avatar
poka committed
  }
poka's avatar
poka committed

poka's avatar
poka committed
  Future<String> setCurrentWallet(WalletData _wallet) async {
    final currentChestNumber = configBox.get('currentChest');
    ChestData _newChestData = chestBox.get(currentChestNumber)!;
    _newChestData.defaultWallet = _wallet.number;
    await chestBox.put(currentChestNumber, _newChestData);

poka's avatar
poka committed
    try {
poka's avatar
poka committed
      final acc = getKeypair(_wallet.address!);
poka's avatar
poka committed
      keyring.setCurrent(acc);
      return acc.address!;
    } catch (e) {
      return (e.toString());
    }
  }

  KeyPairData getCurrentWallet() {
    try {
      final acc = keyring.current;
      return acc;
    } catch (e) {
      return KeyPairData();
    }
  }

  Future<String> pay(
poka's avatar
poka committed
      {required String fromAddress,
      required String destAddress,
      required double amount,
      required String password}) async {
    transactionStatus = '';

poka's avatar
poka committed
    // setCurrentWallet(fromAddress);
poka's avatar
poka committed

poka's avatar
poka committed
    log.d(keyring.current.address);
    log.d(fromAddress);
    log.d(password);
    log.d(await checkPassword(fromAddress, password));

poka's avatar
poka committed
    final sender = TxSenderData(
      keyring.current.address,
      keyring.current.pubKey,
    );
    final txInfo = TxInfoData('balances', 'transfer', sender);
    try {
      final hash = await sdk.api.tx.signAndSend(
        txInfo,
poka's avatar
poka committed
        [destAddress, amount * 100],
poka's avatar
poka committed
        password,
        onStatusChange: (status) {
          log.d('Transaction status: ' + status);
poka's avatar
poka committed
          if (status == 'Ready') {
            transactionStatus = 'sent';
            notifyListeners();
poka's avatar
poka committed
          }
        },
      ).timeout(
        const Duration(seconds: 12),
        onTimeout: () => {},
poka's avatar
poka committed
      );
      log.d(hash.toString());
      if (hash.isEmpty) {
        transactionStatus = 'timeout';
        notifyListeners();

        return 'timeout';
      } else {
        transactionStatus = hash.toString();
        notifyListeners();
        return hash.toString();
      }
    } catch (e) {
      transactionStatus = e.toString();
      notifyListeners();
      return e.toString();
    }
  }

  Future<String> certify(
      String fromAddress, String password, String toAddress) async {
    transactionStatus = '';

poka's avatar
poka committed
    // setCurrentWallet(fromAddress);
    log.d('me: ' + fromAddress);
    log.d('to: ' + toAddress);

    final _myIdtyStatus = await idtyStatus(fromAddress);
    final _toIdtyStatus = await idtyStatus(toAddress);

    log.d(_myIdtyStatus);
    log.d(_toIdtyStatus);

    if (_myIdtyStatus != 'Validated') {
      transactionStatus = 'notMember';
      notifyListeners();
      return 'notMember';
    }

    final sender = TxSenderData(
      keyring.current.address,
      keyring.current.pubKey,
    );
    TxInfoData txInfo;

    if (_toIdtyStatus == 'noid') {
      txInfo = TxInfoData(
        'identity',
        'createIdentity',
        sender,
      );
    } else if (_toIdtyStatus == 'Validated' ||
        _toIdtyStatus == 'ConfirmedByOwner') {
      txInfo = TxInfoData(
        'cert',
        'addCert',
        sender,
      );
    } else {
      transactionStatus = 'cantBeCert';
      notifyListeners();
      return 'cantBeCert';
    }

    log.d('Cert action: ' + txInfo.call!);

    try {
      final hash = await sdk.api.tx
          .signAndSend(
            txInfo,
            [toAddress],
            password,
          )
          .timeout(
            const Duration(seconds: 12),
            onTimeout: () => {},
          );
      log.d(hash);
      if (hash.isEmpty) {
        transactionStatus = 'timeout';
        notifyListeners();

        return 'timeout';
      } else {
        transactionStatus = hash.toString();
        notifyListeners();
        return hash.toString();
      }
poka's avatar
poka committed
    } catch (e) {
      transactionStatus = e.toString();
      notifyListeners();
poka's avatar
poka committed
      return e.toString();
poka's avatar
poka committed
    }
  }
poka's avatar
poka committed
  Future<String> idtyStatus(String address) async {
    //   var tata = await sdk.webView!
    //       .evalJavascript('api.query.system.account("$address")');

    var idtyIndex = await sdk.webView!
        .evalJavascript('api.query.identity.identityIndexOf("$address")');

    if (idtyIndex == null) {
      return 'noid';
    }

    final idtyStatus = await sdk.webView!
        .evalJavascript('api.query.identity.identities($idtyIndex)');

    if (idtyStatus != null) {
      final String _status = idtyStatus['status'];
      log.d(_status);
      return (_status);
    } else {
      return 'expired';
    }
  }

  Future<String> confirmIdentity(
      String fromAddress, String name, String password) async {
poka's avatar
poka committed
    // Confirm identity
poka's avatar
poka committed
    // setCurrentWallet(fromAddress);
    log.d('me: ' + keyring.current.address!);
poka's avatar
poka committed

    final sender = TxSenderData(
      keyring.current.address,
      keyring.current.pubKey,
    );

    final txInfo = TxInfoData(
      'identity',
      'confirmIdentity',
      sender,
    );

    try {
poka's avatar
poka committed
      final result = await sdk.api.tx.signAndSend(
poka's avatar
poka committed
        txInfo,
        [name],
        password,
      );
poka's avatar
poka committed
      log.d(result);
poka's avatar
poka committed
      return 'confirmed';
    } on Exception catch (e) {
      log.e(e);
      return e.toString();
    }
  }

  Future<bool> isMember(String address) async {
    return await idtyStatus(address) == 'Validated';
  }

  Future<String> derive(
poka's avatar
poka committed
      BuildContext context, String address, int number, String password) async {
    final keypair = getKeypair(address);
poka's avatar
poka committed
    final seedMap =
        await keyring.store.getDecryptedSeed(keypair.pubKey, password);

    if (seedMap?['type'] != 'mnemonic') return '';
    final List seedList = seedMap!['seed'].split('//');
poka's avatar
poka committed
    generatedMnemonic = seedList[0];
poka's avatar
poka committed
    return await importAccount(
        fromMnemonic: true, derivePath: '//$number', password: password);
poka's avatar
poka committed

  Future<bool> isMnemonicValid(String mnemonic) async {
    // Needed for bad encoding of UTF-8
    mnemonic = mnemonic.replaceAll('é', 'é');
    mnemonic = mnemonic.replaceAll('è', 'è');

    return await sdk.api.keyring.checkMnemonicValid(mnemonic);
  }

  String? getConnectedEndpoint() {
    return sdk.api.connectedNode?.endpoint;
  }
poka's avatar
poka committed
}

void snack(BuildContext context, String message, {int duration = 2}) {
  final snackBar =
      SnackBar(content: Text(message), duration: Duration(seconds: duration));
  ScaffoldMessenger.of(context).showSnackBar(snackBar);
}

class AddressInfo {
  final String? address;
  double balance;

  AddressInfo({@required this.address, this.balance = 0});
}

void snackNode(BuildContext context, bool isConnected) {
  String _message;
  if (!isConnected) {
    _message =
poka's avatar
poka committed
        "Aucun noeud Duniter disponible, veuillez réessayer ultérieurement:\n${configBox.get('endpoint').first}";
    SubstrateSdk _sub = Provider.of<SubstrateSdk>(context, listen: false);

        "Vous êtes connecté au noeud\n${_sub.getConnectedEndpoint()!.split('//')[1]}";
poka's avatar
poka committed
  final snackBar = SnackBar(
      padding: const EdgeInsets.all(20),
      content: Text(_message, style: const TextStyle(fontSize: 16)),
poka's avatar
poka committed
      duration: const Duration(seconds: 4));
  ScaffoldMessenger.of(context).showSnackBar(snackBar);
}

poka's avatar
poka committed
String getShortPubkey(String pubkey) {
poka's avatar
poka committed
  String pubkeyShort = truncate(pubkey, 7,
poka's avatar
poka committed
          omission: String.fromCharCode(0x2026),
          position: TruncatePosition.end) +
poka's avatar
poka committed
      truncate(pubkey, 6, omission: "", position: TruncatePosition.start);
poka's avatar
poka committed
  return pubkeyShort;
poka's avatar
poka committed

class PasswordException implements Exception {
  String cause;
  PasswordException(this.cause);
}