Skip to content

Latest commit

Β 

History

History
277 lines (204 loc) Β· 13.4 KB

security-considerations.rst

File metadata and controls

277 lines (204 loc) Β· 13.4 KB

λ³΄μ•ˆ μΈ‘λ©΄ 고렀사항

μ†Œν”„νŠΈμ›¨μ–΄λ₯Ό μ›ν•˜λŠ”λŒ€λ‘œ κ°œλ°œν•˜λŠ” 것은 어렡지 μ•Šμ§€λ§Œ, λ‹€λ₯Έ μ‚¬λžŒμ΄ μ•„μ˜ˆ λ‹€λ₯Έ μ˜λ„λ‘œ μž‘λ™μ‹œν‚€λŠ”κ±Έ λ§‰λŠ” 것은 μ–΄λ ΅μŠ΅λ‹ˆλ‹€.

μ†”λ¦¬λ””ν‹°μ—μ„œλŠ” λͺ¨λ“  슀마트 μ»¨νŠΈλž™νŠΈκ°€ 곡개적으둜 μ‹€ν–‰λ˜κ³  λŒ€λΆ€λΆ„μ˜ μ†ŒμŠ€μ½”λ“œλ₯Ό 확인할 수 μžˆλŠ” κ²½μš°κ°€ λ§ŽμŠ΅λ‹ˆλ‹€. λ”°λΌμ„œ 이런 λ³΄μ•ˆ 츑면의 고렀사항은 특히 더 μ€‘μš”ν•©λ‹ˆλ‹€.

λ¬Όλ‘  λ³΄μ•ˆμ— μ–Όλ§ˆλ‚˜ 신경을 μ¨μ•Όν•˜λŠ” μ§€λŠ” 상황에 따라 λ‹€λ¦…λ‹ˆλ‹€. κ°€λ Ή, μ›Ή μ„œλΉ„μŠ€λŠ” 곡격자λ₯Ό ν¬ν•¨ν•œ λŒ€μ€‘μ—κ²Œ κ³΅κ°œλ˜μ–΄μ•Όν•˜κ³ , λˆ„κ΅¬λ‚˜ μ ‘κ·Όν•  수 μžˆμ–΄μ•Όν•˜λ©°, μ–΄λ–€ λ•Œμ—λŠ” μ˜€ν”ˆμ†ŒμŠ€λ‘œ κ΄€λ¦¬λ˜λŠ” κ²½μš°λ„ μžˆμŠ΅λ‹ˆλ‹€. λ§Œμ•½ μ›Ή μ„œλΉ„μŠ€μ— μ€‘μš”μΉ˜ μ•Šμ€ μ •λ³΄λ§Œ μ €μž₯ν•œλ‹€λ©΄ λ¬Έμ œκ°€ λ˜μ§€ μ•Šμ§€λ§Œ, 은행 κ³„μ’Œμ™€ 같은 정보λ₯Ό κ΄€λ¦¬ν•œλ‹€λ©΄ λ”μš± 쑰심할 ν•„μš”κ°€ 있죠.

이 μž₯μ—μ„œλŠ” 쑰심해야할 λ¬Έμ œλ“€κ³Ό 일반적인 λ³΄μ•ˆκ΄€λ ¨ νŒ¨ν„΄λ“€μ„ λ‹€λ£Ήλ‹ˆλ‹€. ν•˜μ§€λ§Œ μ΄λŠ” μ™„λ²½ν•œ 해결법이 μ•„λ‹™λ‹ˆλ‹€. 즉, 슀마트 μ»¨νŠΈλž™νŠΈ μƒμ—λŠ” 버그가 없더라도, μ»΄νŒŒμΌλŸ¬λ‚˜ ν”Œλž«νΌ μžμ²΄μ— 버그가 μžˆμ„ 수 μžˆλ‹€λŠ” μ–˜κΈ°μ£ .

μ–Έμ œλ‚˜ 그렇듯이, 이 λ¬Έμ„œλŠ” μ˜€ν”ˆ μ†ŒμŠ€ 기반의 λ¬Έμ„œμ΄κΈ° λ•Œλ¬Έμ—, λ³΄μ•ˆμ— λŒ€ν•œ λ¬Έμ œκ°€ 생긴닀면 주저없이 λ‚΄μš©μ„ μΆ”κ°€ν•΄μ£Όμ‹œκΈ° λ°”λžλ‹ˆλ‹€.

μœ μ˜μ‚¬ν•­

개인 정보와 λ¬΄μž‘μœ„μ„±

슀마트 μ»¨νŠΈλž™νŠΈ μƒμ˜ λͺ¨λ“  μ •λ³΄λŠ” 곡개적으둜 λ³΄μ—¬μ§‘λ‹ˆλ‹€. 심지어 지역 λ³€μˆ˜ 및 μƒνƒœ λ³€μˆ˜κ°€ ``private``으둜 μ„ μ–Έλ˜μ—ˆλ‹€κ³ ν•΄λ„ λ§ˆμ°¬κ°€μ§€μ£ .

λ§Œμ•½ 당신이 μ±„κ΅΄μžμ˜ λΆ€μ • ν–‰μœ„λ₯Ό λ§‰κ³ μž ν•œλ‹€λ©΄, λ‚œμˆ˜λ₯Ό μƒμ„±ν•˜λŠ” 것이 μ–΄λŠμ •λ„ μœ μš©ν•  수 μžˆμŠ΅λ‹ˆλ‹€.

μž¬μ§„μž… 문제

(A)μ½˜νŠΈλž™νŠΈμ—μ„œ (B)μ½˜νŠΈλž™νŠΈλ‘œ μ—°κ²°λ˜λŠ” μ–΄λ– ν•œ μƒν˜Έμž‘μš© 및 Ether의 전솑은 μ œμ–΄κΆŒμ„ (B)μ—κ²Œ λ„˜κ²¨μ£Όκ²Œ λ©λ‹ˆλ‹€. 이 λ•Œλ¬Έμ— B의 μƒν˜Έμž‘μš©μ΄ λλ‚˜κΈ° 전에 λ‹€μ‹œ Aλ₯Ό ν˜ΈμΆœν•  수 μžˆλŠ” 상황이 λ²Œμ–΄μ§ˆ 수 μžˆμŠ΅λ‹ˆλ‹€. 예λ₯Ό λ“€μ–΄, λ‹€μŒ μ½”λ“œλŠ” 버그λ₯Ό ν¬ν•¨ν•˜κ³  μžˆμŠ΅λ‹ˆλ‹€(μš”μ•½λœ μ½”λ“œμž…λ‹ˆλ‹€).

pragma solidity ^0.4.0;

// 버그가 ν¬ν•¨λœ μ½”λ“œμž…λ‹ˆλ‹€. μ‚¬μš©ν•˜μ§€ λ§ˆμ„Έμš”!
contract Fund {
    /// μ»¨νŠΈλž™νŠΈμ˜ Ether 정보 mapping
    mapping(address => uint) shares;
    /// 지뢄을 μΈμΆœν•˜λŠ” ν•¨μˆ˜
    function withdraw() public {
        if (msg.sender.send(shares[msg.sender]))
            shares[msg.sender] = 0;
    }
}

``send``ν•¨μˆ˜ μžμ²΄μ—μ„œ gas의 μ†ŒλΉ„λŸ‰μ„ μ œμ–΄ν•˜κΈ° λ•Œλ¬Έμ—, 큰 λ¬Έμ œλŠ” λ˜μ§€ μ•Šμ§€λ§Œ, κ·ΈλŸΌμ—λ„ 이 μ½”λ“œλŠ” λ³΄μ•ˆ μƒμ˜ 문제λ₯Ό 가지고 μžˆμŠ΅λ‹ˆλ‹€. Ether의 전솑은 항상 μ½”λ“œμ˜ 싀행을 ν¬ν•¨ν•˜κΈ°μ—, μˆ˜μ‹ μžλŠ” 반볡적으둜 ``withdraw``λ₯Ό μ‹€ν–‰ν•  수 있게되죠. 결과적으둜 μ€‘λ³΅λœ ``withdraw``ν•¨μˆ˜μ˜ 싀행을 톡해 μ»¨νŠΈλž™νŠΈ μƒμ˜ λͺ¨λ“  Etherλ₯Ό κ°€μ Έκ°ˆ 수 μžˆλ‹€λŠ” μ˜λ―Έμž…λ‹ˆλ‹€. 상황에 따라, κ³΅κ²©μžλŠ” μ•„λž˜ μ½”λ“œ 속 ``call``을 톡해 남은 gasλ₯Ό λͺ¨λ‘ κ°€μ Έμ˜¬ 수 μžˆμ„μ§€λ„ λͺ¨λ¦…λ‹ˆλ‹€.

pragma solidity ^0.4.0;

// 버그가 ν¬ν•¨λœ μ½”λ“œμž…λ‹ˆλ‹€. μ‚¬μš©ν•˜μ§€ λ§ˆμ„Έμš”!
contract Fund {
    /// μ»¨νŠΈλž™νŠΈμ˜ Ether 정보 mapping
    mapping(address => uint) shares;
    /// 지뢄을 μΈμΆœν•˜λŠ” ν•¨μˆ˜
    function withdraw() public {
        if (msg.sender.call.value(shares[msg.sender])())
            shares[msg.sender] = 0;
    }
}

μž¬μ§„μž… 곡격을 막기 μœ„ν•΄μ„œλŠ” μ•„λž˜μ™€ 같이 Checks-Effects-Interactions νŒ¨ν„΄μ„ μ‚¬μš©ν•  수 μžˆμŠ΅λ‹ˆλ‹€.

pragma solidity ^0.4.11;

contract Fund {
    /// μ»¨νŠΈλž™νŠΈμ˜ Ether 정보 mapping
    mapping(address => uint) shares;
    /// 지뢄을 μΈμΆœν•˜λŠ” ν•¨μˆ˜
    function withdraw() public {
        var share = shares[msg.sender];
        shares[msg.sender] = 0;
        msg.sender.transfer(share);
    }
}

μž¬μ§„μž… 곡격은 Ether μ „μ†‘μ—μ„œ 뿐만 μ•„λ‹ˆλΌ ν•¨μˆ˜λ₯Ό ν˜ΈμΆœν•˜λŠ” μ–΄λ–€ μƒν™©μ—μ„œλ„ μˆ˜ν–‰λ  수 μžˆμŠ΅λ‹ˆλ‹€. λ‚˜μ•„κ°€, μ—¬λŸ¬λΆ„μ€ ν•˜λ‚˜μ˜ 계정에 λ§Žμ€ μ»¨νŠΈλž™νŠΈλ₯Ό κ°€μ§ˆ μˆ˜λ„ μžˆμ„ ν…λ°μš”, 이 λ•Œ, ν•˜λ‚˜μ˜ μ»¨νŠΈλž™νŠΈκ°€ λ‹€λ₯Έ μ»¨νŠΈλž™νŠΈλ₯Ό ν˜ΈμΆœν•  수 μžˆλ‹€λŠ” 것도 μ•Œμ•„λ‘¬μ•Όν•©λ‹ˆλ‹€.

κ°€μŠ€ μ œν•œ 및 루프

Loops that do not have a fixed number of iterations, for example, loops that depend on storage values, have to be used carefully: Due to the block gas limit, transactions can only consume a certain amount of gas. Either explicitly or just due to normal operation, the number of iterations in a loop can grow beyond the block gas limit which can cause the complete contract to be stalled at a certain point. This may not apply to constant functions that are only executed to read data from the blockchain. Still, such functions may be called by other contracts as part of on-chain operations and stall those. Please be explicit about such cases in the documentation of your contracts.

Ether 보내고 λ°›κΈ°

  • Neither contracts nor "external accounts" are currently able to prevent that someone sends them Ether. Contracts can react on and reject a regular transfer, but there are ways to move Ether without creating a message call. One way is to simply "mine to" the contract address and the second way is using selfdestruct(x).
  • If a contract receives Ether (without a function being called), the fallback function is executed. If it does not have a fallback function, the Ether will be rejected (by throwing an exception). During the execution of the fallback function, the contract can only rely on the "gas stipend" (2300 gas) being available to it at that time. This stipend is not enough to access storage in any way. To be sure that your contract can receive Ether in that way, check the gas requirements of the fallback function (for example in the "details" section in Remix).
  • There is a way to forward more gas to the receiving contract using addr.call.value(x)(). This is essentially the same as addr.transfer(x), only that it forwards all remaining gas and opens up the ability for the recipient to perform more expensive actions (and it only returns a failure code and does not automatically propagate the error). This might include calling back into the sending contract or other state changes you might not have thought of. So it allows for great flexibility for honest users but also for malicious actors.
  • If you want to send Ether using address.transfer, there are certain details to be aware of:
    1. If the recipient is a contract, it causes its fallback function to be executed which can, in turn, call back the sending contract.
    2. Sending Ether can fail due to the call depth going above 1024. Since the caller is in total control of the call depth, they can force the transfer to fail; take this possibility into account or use send and make sure to always check its return value. Better yet, write your contract using a pattern where the recipient can withdraw Ether instead.
    3. Sending Ether can also fail because the execution of the recipient contract requires more than the allotted amount of gas (explicitly by using require, assert, revert, throw or because the operation is just too expensive) - it "runs out of gas" (OOG). If you use transfer or send with a return value check, this might provide a means for the recipient to block progress in the sending contract. Again, the best practice here is to use a :ref:`"withdraw" pattern instead of a "send" pattern <withdrawal_pattern>`.

μ½œμŠ€νƒ 깊이

External function calls can fail any time because they exceed the maximum call stack of 1024. In such situations, Solidity throws an exception. Malicious actors might be able to force the call stack to a high value before they interact with your contract.

Note that .send() does not throw an exception if the call stack is depleted but rather returns false in that case. The low-level functions .call(), .callcode() and .delegatecall() behave in the same way.

tx.origin

Never use tx.origin for authorization. Let's say you have a wallet contract like this:

pragma solidity ^0.4.11;

// THIS CONTRACT CONTAINS A BUG - DO NOT USE
contract TxUserWallet {
    address owner;

    function TxUserWallet() public {
        owner = msg.sender;
    }

    function transferTo(address dest, uint amount) public {
        require(tx.origin == owner);
        dest.transfer(amount);
    }
}

Now someone tricks you into sending ether to the address of this attack wallet:

pragma solidity ^0.4.11;

interface TxUserWallet {
    function transferTo(address dest, uint amount) public;
}

contract TxAttackWallet {
    address owner;

    function TxAttackWallet() public {
        owner = msg.sender;
    }

    function() public {
        TxUserWallet(msg.sender).transferTo(owner, msg.sender.balance);
    }
}

If your wallet had checked msg.sender for authorization, it would get the address of the attack wallet, instead of the owner address. But by checking tx.origin, it gets the original address that kicked off the transaction, which is still the owner address. The attack wallet instantly drains all your funds.

Minor Details

  • In for (var i = 0; i < arrayName.length; i++) { ... }, the type of i will be uint8, because this is the smallest type that is required to hold the value 0. If the array has more than 255 elements, the loop will not terminate.
  • The constant keyword for functions is currently not enforced by the compiler. Furthermore, it is not enforced by the EVM, so a contract function that "claims" to be constant might still cause changes to the state.
  • Types that do not occupy the full 32 bytes might contain "dirty higher order bits". This is especially important if you access msg.data - it poses a malleability risk: You can craft transactions that call a function f(uint8 x) with a raw byte argument of 0xff000001 and with 0x00000001. Both are fed to the contract and both will look like the number 1 as far as x is concerned, but msg.data will be different, so if you use keccak256(msg.data) for anything, you will get different results.

Recommendations

Restrict the Amount of Ether

Restrict the amount of Ether (or other tokens) that can be stored in a smart contract. If your source code, the compiler or the platform has a bug, these funds may be lost. If you want to limit your loss, limit the amount of Ether.

Keep it Small and Modular

Keep your contracts small and easily understandable. Single out unrelated functionality in other contracts or into libraries. General recommendations about source code quality of course apply: Limit the amount of local variables, the length of functions and so on. Document your functions so that others can see what your intention was and whether it is different than what the code does.

Use the Checks-Effects-Interactions Pattern

Most functions will first perform some checks (who called the function, are the arguments in range, did they send enough Ether, does the person have tokens, etc.). These checks should be done first.

As the second step, if all checks passed, effects to the state variables of the current contract should be made. Interaction with other contracts should be the very last step in any function.

Early contracts delayed some effects and waited for external function calls to return in a non-error state. This is often a serious mistake because of the re-entrancy problem explained above.

Note that, also, calls to known contracts might in turn cause calls to unknown contracts, so it is probably better to just always apply this pattern.

Include a Fail-Safe Mode

While making your system fully decentralised will remove any intermediary, it might be a good idea, especially for new code, to include some kind of fail-safe mechanism:

You can add a function in your smart contract that performs some self-checks like "Has any Ether leaked?", "Is the sum of the tokens equal to the balance of the contract?" or similar things. Keep in mind that you cannot use too much gas for that, so help through off-chain computations might be needed there.

If the self-check fails, the contract automatically switches into some kind of "failsafe" mode, which, for example, disables most of the features, hands over control to a fixed and trusted third party or just converts the contract into a simple "give me back my money" contract.

Formal Verification

Using formal verification, it is possible to perform an automated mathematical proof that your source code fulfills a certain formal specification. The specification is still formal (just as the source code), but usually much simpler.

Note that formal verification itself can only help you understand the difference between what you did (the specification) and how you did it (the actual implementation). You still need to check whether the specification is what you wanted and that you did not miss any unintended effects of it.