mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			147 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			ReStructuredText
		
	
	
	
	
	
			
		
		
	
	
			147 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			ReStructuredText
		
	
	
	
	
	
| .. index:: purchase, remote purchase, escrow
 | |
| 
 | |
| ********************
 | |
| Safe Remote Purchase
 | |
| ********************
 | |
| 
 | |
| Purchasing goods remotely currently requires multiple parties that need to trust each other.
 | |
| The simplest configuration involves a seller and a buyer. The buyer would like to receive
 | |
| an item from the seller and the seller would like to get money (or an equivalent)
 | |
| in return. The problematic part is the shipment here: There is no way to determine for
 | |
| sure that the item arrived at the buyer.
 | |
| 
 | |
| There are multiple ways to solve this problem, but all fall short in one or the other way.
 | |
| In the following example, both parties have to put twice the value of the item into the
 | |
| contract as escrow. As soon as this happened, the money will stay locked inside
 | |
| the contract until the buyer confirms that they received the item. After that,
 | |
| the buyer is returned the value (half of their deposit) and the seller gets three
 | |
| times the value (their deposit plus the value). The idea behind
 | |
| this is that both parties have an incentive to resolve the situation or otherwise
 | |
| their money is locked forever.
 | |
| 
 | |
| This contract of course does not solve the problem, but gives an overview of how
 | |
| you can use state machine-like constructs inside a contract.
 | |
| 
 | |
| 
 | |
| ::
 | |
| 
 | |
|     // SPDX-License-Identifier: GPL-3.0
 | |
|     pragma solidity >=0.7.0;
 | |
| 
 | |
|     contract Purchase {
 | |
|         uint public value;
 | |
|         address payable public seller;
 | |
|         address payable public buyer;
 | |
| 
 | |
|         enum State { Created, Locked, Release, Inactive }
 | |
|         // The state variable has a default value of the first member, `State.created`
 | |
|         State public state;
 | |
| 
 | |
|         modifier condition(bool _condition) {
 | |
|             require(_condition);
 | |
|             _;
 | |
|         }
 | |
| 
 | |
|         modifier onlyBuyer() {
 | |
|             require(
 | |
|                 msg.sender == buyer,
 | |
|                 "Only buyer can call this."
 | |
|             );
 | |
|             _;
 | |
|         }
 | |
| 
 | |
|         modifier onlySeller() {
 | |
|             require(
 | |
|                 msg.sender == seller,
 | |
|                 "Only seller can call this."
 | |
|             );
 | |
|             _;
 | |
|         }
 | |
| 
 | |
|         modifier inState(State _state) {
 | |
|             require(
 | |
|                 state == _state,
 | |
|                 "Invalid state."
 | |
|             );
 | |
|             _;
 | |
|         }
 | |
| 
 | |
|         event Aborted();
 | |
|         event PurchaseConfirmed();
 | |
|         event ItemReceived();
 | |
|         event SellerRefunded();
 | |
| 
 | |
|         // Ensure that `msg.value` is an even number.
 | |
|         // Division will truncate if it is an odd number.
 | |
|         // Check via multiplication that it wasn't an odd number.
 | |
|         constructor() payable {
 | |
|             seller = msg.sender;
 | |
|             value = msg.value / 2;
 | |
|             require((2 * value) == msg.value, "Value has to be even.");
 | |
|         }
 | |
| 
 | |
|         /// Abort the purchase and reclaim the ether.
 | |
|         /// Can only be called by the seller before
 | |
|         /// the contract is locked.
 | |
|         function abort()
 | |
|             public
 | |
|             onlySeller
 | |
|             inState(State.Created)
 | |
|         {
 | |
|             emit Aborted();
 | |
|             state = State.Inactive;
 | |
|             // We use transfer here directly. It is
 | |
|             // reentrancy-safe, because it is the
 | |
|             // last call in this function and we
 | |
|             // already changed the state.
 | |
|             seller.transfer(address(this).balance);
 | |
|         }
 | |
| 
 | |
|         /// Confirm the purchase as buyer.
 | |
|         /// Transaction has to include `2 * value` ether.
 | |
|         /// The ether will be locked until confirmReceived
 | |
|         /// is called.
 | |
|         function confirmPurchase()
 | |
|             public
 | |
|             inState(State.Created)
 | |
|             condition(msg.value == (2 * value))
 | |
|             payable
 | |
|         {
 | |
|             emit PurchaseConfirmed();
 | |
|             buyer = msg.sender;
 | |
|             state = State.Locked;
 | |
|         }
 | |
| 
 | |
|         /// Confirm that you (the buyer) received the item.
 | |
|         /// This will release the locked ether.
 | |
|         function confirmReceived()
 | |
|             public
 | |
|             onlyBuyer
 | |
|             inState(State.Locked)
 | |
|         {
 | |
|             emit ItemReceived();
 | |
|             // It is important to change the state first because
 | |
|             // otherwise, the contracts called using `send` below
 | |
|             // can call in again here.
 | |
|             state = State.Release;
 | |
| 
 | |
|             buyer.transfer(value);
 | |
|         }
 | |
| 
 | |
|         /// This function refunds the seller, i.e.
 | |
|         /// pays back the locked funds of the seller.
 | |
|         function refundSeller()
 | |
|             public
 | |
|             onlySeller
 | |
|             inState(State.Release)
 | |
|         {
 | |
|             emit SellerRefunded();
 | |
|             // It is important to change the state first because
 | |
|             // otherwise, the contracts called using `send` below
 | |
|             // can call in again here.
 | |
|             state = State.Inactive;
 | |
| 
 | |
|             seller.transfer(3 * value);
 | |
|         }
 | |
|     }
 |