import {SchemaValidator} from '../schemas';
import {addressUtils, BigNumber, B, logUtils} from '../utils';
import * as _ from 'lodash';
import * as validUrl from 'valid-url';
import BN from "bn.js"

const HEX_REGEX = /^0x[0-9A-F]*$/i;
const schemaValidator = new SchemaValidator();
export const assert = {
    isBigNumber(variableName: string, value: BigNumber): void {
        const isBigNumber = B.BigNumber.isBigNumber(value);
        assert.assert(isBigNumber, assert.typeAssertionMessage(variableName, 'BigNumber', value));
    },
    isNumberLike(variableName: string, value: BigNumber | number): void {
        const isBigNumber = B.BigNumber.isBigNumber(value);
        const isNumber = typeof value === 'number';
        assert.assert(isBigNumber || isNumber, assert.typeAssertionMessage(variableName, 'BigNumber | number', value));
    },
    isValidBaseUnitAmount(variableName: string, value: BigNumber): void {
        assert.isBigNumber(variableName, value);
        const isNegative = value.isLessThan(0);
        assert.assert(!isNegative, `${variableName} cannot be a negative number, found value: ${value.toNumber()}`);
        const hasDecimals = value.decimalPlaces() !== 0;
        assert.assert(
            !hasDecimals,
            `${variableName} should be in baseUnits (no decimals), found value: ${value.toNumber()}`,
        );
    },
    isString(variableName: string, value: string): void {
        assert.assert(_.isString(value), assert.typeAssertionMessage(variableName, 'string', value));
    },
    isFunction(variableName: string, value: any): void {
        assert.assert(_.isFunction(value), assert.typeAssertionMessage(variableName, 'function', value));
    },
    isHexString(variableName: string, value: string): void {
        assert.assert(
            _.isString(value) && HEX_REGEX.test(value),
            assert.typeAssertionMessage(variableName, 'HexString', value),
        );
    },
    isETHAddressHex(variableName: string, value: string): void {
        assert.assert(_.isString(value), assert.typeAssertionMessage(variableName, 'string', value));
        assert.assert(addressUtils.isAddress(value), assert.typeAssertionMessage(variableName, 'ETHAddressHex', value));
    },
    doesBelongToStringEnum(
        variableName: string,
        value: string,
        stringEnum: any /* There is no base type for every string enum */,
    ): void {
        const enumValues = _.values(stringEnum);
        const doesBelongToStringEnum = _.includes(enumValues, value);
        const enumValuesAsStrings = _.map(enumValues, enumValue => `'${enumValue}'`);
        const enumValuesAsString = enumValuesAsStrings.join(', ');
        assert.assert(
            doesBelongToStringEnum,
            `Expected ${variableName} to be one of: ${enumValuesAsString}, encountered: ${value}`,
        );
    },
    hasAtMostOneUniqueValue(value: any[], errMsg: string): void {
        assert.assert(_.uniq(value).length <= 1, errMsg);
    },
    isNumber(variableName: string, value: number): void {
        assert.assert(_.isFinite(value), assert.typeAssertionMessage(variableName, 'number', value));
    },
    isNumberOrBigNumber(variableName: string, value: any): void {
        if (_.isFinite(value)) {
            return;
        } else {
            let finalbignum = false
            if (typeof value === "object") {
                if (B.BigNumber.isBigNumber(value)) {
                    finalbignum = true
                }
                if (Object(value).hasOwnProperty("_isBigNumber")) {
                    if (value._isBigNumber === true) {
                        finalbignum = true
                    }
                }
                if (Object(value).hasOwnProperty("words")) {
                    finalbignum = true
                }
                if (Object(value).hasOwnProperty("s") && Object(value).hasOwnProperty("e") && Object(value).hasOwnProperty("c")) {
                    finalbignum = true
                }
                if (value instanceof B.BigNumber) {
                    finalbignum = true
                }
                if (value instanceof BN) {
                    finalbignum = true
                }
            }
            assert.assert(
                finalbignum,
                assert.typeAssertionMessage(variableName, 'number or BigNumber or BN', value),
            );
        }
    },
    isBoolean(variableName: string, value: boolean): void {
        assert.assert(_.isBoolean(value), assert.typeAssertionMessage(variableName, 'boolean', value));
    },
    isWeb3Provider(variableName: string, value: any): void {
        logUtils.warn('DEPRECATED: Please use providerUtils.standardizeOrThrow() instead');
        const isWeb3Provider = _.isFunction(value.send) || _.isFunction(value.sendAsync);
        assert.assert(isWeb3Provider, assert.typeAssertionMessage(variableName, 'Provider', value));
    },
    doesConformToSchema(variableName: string, value: any, schema: object, subSchemas?: object[]): void {
        if (value === undefined) {
            throw new Error(`${variableName} can't be undefined`);
        }
        if (subSchemas !== undefined) {
            schemaValidator.addSchema(subSchemas);
        }
        const validationResult = schemaValidator.validate(value, schema);
        const hasValidationErrors = validationResult.errors && validationResult.errors.length > 0;
        const msg = hasValidationErrors
            ? `Expected ${variableName} to conform to schema ${(schema as any).id}
Encountered: ${JSON.stringify(value, null, '\t')}
Validation errors: ${validationResult.errors!.join(', ')}`
            : '';
        assert.assert(!hasValidationErrors, msg);
    },
    doesMatchRegex(variableName: string, value: string, regex: RegExp): void {
        assert.assert(regex.test(value), assert.typeAssertionMessage(variableName, String(regex), value));
    },
    isWebUri(variableName: string, value: any): void {
        const isValidUrl = validUrl.isWebUri(value) !== undefined;
        assert.assert(isValidUrl, assert.typeAssertionMessage(variableName, 'web uri', value));
    },
    isUri(variableName: string, value: any): void {
        const isValidUri = validUrl.isUri(value) !== undefined;
        assert.assert(isValidUri, assert.typeAssertionMessage(variableName, 'uri', value));
    },
    isBlockParam(variableName: string, value: any): void {
        if (Number.isInteger(value) && value >= 0) {
            return;
        }
        if (value === 'earliest' || value === 'latest' || value === 'pending') {
            return;
        }
        throw new Error(assert.typeAssertionMessage(variableName, 'BlockParam', value));
    },
    isArray(variableName: string, value: any): void {
        if (!Array.isArray(value)) {
            throw new Error(assert.typeAssertionMessage(variableName, 'Array', value));
        }
    },
    assert(condition: boolean, message: string): void {
        if (!condition) {
            throw new Error(message);
        }
    },
    typeAssertionMessage(variableName: string, type: string, value: any): string {
        return `Expected ${variableName} to be of type ${type}, encountered: ${value}`;
    },
};
