aboutsummaryrefslogblamecommitdiffstats
path: root/test/contracts/AuctionRegistrar.cpp
blob: 7c5d9fa3518adf27243a6c09aac038787fc20f0e (plain) (tree)


































































































































































































































































































                                                                                                                               


                                                             







































































































                                                                                                                    
                                                           











                                                           

















                                                                           

 



























































                                                                                                





                           
/*
    This file is part of cpp-ethereum.

    cpp-ethereum is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    cpp-ethereum is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with cpp-ethereum.  If not, see <http://www.gnu.org/licenses/>.
*/
/**
 * @author Christian <c@ethdev.com>
 * @date 2015
 * Tests for a fixed fee registrar contract.
 */

#include <string>
#include <tuple>
#include <boost/test/unit_test.hpp>
#include <libdevcore/Hash.h>
#include <libethcore/ABI.h>
#include <test/libsolidity/solidityExecutionFramework.h>

using namespace std;

namespace dev
{
namespace solidity
{
namespace test
{

namespace
{

static char const* registrarCode = R"DELIMITER(
//sol

contract NameRegister {
    function addr(string _name) constant returns (address o_owner);
    function name(address _owner) constant returns (string o_name);
}

contract Registrar is NameRegister {
    event Changed(string indexed name);
    event PrimaryChanged(string indexed name, address indexed addr);

    function owner(string _name) constant returns (address o_owner);
    function addr(string _name) constant returns (address o_address);
    function subRegistrar(string _name) constant returns (address o_subRegistrar);
    function content(string _name) constant returns (bytes32 o_content);

    function name(address _owner) constant returns (string o_name);
}

contract AuctionSystem {
    event AuctionEnded(string indexed _name, address _winner);
    event NewBid(string indexed _name, address _bidder, uint _value);

    /// Function that is called once an auction ends.
    function onAuctionEnd(string _name) internal;

    function bid(string _name, address _bidder, uint _value) internal {
        var auction = m_auctions[_name];
        if (auction.endDate > 0 && now > auction.endDate)
        {
            AuctionEnded(_name, auction.highestBidder);
            onAuctionEnd(_name);
            delete m_auctions[_name];
            return;
        }
        if (msg.value > auction.highestBid)
        {
            // new bid on auction
            auction.secondHighestBid = auction.highestBid;
            auction.sumOfBids += _value;
            auction.highestBid = _value;
            auction.highestBidder = _bidder;
            auction.endDate = now + c_biddingTime;

            NewBid(_name, _bidder, _value);
        }
    }

    uint constant c_biddingTime = 7 days;

    struct Auction {
        address highestBidder;
        uint highestBid;
        uint secondHighestBid;
        uint sumOfBids;
        uint endDate;
    }
    mapping(string => Auction) m_auctions;
}

contract GlobalRegistrar is Registrar, AuctionSystem {
    struct Record {
        address owner;
        address primary;
        address subRegistrar;
        bytes32 content;
        uint renewalDate;
    }

    uint constant c_renewalInterval = 1 years;
    uint constant c_freeBytes = 12;

    function Registrar() {
        // TODO: Populate with hall-of-fame.
    }

    function() {
        // prevent people from just sending funds to the registrar
        __throw();
    }

    function onAuctionEnd(string _name) internal {
        var auction = m_auctions[_name];
        var record = m_toRecord[_name];
        if (record.owner != 0)
            record.owner.send(auction.sumOfBids - auction.highestBid / 100);
        else
            auction.highestBidder.send(auction.highestBid - auction.secondHighestBid);
        record.renewalDate = now + c_renewalInterval;
        record.owner = auction.highestBidder;
        Changed(_name);
    }

    function reserve(string _name) external {
        if (bytes(_name).length == 0)
            __throw();
        bool needAuction = requiresAuction(_name);
        if (needAuction)
        {
            if (now < m_toRecord[_name].renewalDate)
                __throw();
            bid(_name, msg.sender, msg.value);
        }
        else
        {
            Record record = m_toRecord[_name];
            if (record.owner != 0)
                __throw();
            m_toRecord[_name].owner = msg.sender;
            Changed(_name);
        }
    }

    function requiresAuction(string _name) internal returns (bool) {
        return bytes(_name).length < c_freeBytes;
    }

    modifier onlyrecordowner(string _name) { if (m_toRecord[_name].owner == msg.sender) _ }

    function transfer(string _name, address _newOwner) onlyrecordowner(_name) {
        m_toRecord[_name].owner = _newOwner;
        Changed(_name);
    }

    function disown(string _name) onlyrecordowner(_name) {
        if (stringsEqual(m_toName[m_toRecord[_name].primary], _name))
        {
            PrimaryChanged(_name, m_toRecord[_name].primary);
            m_toName[m_toRecord[_name].primary] = "";
        }
        delete m_toRecord[_name];
        Changed(_name);
    }

    function setAddress(string _name, address _a, bool _primary) onlyrecordowner(_name) {
        m_toRecord[_name].primary = _a;
        if (_primary)
        {
            PrimaryChanged(_name, _a);
            m_toName[_a] = _name;
        }
        Changed(_name);
    }
    function setSubRegistrar(string _name, address _registrar) onlyrecordowner(_name) {
        m_toRecord[_name].subRegistrar = _registrar;
        Changed(_name);
    }
    function setContent(string _name, bytes32 _content) onlyrecordowner(_name) {
        m_toRecord[_name].content = _content;
        Changed(_name);
    }

    function stringsEqual(string storage _a, string memory _b) internal returns (bool) {
        bytes storage a = bytes(_a);
        bytes memory b = bytes(_b);
        if (a.length != b.length)
            return false;
        // @todo unroll this loop
        for (uint i = 0; i < a.length; i ++)
            if (a[i] != b[i])
                return false;
        return true;
    }

    function owner(string _name) constant returns (address) { return m_toRecord[_name].owner; }
    function addr(string _name) constant returns (address) { return m_toRecord[_name].primary; }
    function subRegistrar(string _name) constant returns (address) { return m_toRecord[_name].subRegistrar; }
    function content(string _name) constant returns (bytes32) { return m_toRecord[_name].content; }
    function name(address _addr) constant returns (string o_name) { return m_toName[_addr]; }

    function __throw() internal {
        // workaround until we have "throw"
        uint[] x; x[1];
    }

    mapping (address => string) m_toName;
    mapping (string => Record) m_toRecord;
}
)DELIMITER";

static unique_ptr<bytes> s_compiledRegistrar;

class AuctionRegistrarTestFramework: public ExecutionFramework
{
protected:
    void deployRegistrar()
    {
        if (!s_compiledRegistrar)
        {
            m_optimize = true;
            m_compiler.reset(false, m_addStandardSources);
            m_compiler.addSource("", registrarCode);
            ETH_TEST_REQUIRE_NO_THROW(m_compiler.compile(m_optimize, m_optimizeRuns), "Compiling contract failed");
            s_compiledRegistrar.reset(new bytes(m_compiler.getBytecode("GlobalRegistrar")));
        }
        sendMessage(*s_compiledRegistrar, true);
        BOOST_REQUIRE(!m_output.empty());
    }

    using ContractInterface = ExecutionFramework::ContractInterface;
    class RegistrarInterface: public ContractInterface
    {
    public:
        RegistrarInterface(ExecutionFramework& _framework): ContractInterface(_framework) {}
        void reserve(string const& _name)
        {
            callString("reserve", _name);
        }
        u160 owner(string const& _name)
        {
            return callStringReturnsAddress("owner", _name);
        }
        void setAddress(string const& _name, u160 const& _address, bool _primary)
        {
            callStringAddressBool("setAddress", _name, _address, _primary);
        }
        u160 addr(string const& _name)
        {
            return callStringReturnsAddress("addr", _name);
        }
        string name(u160 const& _addr)
        {
            return callAddressReturnsString("name", _addr);
        }
        void setSubRegistrar(string const& _name, u160 const& _address)
        {
            callStringAddress("setSubRegistrar", _name, _address);
        }
        u160 subRegistrar(string const& _name)
        {
            return callStringReturnsAddress("subRegistrar", _name);
        }
        void setContent(string const& _name, h256 const& _content)
        {
            callStringBytes32("setContent", _name, _content);
        }
        h256 content(string const& _name)
        {
            return callStringReturnsBytes32("content", _name);
        }
        void transfer(string const& _name, u160 const& _target)
        {
            return callStringAddress("transfer", _name, _target);
        }
        void disown(string const& _name)
        {
            return callString("disown", _name);
        }
    };

    u256 const m_biddingTime = u256(7 * 24 * 3600);
    u256 const m_renewalInterval = u256(365 * 24 * 3600);
};

}

/// This is a test suite that tests optimised code!
BOOST_FIXTURE_TEST_SUITE(SolidityAuctionRegistrar, AuctionRegistrarTestFramework)

BOOST_AUTO_TEST_CASE(creation)
{
    deployRegistrar();
}

BOOST_AUTO_TEST_CASE(reserve)
{
    // Test that reserving works for long strings
    deployRegistrar();
    vector<string> names{"abcabcabcabcabc", "defdefdefdefdef", "ghighighighighighighighighighighighighighighi"};
    m_sender = Address(0x123);

    RegistrarInterface registrar(*this);

    // should not work
    registrar.reserve("");
    BOOST_CHECK_EQUAL(registrar.owner(""), u160(0));

    for (auto const& name: names)
    {
        registrar.reserve(name);
        BOOST_CHECK_EQUAL(registrar.owner(name), u160(0x123));
    }
}

BOOST_AUTO_TEST_CASE(double_reserve_long)
{
    // Test that it is not possible to re-reserve from a different address.
    deployRegistrar();
    string name = "abcabcabcabcabcabcabcabcabcabca";
    m_sender = Address(0x123);
    RegistrarInterface registrar(*this);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), u160(0x123));

    m_sender = Address(0x124);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), u160(0x123));
}

BOOST_AUTO_TEST_CASE(properties)
{
    // Test setting and retrieving  the various properties works.
    deployRegistrar();
    RegistrarInterface registrar(*this);
    string names[] = {"abcaeouoeuaoeuaoeu", "defncboagufra,fui", "ghagpyajfbcuajouhaeoi"};
    size_t addr = 0x9872543;
    for (string const& name: names)
    {
        addr++;
        size_t sender = addr + 10007;
        m_sender = Address(sender);
        // setting by sender works
        registrar.reserve(name);
        BOOST_CHECK_EQUAL(registrar.owner(name), u160(sender));
        registrar.setAddress(name, addr, true);
        BOOST_CHECK_EQUAL(registrar.addr(name), u160(addr));
        registrar.setSubRegistrar(name, addr + 20);
        BOOST_CHECK_EQUAL(registrar.subRegistrar(name), u160(addr + 20));
        registrar.setContent(name, h256(u256(addr + 90)));
        BOOST_CHECK_EQUAL(registrar.content(name), h256(u256(addr + 90)));

        // but not by someone else
        m_sender = Address(h256(addr + 10007 - 1));
        BOOST_CHECK_EQUAL(registrar.owner(name), u160(sender));
        registrar.setAddress(name, addr + 1, true);
        BOOST_CHECK_EQUAL(registrar.addr(name), u160(addr));
        registrar.setSubRegistrar(name, addr + 20 + 1);
        BOOST_CHECK_EQUAL(registrar.subRegistrar(name), u160(addr + 20));
        registrar.setContent(name, h256(u256(addr + 90 + 1)));
        BOOST_CHECK_EQUAL(registrar.content(name), h256(u256(addr + 90)));
    }
}

BOOST_AUTO_TEST_CASE(transfer)
{
    deployRegistrar();
    string name = "abcaoeguaoucaeoduceo";
    m_sender = Address(0x123);
    RegistrarInterface registrar(*this);
    registrar.reserve(name);
    registrar.setContent(name, h256(u256(123)));
    registrar.transfer(name, u160(555));
    BOOST_CHECK_EQUAL(registrar.owner(name), u160(555));
    BOOST_CHECK_EQUAL(registrar.content(name), h256(u256(123)));
}

BOOST_AUTO_TEST_CASE(disown)
{
    deployRegistrar();
    string name = "abcaoeguaoucaeoduceo";
    m_sender = Address(0x123);
    RegistrarInterface registrar(*this);
    registrar.reserve(name);
    registrar.setContent(name, h256(u256(123)));
    registrar.setAddress(name, u160(124), true);
    registrar.setSubRegistrar(name, u160(125));
    BOOST_CHECK_EQUAL(registrar.name(u160(124)), name);

    // someone else tries disowning
    m_sender = Address(0x128);
    registrar.disown(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0x123);

    m_sender = Address(0x123);
    registrar.disown(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0);
    BOOST_CHECK_EQUAL(registrar.addr(name), 0);
    BOOST_CHECK_EQUAL(registrar.subRegistrar(name), 0);
    BOOST_CHECK_EQUAL(registrar.content(name), h256());
    BOOST_CHECK_EQUAL(registrar.name(u160(124)), "");
}

BOOST_AUTO_TEST_CASE(auction_simple)
{
    deployRegistrar();
    string name = "x";
    m_sender = Address(0x123);
    RegistrarInterface registrar(*this);
    // initiate auction
    registrar.setNextValue(8);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0);
    // "wait" until auction end
    m_envInfo.setTimestamp(m_envInfo.timestamp() + m_biddingTime + 10);
    // trigger auction again
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0x123);
}

BOOST_AUTO_TEST_CASE(auction_bidding)
{
    deployRegistrar();
    string name = "x";
    m_sender = Address(0x123);
    RegistrarInterface registrar(*this);
    // initiate auction
    registrar.setNextValue(8);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0);
    // overbid self
    m_envInfo.setTimestamp(m_biddingTime - 10);
    registrar.setNextValue(12);
    registrar.reserve(name);
    // another bid by someone else
    m_sender = Address(0x124);
    m_envInfo.setTimestamp(2 * m_biddingTime - 50);
    registrar.setNextValue(13);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0);
    // end auction by first bidder (which is not highest) trying to overbid again (too late)
    m_sender = Address(0x123);
    m_envInfo.setTimestamp(4 * m_biddingTime);
    registrar.setNextValue(20);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0x124);
}

BOOST_AUTO_TEST_CASE(auction_renewal)
{
    deployRegistrar();
    string name = "x";
    RegistrarInterface registrar(*this);
    // register name by auction
    m_sender = Address(0x123);
    registrar.setNextValue(8);
    registrar.reserve(name);
    m_envInfo.setTimestamp(4 * m_biddingTime);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0x123);

    // try to re-register before interval end
    m_sender = Address(0x222);
    registrar.setNextValue(80);
    m_envInfo.setTimestamp(m_envInfo.timestamp() + m_renewalInterval - 1);
    registrar.reserve(name);
    m_envInfo.setTimestamp(m_envInfo.timestamp() + m_biddingTime);
    // if there is a bug in the renewal logic, this would transfer the ownership to 0x222,
    // but if there is no bug, this will initiate the auction, albeit with a zero bid
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0x123);

    m_envInfo.setTimestamp(m_envInfo.timestamp() + 2);
    registrar.setNextValue(80);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0x123);
    m_envInfo.setTimestamp(m_envInfo.timestamp() + m_biddingTime + 2);
    registrar.reserve(name);
    BOOST_CHECK_EQUAL(registrar.owner(name), 0x222);
}

BOOST_AUTO_TEST_SUITE_END()

}
}
} // end namespaces