SwiftHTML & CSSSolidityDesenvolvimento de JogosSolana/Rust
16.11.2024

Aula 048: Padrões de Tratamento de Exceções em Solidity

Em Solidity, gerenciar exceções é uma parte crítica do desenvolvimento de contratos inteligentes. Padrões adequados de tratamento de exceções podem proteger seu contrato contra comportamentos inesperados e garantir operações robustas. Nesta aula, exploraremos várias técnicas de tratamento de exceções que podem ser empregadas em seus contratos inteligentes em Solidity.

1. A Declaração require

A declaração require é amplamente utilizada para impor pré-condições. Se a condição for avaliada como falsa, a transação é revertida e quaisquer mudanças de estado feitas durante a transação são desfeitas.

Exemplo

pragma solidity ^0.8.0;

contract ArmazenamentoSimples {
    uint256 private dados;

    function armazenarDados(uint256 _dados) public {
        require(_dados > 0, "Os dados devem ser positivos!");
        dados = _dados;
    }

    function recuperarDados() public view returns (uint256) {
        return dados;
    }
}

No exemplo acima, garantimos que os dados armazenados sejam positivos. Se a condição não for atendida, a transação é revertida com a mensagem de erro especificada.

2. A Declaração assert

A declaração assert é utilizada para checar condições que nunca deveriam acontecer. Ela é empregada para validar falhas internas e invariantes. Se assert falhar, isso indica um bug no contrato e a transação é revertida.

Exemplo

pragma solidity ^0.8.0;

contract Contador {
    uint256 private contagem;

    function incrementar() public {
        contagem += 1;
        assert(contagem > 0); // a contagem nunca deve ser negativa
    }

    function obterContagem() public view returns (uint256) {
        return contagem;
    }
}

No exemplo acima, usamos assert para garantir que a contagem nunca se torne negativa após uma operação de incremento. Se a condição for avaliada como falsa, isso indica um erro crítico.

3. A Declaração revert

A declaração revert pode ser utilizada em qualquer lugar de uma função para reverter a transação. Você também pode incluir uma mensagem de erro para maior clareza.

Exemplo

pragma solidity ^0.8.0;

contract Retirada {
    address private proprietario;
    mapping(address => uint256) private saldos;

    constructor() {
        proprietario = msg.sender;
    }

    function depositar() public payable {
        saldos[msg.sender] += msg.value;
    }

    function retirar(uint256 quantidade) public {
        if (saldos[msg.sender] < quantidade) {
            revert("Saldo insuficiente!");
        }
        saldos[msg.sender] -= quantidade;
        payable(msg.sender).transfer(quantidade);
    }

    function obterSaldo() public view returns (uint256) {
        return saldos[msg.sender];
    }
}

Neste exemplo, antes de permitir uma retirada, verificamos o saldo do usuário. Se o saldo for insuficiente, chamamos revert com uma mensagem de erro apropriada.

4. Tipos de Erros Personalizados

Solidity permite a definição de tipos de erros personalizados que são mais eficientes em termos de gás do que mensagens de require ou revert. Erros personalizados permitem um tratamento de erro e depuração melhores.

Exemplo

pragma solidity ^0.8.0;

contract Token {
    uint256 private totalSupply;
    mapping(address => uint256) private saldos;

    error FundosInsuficientes(uint256 solicitado, uint256 disponível);

    function mintear(uint256 quantidade) public {
        totalSupply += quantidade;
        saldos[msg.sender] += quantidade;
    }

    function transferir(address para, uint256 quantidade) public {
        if (saldos[msg.sender] < quantidade) {
            revert FundosInsuficientes(quantidade, saldos[msg.sender]);
        }
        saldos[msg.sender] -= quantidade;
        saldos[para] += quantidade;
    }
}

Neste caso, definimos um erro personalizado FundosInsuficientes, que fornece mais contexto de forma transparente quando uma transferência falha.

Conclusão

O tratamento de exceções é um aspecto fundamental do desenvolvimento em Solidity. Usar construções como require, assert, revert e erros personalizados de forma eficaz pode ajudá-lo a construir contratos inteligentes mais resilientes. Sempre lembre-se de tratar exceções de maneira adequada para evitar possíveis explorações, garantir segurança e melhorar a experiência do usuário.

Video

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

Thank you for voting!