All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.klaytn.caver.wallet.KlayWalletUtils Maven / Gradle / Ivy

/*
 * Modifications copyright 2019 The caver-java Authors
 * Copyright 2016 Conor Svensson
 *
 * Licensed under the Apache License, Version 2.0 (the “License”);
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an “AS IS” BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *
 * This file is derived from web3j/core/src/main/java/org/web3j/crypto/WalletUtils.java (2019/06/13).
 * Modified and improved for the caver-java development.
 */

package com.klaytn.caver.wallet;

import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.klaytn.caver.crypto.KlayCredentials;
import com.klaytn.caver.utils.SecureRandomUtils;
import org.web3j.crypto.*;
import org.web3j.utils.Numeric;

import java.io.File;
import java.io.IOException;
import java.security.InvalidAlgorithmParameterException;
import java.security.NoSuchAlgorithmException;
import java.security.NoSuchProviderException;
import java.security.SecureRandom;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.util.regex.Pattern;

import static org.web3j.crypto.Hash.sha256;

/**
 * @deprecated This class is deprecated since caver-java:1.5.0
 */
public class KlayWalletUtils {

    public static final int ADDRESS_HEX_SIZE = 40;
    private static final int PRIVATE_KEY_HEX_SIZE = 64;

    public static final String CHECKSUM = "0x00";

    private static final ObjectMapper objectMapper = new ObjectMapper();
    private static final SecureRandom secureRandom = SecureRandomUtils.secureRandom();

    private static final Pattern HEX_STRING = Pattern.compile("^[0-9A-Fa-f]+$");

    static {
        objectMapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true);
        objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
    }

    public static String generateFullNewWalletFile(String address, String password, File destinationDirectory)
            throws NoSuchAlgorithmException, NoSuchProviderException,
            InvalidAlgorithmParameterException, CipherException, IOException {

        return generateNewWalletFile(address, password, destinationDirectory, true);
    }

    public static String generateFullNewWalletFile(String password, File destinationDirectory)
            throws NoSuchAlgorithmException, NoSuchProviderException,
            InvalidAlgorithmParameterException, CipherException, IOException {

        return generateNewWalletFile("", password, destinationDirectory, true);
    }

    public static String generateNewWalletFile(String address, String password, File destinationDirectory)
            throws CipherException, InvalidAlgorithmParameterException,
            NoSuchAlgorithmException, NoSuchProviderException, IOException {

        return generateNewWalletFile(address, password, destinationDirectory, false);
    }

    public static String generateNewWalletFile(String password, File destinationDirectory)
            throws CipherException, InvalidAlgorithmParameterException,
            NoSuchAlgorithmException, NoSuchProviderException, IOException {

        return generateNewWalletFile("", password, destinationDirectory, false);
    }

    public static String generateNewWalletFile(
            String address, String password, File destinationDirectory, boolean useFullScrypt)
            throws CipherException, IOException, InvalidAlgorithmParameterException,
            NoSuchAlgorithmException, NoSuchProviderException {

        ECKeyPair ecKeyPair = Keys.createEcKeyPair();
        return generateWalletFile(address, password, ecKeyPair, destinationDirectory, useFullScrypt);
    }

    private static String generateWalletFile(
            String address, String password, ECKeyPair ecKeyPair, File destinationDirectory, boolean useFullScrypt)
            throws CipherException, IOException {

        WalletFile walletFile;
        if (useFullScrypt) {
            walletFile = Wallet.createFull(password, ecKeyPair, address);
        } else {
            walletFile = Wallet.createStandard(password, ecKeyPair, address);
        }

        String fileName = getWalletFileName(walletFile);
        File destination = new File(destinationDirectory, fileName);

        objectMapper.writeValue(destination, walletFile);

        return destination.getAbsolutePath();
    }

    /**
     * Generates a BIP-39 compatible Ethereum wallet. The private key for the wallet can
     * be calculated using following algorithm:
     * 
     *     Key = SHA-256(BIP_39_SEED(mnemonic, password))
     * 
* * @param address address * @param password Will be used for both wallet encryption and passphrase for BIP-39 seed * @param destinationDirectory The directory containing the wallet * @return A BIP-39 compatible Ethereum wallet * @throws CipherException if the underlying cipher is not available * @throws IOException if the destination cannot be written to */ public static Bip39Wallet generateBip39Wallet(String address, String password, File destinationDirectory) throws CipherException, IOException { byte[] initialEntropy = new byte[16]; secureRandom.nextBytes(initialEntropy); String mnemonic = MnemonicUtils.generateMnemonic(initialEntropy); byte[] seed = MnemonicUtils.generateSeed(mnemonic, password); ECKeyPair privateKey = ECKeyPair.create(sha256(seed)); String walletFile = generateWalletFile(address, password, privateKey, destinationDirectory, false); return new Bip39Wallet(walletFile, mnemonic); } public static KlayCredentials loadCredentials(String password, String source) throws IOException, CipherException { return loadCredentials(password, new File(source)); } public static KlayCredentials loadCredentials(String password, File source) throws IOException, CipherException { WalletFile walletFile = objectMapper.readValue(source, WalletFile.class); return Wallet.decrypt(password, walletFile); } public static KlayCredentials loadCredentials(String klaytnWalletKey) { return KlayCredentials.createWithKlaytnWalletKey(klaytnWalletKey); } public static KlayCredentials loadBip39Credentials(String password, String mnemonic) { byte[] seed = MnemonicUtils.generateSeed(mnemonic, password); ECKeyPair ecKeyPair = ECKeyPair.create(sha256(seed)); String address = Numeric.prependHexPrefix(Keys.getAddress(ecKeyPair)); return KlayCredentials.create(ECKeyPair.create(sha256(seed)), address); } private static String getWalletFileName(WalletFile walletFile) { DateTimeFormatter format = DateTimeFormatter.ofPattern( "'UTC--'yyyy-MM-dd'T'HH-mm-ss.nVV'--'"); ZonedDateTime now = ZonedDateTime.now(ZoneOffset.UTC); return now.format(format) + walletFile.getAddress() + ".json"; } public static String getBaobabKeyDirectory() { String directory = String.format("%s%sbaobab", getDefaultKeyDirectory(), File.separator); createDirectoryIfNotPresent(directory); return directory; } public static String getMainnetKeyDirectory() { String directory = String.format("%s%smainnet", getDefaultKeyDirectory(), File.separator); createDirectoryIfNotPresent(directory); return directory; } @Deprecated public static boolean isValidPrivateKey(String privateKey) { String cleanPrivateKey = Numeric.cleanHexPrefix(privateKey); return cleanPrivateKey.length() <= PRIVATE_KEY_HEX_SIZE && HEX_STRING.matcher(cleanPrivateKey).matches(); } @Deprecated public static boolean isValidAddress(String input) { String cleanInput = Numeric.cleanHexPrefix(input); try { Numeric.toBigIntNoPrefix(cleanInput); } catch (NumberFormatException e) { return false; } return cleanInput.length() == ADDRESS_HEX_SIZE && HEX_STRING.matcher(cleanInput).matches(); } public static String getDefaultKeyDirectory() { String defaultDirectory = getDefaultKeyDirectory(System.getProperty("os.name")); createDirectoryIfNotPresent(defaultDirectory); return getDefaultKeyDirectory(System.getProperty("os.name")); } private static void createDirectoryIfNotPresent(String defaultDirectory) { File file = new File(defaultDirectory); if (!file.exists()) { if (!file.mkdirs()) { throw new RuntimeException( String.format("Default key directory is not created", defaultDirectory)); } } } static String getDefaultKeyDirectory(String osName1) { String osName = osName1.toLowerCase(); if (osName.startsWith("mac")) { return String.format("%s%sLibrary%sKlaytn%skeystore", System.getProperty("user.home"), File.separator, File.separator, File.separator); } else if (osName.startsWith("win")) { return String.format("%s%sKlaytn%skeystore", System.getenv("APPDATA"), File.separator, File.separator); } else { return String.format("%s%s.Klaytn%skeystore", System.getProperty("user.home"), File.separator, File.separator); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy