SwiftHTML & CSSSolidityDesenvolvimento de JogosSolana/Rust
16.11.2024

Aula 046: Tratamento de Erros em Solidity

O tratamento de erros é um aspecto crucial no desenvolvimento de contratos inteligentes utilizando Solidity. Nesta aula, iremos explorar vários métodos para o tratamento de erros em Solidity, incluindo require, assert e revert. Também discutiremos como personalizar mensagens de erro para uma melhor depuração e experiência do usuário.

Entendendo os Tipos de Erros

Em Solidity, existem principalmente três maneiras de lidar com erros:

  1. require: Usado para validar entradas e condições antes de executar uma função. Se a condição falhar, a execução é interrompida e quaisquer alterações realizadas durante a transação são revertidas.

  2. assert: Utilizado para verificar condições que nunca deveriam falhar. É usado principalmente para capturar erros internos e invariantes. Se uma assert falhar, isso indica um bug no contrato.

  3. revert: Utilizado para parar a execução e retornar uma mensagem de erro. É mais flexível que o require e pode ser usado para condições complexas, permitindo que os desenvolvedores forneçam uma mensagem de erro personalizada.

Usando require

A função require é adequada para validar as entradas e estados antes de executar a lógica em uma função. Veja como funciona:

pragma solidity ^0.8.0;

contract ExemploTratamentoErro {
    uint public saldo;

    // Função para depositar fundos
    function depositar(uint valor) public {
        require(valor > 0, "O valor do depósito deve ser maior que zero");

        saldo += valor;
    }
}

Neste exemplo, require verifica se o valor do depósito é maior que zero. Se esta condição falhar, a transação é revertida e o usuário recebe uma mensagem de erro.

Usando assert

A função assert deve ser utilizada para verificar condições que nunca deveriam falhar. Geralmente é usada para verificações de consistência interna. Aqui está um exemplo:

pragma solidity ^0.8.0;

contract Contador {
    uint public contador;

    // Função para incrementar o contador
    function incrementar() public {
        contador += 1;
        assert(contador > 0); // Isso nunca deveria falhar
    }
}

Neste caso, afirmamos que o contador incrementado deve sempre ser maior que zero. Se contador se tornar negativo devido a um erro no código, a assert irá falhar, indicando um problema sério na lógica do contrato.

Usando revert

A função revert é versátil e permite especificar uma mensagem de erro personalizada sem precisar atender a uma condição simples. Veja como utilizá-la:

pragma solidity ^0.8.0;

contract GerenciadorDeFundos {
    mapping(address => uint) public fundos;

    // Função para retirar fundos
    function retirar(uint valor) public {
        if (fundos[msg.sender] < valor) {
            revert("Fundos insuficientes para retirada");
        }

        fundos[msg.sender] -= valor;
    }
}

Neste exemplo, se o usuário tentar retirar mais fundos do que possui, a função revert é chamada com uma mensagem de erro personalizada, fornecendo mais contexto sobre a falha.

Erros Personalizados (Solidity 0.8.4 e acima)

A versão 0.8.4 do Solidity introduziu um recurso para definir erros personalizados. Esta é uma maneira mais eficiente em termos de gás para lidar com erros, pois economiza no custo de gás que seria gasto em uma mensagem de string:

pragma solidity ^0.8.4;

contract ErrosPersonalizados {
    error FundosInsuficientes(uint disponivel, uint requerido);

    mapping(address => uint) public saldos;

    function retirar(uint valor) public {
        uint saldoDisponivel = saldos[msg.sender];
        if (saldoDisponivel < valor) {
            revert FundosInsuficientes(saldoDisponivel, valor);
        }

        saldos[msg.sender] -= valor;
    }
}

Neste exemplo, um erro personalizado FundosInsuficientes é definido, aceitando dois parâmetros: o saldo disponível e o valor requerido. Se a retirada falhar, ele fornece parâmetros claros para entender o contexto do erro sem incorrer em custos adicionais de gás por uma mensagem de string.

Conclusão

Um tratamento adequado de erros é vital para escrever contratos inteligentes robustos e seguros em Solidity. Ao usar efetivamente require, assert, revert e erros personalizados, os desenvolvedores podem garantir que seus contratos se comportem como esperado e forneçam feedback útil aos usuários. Lembre-se de sempre pensar sobre casos extremos e potenciais pontos de falha em seus contratos ao planejar suas estratégias de tratamento de erros. Boa codificação!

Video

Did you like this article? Rate it from 1 to 5:

Thank you for voting!