// 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 . import validator from 'validator'; import { api } from '../parity'; import { getTokenTotalSupply } from '../utils'; const { isHex, isAddressValid, toChecksumAddress } = api.util; export const ADDRESS_TYPE = 'ADDRESS_TYPE'; export const TOKEN_ADDRESS_TYPE = 'TOKEN_ADDRESS_TYPE'; export const SIMPLE_TOKEN_ADDRESS_TYPE = 'SIMPLE_TOKEN_ADDRESS_TYPE'; export const TLA_TYPE = 'TLA_TYPE'; export const SIMPLE_TLA_TYPE = 'SIMPLE_TLA_TYPE'; export const UINT_TYPE = 'UINT_TYPE'; export const DECIMAL_TYPE = 'DECIMAL_TYPE'; export const STRING_TYPE = 'STRING_TYPE'; export const HEX_TYPE = 'HEX_TYPE'; export const URL_TYPE = 'URL_TYPE'; export const ERRORS = { invalidTLA: 'The TLA should be 3 characters long', invalidUint: 'Please enter a non-negative integer', invalidDecimal: 'Please enter a value between 0 and 18', invalidString: 'Please enter at least a character', invalidAccount: 'Please select an account to transact with', invalidRecipient: 'Please select an account to send to', invalidAddress: 'The address is not in the correct format', invalidTokenAddress: 'The address is not a regular token contract address', invalidHex: 'Please enter an hexadecimal string (digits and letters from a to z)', invalidAmount: 'Please enter a positive amount > 0', invalidTotal: 'The amount is greater than the availale balance', tlaAlreadyTaken: 'This TLA address is already registered', addressAlreadyTaken: 'This Token address is already registered', invalidURL: 'Please enter a valid URL' }; const validateAddress = (address) => { if (!isAddressValid(address)) { return { error: ERRORS.invalidAddress, valid: false }; } return { value: toChecksumAddress(address), error: null, valid: true }; }; const validateTokenAddress = (address, contract, simple) => { const addressValidation = validateAddress(address); if (!addressValidation.valid) { return addressValidation; } if (simple) { return addressValidation; } return getTokenTotalSupply(address) .then(balance => { if (balance === null || balance.equals(0)) { return { error: ERRORS.invalidTokenAddress, valid: false }; } return contract.instance .fromAddress.call({}, [ address ]) .then(([id, tla, base, name, owner]) => { if (owner !== '0x0000000000000000000000000000000000000000') { return { error: ERRORS.addressAlreadyTaken, valid: false }; } }); }) .then((result) => { if (result) { return result; } return addressValidation; }); }; const validateTLA = (tla, contract, simple) => { if (tla.toString().length !== 3) { return { error: ERRORS.invalidTLA, valid: false }; } const fTLA = tla.toString().toUpperCase(); if (simple) { return { value: fTLA, error: null, valid: true }; } return contract.instance .fromTLA.call({}, [ fTLA ]) .then(([id, address, base, name, owner]) => { if (owner !== '0x0000000000000000000000000000000000000000') { return { error: ERRORS.tlaAlreadyTaken, valid: false }; } }) .then((result) => { if (result) { return result; } return { value: fTLA, error: null, valid: true }; }); }; const validateUint = (uint) => { if (!/^\d+$/.test(uint) || parseInt(uint) <= 0) { return { error: ERRORS.invalidUint, valid: false }; } return { value: parseInt(uint), error: null, valid: true }; }; const validateDecimal = (decimal) => { if (!/^\d+$/.test(decimal) || parseInt(decimal) < 0 || parseInt(decimal) > 18) { return { error: ERRORS.invalidDecimal, valid: false }; } return { value: parseInt(decimal), error: null, valid: true }; }; const validateString = (string) => { if (string.toString().length === 0) { return { error: ERRORS.invalidString, valid: false }; } return { value: string.toString(), error: null, valid: true }; }; const validateHex = (string) => { if (!isHex(string.toString())) { return { error: ERRORS.invalidHex, valid: false }; } return { value: string.toString(), error: null, valid: true }; }; const validateURL = (string) => { if (!validator.isURL(string.toString())) { return { error: ERRORS.invalidURL, valid: false }; } return { value: string.toString(), error: null, valid: true }; }; export const validate = (value, type, contract) => { switch (type) { case ADDRESS_TYPE: return validateAddress(value); case TOKEN_ADDRESS_TYPE: return validateTokenAddress(value, contract); case SIMPLE_TOKEN_ADDRESS_TYPE: return validateTokenAddress(value, contract, true); case TLA_TYPE: return validateTLA(value, contract); case SIMPLE_TLA_TYPE: return validateTLA(value, contract, true); case UINT_TYPE: return validateUint(value); case DECIMAL_TYPE: return validateDecimal(value); case STRING_TYPE: return validateString(value); case HEX_TYPE: return validateHex(value); case URL_TYPE: return validateURL(value); default: return { valid: true, error: null }; } };