285 lines
12 KiB
HTML
285 lines
12 KiB
HTML
<html>
|
|
<head>
|
|
<title>Ethereum Classic Split Tool</title>
|
|
<link rel="stylesheet" type="text/css" href="../style.css">
|
|
</head>
|
|
<body>
|
|
<div class="centerer">
|
|
<div class="centered">
|
|
<h1>Split Ether Classic</h1>
|
|
<hr />
|
|
<h2>What does this tool do?</h2>
|
|
<p>
|
|
This tool will take a <i>geth</i> (or crowdsale) JSON wallet, decrypt it and
|
|
send all its funds to <a href="http://etherscan.io/address/0x3474627d4f63a678266bc17171d87f8570936622#code">this contract</a>,
|
|
which will:
|
|
</p>
|
|
<ul>
|
|
<li>On the ETH branch — send the funds back to the original address</li>
|
|
<li>On the ETC branch — send the funds to the provided target address (for example, a <a href="https://www.poloniex.com">Poloniex</a> deposit address)</li>
|
|
</ul>
|
|
<br />
|
|
<h3>Disclaimer:</h3>
|
|
<p>
|
|
I threw this together in couple of hours, mainly to split my own ether
|
|
and test my <i>ethers-wallet</i> library (which is still missing features
|
|
and is itself not ready for production use). Testing has been fairly minimal
|
|
beyond trying it on a few wallets. <b>Use this at your own risk.</b>
|
|
</p>
|
|
<hr />
|
|
<h2>Check Current ETC Balance</h2>
|
|
<table>
|
|
<tr>
|
|
<th>ETC Address:</th>
|
|
<td><input type="text" id="checkAddress" /></td>
|
|
</tr>
|
|
<tr>
|
|
<td> </td>
|
|
<td>
|
|
<div id="submitCheck" class="submit disable">Check Classic Ether Balance</div>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
<hr />
|
|
<h2>Split ETC/ETH</h2>
|
|
<table>
|
|
<tr>
|
|
<th>JSON Wallet:</th>
|
|
<td><div class="file" id="drop">Drop JSON wallet file here</div><input type="file" id="json" /></td>
|
|
</tr>
|
|
<tr>
|
|
<th>Password:</th>
|
|
<td><input type="password" id="password" /></td>
|
|
</tr>
|
|
<tr>
|
|
<th>Target ETC Address:</th>
|
|
<td><input type="text" id="targetAddress" /></td>
|
|
</tr>
|
|
<tr>
|
|
<td> </td>
|
|
<td>
|
|
<div id="submitSplit" class="submit disable">Split Classic Ether</div>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
|
|
<script type="text/javascript" src="../../dist/ethers-wallet.js"></script>
|
|
<script type="text/javascript">
|
|
var submitCheck = document.getElementById('submitCheck');
|
|
var submitSplit = document.getElementById('submitSplit');
|
|
|
|
var inputJson = document.getElementById('json');
|
|
var inputCheckAddress = document.getElementById('checkAddress')
|
|
var inputTargetAddress = document.getElementById('targetAddress')
|
|
var inputPassword = document.getElementById('password');
|
|
|
|
var targetDrop = document.getElementById('drop');
|
|
|
|
var provider = new Wallet.providers.HttpProvider('https://linode-newark.ethers.ws:8002');
|
|
|
|
var contractAddress = '0x3474627d4f63a678266bc17171d87f8570936622';
|
|
var contractAbi = JSON.parse('[{"constant":false,"inputs":[{"name":"balance","type":"uint256"}],"name":"claimDonations","outputs":[],"type":"function"},{"constant":true,"inputs":[],"name":"isClassic","outputs":[{"name":"","type":"bool"}],"type":"function"},{"constant":false,"inputs":[{"name":"classicAddress","type":"address"}],"name":"split","outputs":[],"type":"function"},{"inputs":[],"type":"constructor"}]');
|
|
|
|
submitCheck.onclick = function() {
|
|
if (submitCheck.classList.contains('disable')) { return; }
|
|
|
|
var address = document.getElementById('checkAddress').value;
|
|
try {
|
|
address = Wallet.getAddress(address);
|
|
} catch (error) {
|
|
console.log(error);
|
|
alert('invalid address');
|
|
return;
|
|
}
|
|
provider.getBalance(address, 'latest').then(function(balance) {
|
|
alert('Balance: ' + Wallet.formatEther(balance, {commify: true}) + ' ' + Wallet.etherSymbol);
|
|
}, function(error) {
|
|
if (error) {
|
|
console.log(error);
|
|
alert('Faied to get balance');
|
|
}
|
|
});
|
|
|
|
console.log('check', address);
|
|
}
|
|
|
|
submitSplit.onclick = function() {
|
|
if (submitSplit.classList.contains('disable')) { return; }
|
|
|
|
function done() {
|
|
submitSplit.textContent = 'Split Classic Ether';
|
|
inputJson.readOnly = false;
|
|
inputPassword.readOnly = false;
|
|
inputTargetAddress.readOnly = false;
|
|
checkSplit();
|
|
}
|
|
|
|
inputJson.readOnly = true;
|
|
inputPassword.readOnly = true;
|
|
inputTargetAddress.readOnly = true;
|
|
|
|
submitSplit.classList.add('disable');
|
|
|
|
var files = inputJson.files;
|
|
if (files.length !== 1) {
|
|
alert('No wallet found');
|
|
return done();
|
|
}
|
|
|
|
var password = new Wallet.utils.Buffer(inputPassword.value, 'utf8');
|
|
|
|
var targetAddress = document.getElementById('targetAddress').value;
|
|
try {
|
|
targetAddress = Wallet.getAddress(targetAddress);
|
|
} catch (error) {
|
|
console.log(error);
|
|
alert('invalid target address');
|
|
return done();
|
|
}
|
|
|
|
function processWallet(wallet) {
|
|
console.log(wallet);
|
|
if (wallet.address === targetAddress) {
|
|
alert('Wallet address and target address cannot be the same.');
|
|
return done();
|
|
}
|
|
|
|
submitSplit.textContent = 'Decrypted \u2014 Processing (please wait)';
|
|
|
|
var contract = wallet.getContract(contractAddress, contractAbi);
|
|
var data = contract.interface.split(targetAddress).data;
|
|
|
|
var transaction = {
|
|
to: contractAddress,
|
|
data: data
|
|
};
|
|
|
|
Promise.all([
|
|
provider.getBalance(wallet.address, 'latest'),
|
|
provider.getTransactionCount(wallet.address, 'latest'),
|
|
provider.getGasPrice(),
|
|
// provider.estimateGas(transaction)
|
|
]).then(function(results) {
|
|
var balance = results[0];
|
|
var transactionCount = results[1];
|
|
var gasPrice = results[2];
|
|
var gasEstimate = new Wallet.utils.BN('29735');
|
|
//results[3].add(new Wallet.utils.BN(21000));
|
|
|
|
transaction.gasPrice = '0x' + gasPrice.toString(16);
|
|
transaction.nonce = transactionCount;
|
|
transaction.gasLimit = gasEstimate;
|
|
|
|
var toSend = balance.sub(gasPrice.mul(gasEstimate));
|
|
transaction.value = toSend;
|
|
|
|
console.log(transaction);
|
|
|
|
var accept = confirm('Balance: ' + Wallet.formatEther(balance, {commify: true}) + ' ' + Wallet.etherSymbol + '. Are you sure you want to split Classic Ether to ' + targetAddress + '?');
|
|
if (accept) {
|
|
var signedTransaction = wallet.sign(transaction);
|
|
function showError(error) {
|
|
console.log(error);
|
|
done();
|
|
alert('Error sending transaction');
|
|
}
|
|
provider.sendTransaction(signedTransaction).then(function(txid) {
|
|
if (txid.match(/^0x0+$/)) {
|
|
showError(new Error('txid was zero'));
|
|
return;
|
|
}
|
|
done();
|
|
console.log('txid:' + txid);
|
|
alert('Success \u2014 ' + txid);
|
|
}, function(error) {
|
|
showError(error);
|
|
});
|
|
} else {
|
|
done();
|
|
}
|
|
}, function(error) {
|
|
done();
|
|
console.log(error);
|
|
alert('Error: Faied to fetch info');
|
|
});
|
|
}
|
|
|
|
var fileReader = new FileReader();
|
|
fileReader.onload = function(e) {
|
|
var json = e.target.result;
|
|
if (Wallet.isCrowdsaleWallet(json)) {
|
|
var wallet = Wallet.decryptCrowdsale(json, password);
|
|
processWallet(wallet);
|
|
} else if (Wallet.isValidWallet(json)) {
|
|
Wallet.decrypt(json, password, function(error, wallet, progress) {
|
|
if (error) {
|
|
done();
|
|
console.log(error);
|
|
if (error.message === 'invalid password') {
|
|
alert('Wrong Password');
|
|
} else {
|
|
alert('Error Decrypting Wallet');
|
|
}
|
|
} else if (wallet) {
|
|
processWallet(wallet);
|
|
} else {
|
|
submitSplit.textContent = 'Decrypting \u2014 ' + (parseInt(progress * 100) + '%');
|
|
}
|
|
});
|
|
} else {
|
|
alert('unknown walet format');
|
|
done();
|
|
}
|
|
}
|
|
|
|
fileReader.readAsText(files[0]);
|
|
}
|
|
|
|
inputCheckAddress.oninput = function() {
|
|
try {
|
|
Wallet.getAddress(inputCheckAddress.value);
|
|
submitCheck.classList.remove('disable');
|
|
} catch (error) {
|
|
submitCheck.classList.add('disable');
|
|
}
|
|
}
|
|
|
|
function checkSplit() {
|
|
if (!inputJson.files || inputJson.files.length !== 1) {
|
|
submitSplit.classList.add('disable');
|
|
return;
|
|
}
|
|
targetDrop.textContent = inputJson.files[0].name;
|
|
|
|
try {
|
|
Wallet.getAddress(inputTargetAddress.value);
|
|
} catch (error) {
|
|
submitSplit.classList.add('disable');
|
|
return;
|
|
}
|
|
|
|
submitSplit.classList.remove('disable');
|
|
}
|
|
|
|
inputTargetAddress.oninput = checkSplit;
|
|
inputJson.onchange = checkSplit;
|
|
|
|
inputJson.addEventListener('dragover', function(event) {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
targetDrop.classList.add('highlight');
|
|
}, true);
|
|
|
|
inputJson.addEventListener('drop', function(event) {
|
|
targetDrop.classList.remove('highlight');
|
|
}, true);
|
|
|
|
|
|
var check
|
|
</script>
|
|
</body>
|
|
</html>
|
|
|