Bloquear una cuenta en un entorno de cadena de bloques autorizado

Estamos desarrollando un entorno de cadena de bloques autorizado con Ethermint. Las empresas deben presentar una solicitud para unirse al entorno blockchain, creamos y les damos su cuenta Ethereum.

Una de las principales preocupaciones es cómo eliminar o bloquear una cuenta del entorno si resulta que esta empresa no está jugando bien.

Pensamos en marcar el contrato de la organización o destruirlo, pero luego tendría que poner un modificador en casi todos los métodos Y aún pueden realizar transacciones con su cuenta Ethereum.

No estoy seguro de que sea posible bloquear dicha cuenta desde un entorno de cadena de bloques autorizado, porque, bueno, de eso se trata obviamente la descentralización :-) Pero tal vez haya una solución alternativa o un "patrón" para usar aquí.

Respuestas (1)

Muy fácil de hacer en realidad.

Esta función es una sola entrada para modificar saldos en Ethereum ( core/state_transition.go):

// TransitionDb will transition the state by applying the current message and
// returning the result including the the used gas. It returns an error if it
// failed. An error indicates a consensus issue.
func (st *StateTransition) TransitionDb() (ret []byte, usedGas uint64, failed bool, err error) {
    if err = st.preCheck(); err != nil {
        return
    }
    msg := st.msg
    sender := vm.AccountRef(msg.From())
    homestead := st.evm.ChainConfig().IsHomestead(st.evm.BlockNumber)
    contractCreation := msg.To() == nil

    // Pay intrinsic gas
    gas, err := IntrinsicGas(st.data, contractCreation, homestead)
    if err != nil {
        return nil, 0, false, err
    }
    if err = st.useGas(gas); err != nil {
        return nil, 0, false, err
    }

    var (
        evm = st.evm
        // vm errors do not effect consensus and are therefor
        // not assigned to err, except for insufficient balance
        // error.
        vmerr error
    )
    if contractCreation {
        ret, _, st.gas, vmerr = evm.Create(sender, st.data, st.gas, st.value)
    } else {
        // Increment the nonce for the next transaction
        st.state.SetNonce(msg.From(), st.state.GetNonce(sender.Address())+1)
        ret, st.gas, vmerr = evm.Call(sender, st.to(), st.data, st.gas, st.value)
    }
    if vmerr != nil {
        log.Debug("VM returned with error", "err", vmerr)
        // The only possible consensus-error would be if there wasn't
        // sufficient balance to make the transfer happen. The first
        // balance transfer may never fail.
        if vmerr == vm.ErrInsufficientBalance {
            return nil, 0, false, vmerr
        }
    }
    st.refundGas()
    st.state.AddBalance(st.evm.Coinbase, new(big.Int).Mul(new(big.Int).SetUint64(st.gasUsed()), st.gasPrice))

    return ret, st.gasUsed(), vmerr != nil, err
}

Tienes que hacer algo como esto:

func (st *StateTransition) TransitionDb() (ret []byte, usedGas uint64, failed bool, err error) {
    if err = st.preCheck(); err != nil {
        return
    }
    msg := st.msg
    sender := vm.AccountRef(msg.From())
    homestead := st.evm.ChainConfig().IsHomestead(st.evm.BlockNumber)
    contractCreation := msg.To() == nil

    // here is where your custom code goes:

    errAccessDenied:=errors.New("Company isn't playing nice")
    blacklisted_address,exists:=blacklists[sender]
    if exists {
        return ret,0,false, errAccessDenied
    }
    blacklisted_address,exists=blaclists[st.to()]
    if exists {
        return ret,0,false,errAccessDenied
    }
    // here is where your custom code ends   

    // Pay intrinsic gas
    gas, err := IntrinsicGas(st.data, contractCreation, homestead)
    if err != nil {
        return nil, 0, false, err
    }
    if err = st.useGas(gas); err != nil {
        return nil, 0, false, err
    }

    var (
        evm = st.evm
        // vm errors do not effect consensus and are therefor
        // not assigned to err, except for insufficient balance
        // error.
        vmerr error
    )

    if contractCreation {
        ret, _, st.gas, vmerr = evm.Create(sender, st.data, st.gas, st.value)
    } else {
        // Increment the nonce for the next transaction
        st.state.SetNonce(msg.From(), st.state.GetNonce(sender.Address())+1)
        ret, st.gas, vmerr = evm.Call(sender, st.to(), st.data, st.gas, st.value)
    }
    if vmerr != nil {
        log.Debug("VM returned with error", "err", vmerr)
        // The only possible consensus-error would be if there wasn't
        // sufficient balance to make the transfer happen. The first
        // balance transfer may never fail.
        if vmerr == vm.ErrInsufficientBalance {
            return nil, 0, false, vmerr
        }
    }
    st.refundGas()
    st.state.AddBalance(st.evm.Coinbase, new(big.Int).Mul(new(big.Int).SetUint64(st.gasUsed()), st.gasPrice))

    return ret, st.gasUsed(), vmerr != nil, err
}

Por supuesto, no es un código probado, pero te haces una idea. También debe incluir la funcionalidad para distribuir la lista de empresas que no funcionan bien entre los nodos. El hash de la lista negra debe incluirse en el encabezado del bloque (types.Header)