Files
openethereum/js/src/util/wallets/foundation-wallet.js
Nicolas Gotchac b0f89becfd UI backports (#6332)
* Time should not contribue to overall status. (#6276)

* Add warning to web browser and fix links. (#6232)

* Extension fixes (#6284)

* Fix token symbols in extension.

* Allow connections from firefox extension.

* Add support for ConsenSys multisig wallet (#6153)

* First draft of ConsenSys wallet

* Fix transfer store // WIP Consensys Wallet

* Rename walletABI JSON file

* Fix linting

* Fix wrong daylimit in wallet modal

* Confirm/Revoke ConsensysWallet txs

* Linting

* Change of settings for the Multisig Wallet
2017-08-18 16:51:17 +02:00

501 lines
14 KiB
JavaScript

// Copyright 2015-2017 Parity Technologies (UK) Ltd.
// This file is part of Parity.
// Parity is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// Parity is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
import { range, uniq } from 'lodash';
import Abi from '~/abi';
import Contract from '~/api/contract';
import { bytesToHex, toHex } from '~/api/util/format';
import { validateAddress } from '~/util/validation';
import WalletAbi from '~/contracts/abi/foundation-multisig-wallet.json';
import OldWalletAbi from '~/contracts/abi/old-wallet.json';
import PendingContracts from './pending-contracts';
import {
UPDATE_OWNERS,
UPDATE_REQUIRE,
UPDATE_TRANSACTIONS,
UPDATE_CONFIRMATIONS
} from './updates';
const WALLET_CONTRACT = new Contract({}, WalletAbi);
const WALLET_ABI = new Abi(WalletAbi);
const OLD_WALLET_ABI = new Abi(OldWalletAbi);
const walletEvents = WALLET_ABI.events.reduce((events, event) => {
events[event.name] = event;
return events;
}, {});
const oldWalletEvents = OLD_WALLET_ABI.events.reduce((events, event) => {
events[event.name] = event;
return events;
}, {});
const WalletSignatures = {
OwnerChanged: toHex(walletEvents.OwnerChanged.signature),
OwnerAdded: toHex(walletEvents.OwnerAdded.signature),
OwnerRemoved: toHex(walletEvents.OwnerRemoved.signature),
RequirementChanged: toHex(walletEvents.RequirementChanged.signature),
Confirmation: toHex(walletEvents.Confirmation.signature),
Revoke: toHex(walletEvents.Revoke.signature),
Deposit: toHex(walletEvents.Deposit.signature),
SingleTransact: toHex(walletEvents.SingleTransact.signature),
MultiTransact: toHex(walletEvents.MultiTransact.signature),
ConfirmationNeeded: toHex(walletEvents.ConfirmationNeeded.signature),
Old: {
SingleTransact: toHex(oldWalletEvents.SingleTransact.signature),
MultiTransact: toHex(oldWalletEvents.MultiTransact.signature)
}
};
export default class FoundationWalletUtils {
static fetchConfirmations (walletContract, operation, _owners = null) {
const ownersPromise = _owners
? Promise.resolve(_owners)
: FoundationWalletUtils.fetchOwners(walletContract);
return ownersPromise
.then((owners) => {
const promises = owners.map((owner) => {
return walletContract.instance.hasConfirmed.call({}, [ operation, owner ]);
});
return Promise
.all(promises)
.then((data) => {
return owners.filter((_, index) => data[index]);
});
});
}
static fetchDailylimit (walletContract) {
const walletInstance = walletContract.instance;
return Promise
.all([
walletInstance.m_dailyLimit.call(),
walletInstance.m_spentToday.call(),
walletInstance.m_lastDay.call()
])
.then(([ limit, spent, last ]) => ({
limit, spent, last
}));
}
static fetchOwners (walletContract) {
const walletInstance = walletContract.instance;
return walletInstance
.m_numOwners.call()
.then((mNumOwners) => {
const promises = range(mNumOwners.toNumber())
.map((idx) => walletInstance.getOwner.call({}, [ idx ]));
return Promise
.all(promises)
.then((_owners) => {
const owners = validateOwners(_owners);
// If all owners are the zero account : must be Mist wallet contract
if (!owners) {
return fetchMistOwners(walletContract, mNumOwners.toNumber());
}
return owners;
});
});
}
static fetchPendingTransactions (walletContract, cache = {}) {
const { owners, transactions } = cache;
return walletContract
.instance
.ConfirmationNeeded
.getAllLogs()
.then((logs) => {
return logs.map((log) => ({
initiator: log.params.initiator.value,
to: log.params.to.value,
data: log.params.data.value,
value: log.params.value.value,
operation: bytesToHex(log.params.operation.value),
transactionIndex: log.transactionIndex,
transactionHash: log.transactionHash,
blockNumber: log.blockNumber,
confirmedBy: []
}));
})
.then((logs) => {
return logs.sort((logA, logB) => {
const comp = logA.blockNumber.comparedTo(logB.blockNumber);
if (comp !== 0) {
return comp;
}
return logA.transactionIndex.comparedTo(logB.transactionIndex);
});
})
.then((pendingTxs) => {
if (pendingTxs.length === 0) {
return pendingTxs;
}
// Only fetch confirmations for operations not
// yet confirmed (ie. not yet a transaction)
if (transactions) {
const operations = transactions
.filter((t) => t.operation)
.map((t) => t.operation);
return pendingTxs.filter((pendingTx) => {
return !operations.includes(pendingTx.operation);
});
}
return pendingTxs;
})
.then((pendingTxs) => {
const promises = pendingTxs.map((tx) => {
return FoundationWalletUtils
.fetchConfirmations(walletContract, tx.operation, owners)
.then((confirmedBy) => {
tx.confirmedBy = confirmedBy;
return tx;
});
});
return Promise.all(promises);
});
}
static fetchRequire (wallet) {
return wallet.instance.m_required.call();
}
static fetchTransactions (walletContract) {
const { api } = walletContract;
return walletContract
.getAllLogs({
topics: [ [
WalletSignatures.SingleTransact,
WalletSignatures.MultiTransact,
WalletSignatures.Deposit,
WalletSignatures.Old.SingleTransact,
WalletSignatures.Old.MultiTransact
] ]
})
.then((logs) => {
const transactions = logs.map((log) => {
const signature = toHex(log.topics[0]);
const value = log.params.value.value;
const from = signature === WalletSignatures.Deposit
? log.params['_from'].value
: walletContract.address;
const to = signature === WalletSignatures.Deposit
? walletContract.address
: log.params.to.value;
const transaction = {
transactionHash: log.transactionHash,
blockNumber: log.blockNumber,
from, to, value
};
if (log.params.created && log.params.created.value && !/^(0x)?0*$/.test(log.params.created.value)) {
transaction.creates = log.params.created.value;
delete transaction.to;
}
if (log.params.operation) {
transaction.operation = bytesToHex(log.params.operation.value);
checkPendingOperation(api, log, transaction.operation);
}
if (log.params.data) {
transaction.data = log.params.data.value;
}
return transaction;
});
return transactions;
});
}
static getChangeMethod (api, address, change) {
const wallet = new Contract(api, WalletAbi).at(address);
const walletInstance = wallet.instance;
if (change.type === 'require') {
return {
method: walletInstance.changeRequirement,
values: [ change.value ]
};
}
if (change.type === 'dailylimit') {
return {
method: walletInstance.setDailyLimit,
values: [ change.value ]
};
}
if (change.type === 'add_owner') {
return {
method: walletInstance.addOwner,
values: [ change.value ]
};
}
if (change.type === 'change_owner') {
return {
method: walletInstance.changeOwner,
values: [ change.value.from, change.value.to ]
};
}
if (change.type === 'remove_owner') {
return {
method: walletInstance.removeOwner,
values: [ change.value ]
};
}
}
static getModifyOperationMethod (modification) {
switch (modification) {
case 'confirm':
return 'confirm';
case 'revoke':
return 'revoke';
default:
return '';
}
}
static getSubmitMethod () {
return 'execute';
}
static getWalletContract (api) {
return new Contract(api, WalletAbi);
}
static getWalletSignatures () {
return WalletSignatures;
}
static isWallet (api, address) {
const walletContract = new Contract(api, WalletAbi);
return walletContract
.at(address)
.instance
.m_numOwners
.call()
.then((result) => {
if (!result || result.equals(0)) {
return false;
}
return true;
});
}
static logToUpdate (log) {
const eventSignature = toHex(log.topics[0]);
switch (eventSignature) {
case WalletSignatures.OwnerChanged:
case WalletSignatures.OwnerAdded:
case WalletSignatures.OwnerRemoved:
return { [ UPDATE_OWNERS ]: true };
case WalletSignatures.RequirementChanged:
return { [ UPDATE_REQUIRE ]: true };
case WalletSignatures.ConfirmationNeeded:
case WalletSignatures.Confirmation:
case WalletSignatures.Revoke:
const parsedLog = WALLET_CONTRACT.parseEventLogs([ log ])[0];
const operation = bytesToHex(parsedLog.params.operation.value);
return { [ UPDATE_CONFIRMATIONS ]: operation };
case WalletSignatures.Deposit:
case WalletSignatures.SingleTransact:
case WalletSignatures.MultiTransact:
case WalletSignatures.Old.SingleTransact:
case WalletSignatures.Old.MultiTransact:
return { [ UPDATE_TRANSACTIONS ]: true };
default:
return {};
}
}
static parseLogs (api, logs = []) {
const walletContract = new Contract(api, WalletAbi);
return walletContract.parseEventLogs(logs);
}
static parseTransactionLogs (api, options, rawLogs) {
const { metadata } = options;
const address = options.from;
const logs = FoundationWalletUtils.parseLogs(api, rawLogs);
const confirmationLog = logs.find((log) => log.event === 'ConfirmationNeeded');
const transactionLog = logs.find((log) => log.event === 'SingleTransact');
if (!confirmationLog && !transactionLog) {
return null;
}
// Confirmations are needed from the other owners
if (confirmationLog) {
const operationHash = bytesToHex(confirmationLog.params.operation.value);
// Add the contract to pending contracts
PendingContracts.addPendingContract(address, operationHash, metadata);
return { pending: true };
}
return { contractAddress: transactionLog.params.created.value };
}
}
function checkPendingOperation (api, log, operation) {
const pendingContracts = PendingContracts.getPendingContracts();
// Add the pending contract to the contracts
if (pendingContracts[operation]) {
const { metadata } = pendingContracts[operation];
const contractName = metadata.name;
metadata.blockNumber = log.blockNumber;
// The contract creation might not be in the same log,
// but must be in the same transaction (eg. Contract creation
// from Wallet within a Wallet)
api.eth
.getTransactionReceipt(log.transactionHash)
.then((transactionReceipt) => {
const transactionLogs = FoundationWalletUtils.parseLogs(api, transactionReceipt.logs);
const creationLog = transactionLogs.find((log) => {
return log.params.created && !/^(0x)?0*$/.test(log.params.created.value);
});
if (!creationLog) {
return false;
}
const contractAddress = creationLog.params.created.value;
return Promise
.all([
api.parity.setAccountName(contractAddress, contractName),
api.parity.setAccountMeta(contractAddress, metadata)
])
.then(() => {
PendingContracts.removePendingContract(operation);
});
})
.catch((error) => {
console.error('adding wallet contract', error);
});
}
}
function fetchMistOwners (walletContract, mNumOwners) {
const walletAddress = walletContract.address;
return getMistOwnersOffset(walletContract)
.then((result) => {
if (!result || result.offset === -1) {
return [];
}
const owners = [ result.address ];
if (mNumOwners === 1) {
return owners;
}
const initOffset = result.offset + 1;
let promise = Promise.resolve();
range(initOffset, initOffset + mNumOwners - 1).forEach((offset) => {
promise = promise
.then(() => {
return walletContract.api.eth.getStorageAt(walletAddress, offset);
})
.then((result) => {
const resultAddress = '0x' + (result || '').slice(-40);
const { address } = validateAddress(resultAddress);
owners.push(address);
});
});
return promise.then(() => owners);
});
}
function getMistOwnersOffset (walletContract, offset = 3) {
return walletContract.api.eth
.getStorageAt(walletContract.address, offset)
.then((result) => {
if (result && !/^(0x)?0*$/.test(result)) {
const resultAddress = '0x' + result.slice(-40);
const { address, addressError } = validateAddress(resultAddress);
if (!addressError) {
return { offset, address };
}
}
if (offset >= 100) {
return { offset: -1 };
}
return getMistOwnersOffset(walletContract, offset + 1);
});
}
function validateOwners (owners) {
const uniqOwners = uniq(owners);
// If all owners are the zero account : must be Mist wallet contract
if (uniqOwners.length === 1 && /^(0x)?0*$/.test(owners[0])) {
return null;
}
return uniqOwners;
}