* 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
501 lines
14 KiB
JavaScript
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;
|
|
}
|