ETH Price: $1,817.64 (+0.33%)

Transaction Decoder

Block:
11994950 at Mar-08-2021 01:46:21 AM +UTC
Transaction Fee:
0.004936342 ETH $8.97
Gas Used:
37,682 Gas / 131 Gwei

Emitted Events:

155 AdminUpgradeabilityProxy.0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef( 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef, 0x00000000000000000000000016fd342933ec13543f4da7ea23dbb1424f688163, 0x0000000000000000000000009f982aac7a6bc246b6ea3f4e83bdaa75caa576cc, 00000000000000000000000000000000000000000003f8fa4a03a7ddcb2e8b0c )

Account State Difference:

  Address   Before After State Difference Code
0x16fd3429...24F688163
1.228664386626239119 Eth
Nonce: 49
1.223728044626239119 Eth
Nonce: 50
0.004936342
0x7eAf7C84...4E62F2bD8
(Ethermine)
784.529439602013124835 Eth784.534375944013124835 Eth0.004936342

Execution Trace

AdminUpgradeabilityProxy.a9059cbb( )
  • CardstackToken.transfer( recipient=0x9f982Aac7a6bc246B6Ea3F4e83BDAa75CAa576CC, amount=4802541365904941700254476 ) => ( True )
    • AdminUpgradeabilityProxy.beabacc8( )
      • 0xcdcfc2791b51190a8e5a8514a0157a2212b083d1.beabacc8( )
        File 1 of 3: AdminUpgradeabilityProxy
        pragma solidity ^0.4.13;
        
        library AddressUtils {
        
          /**
           * Returns whether the target address is a contract
           * @dev This function will return false if invoked during the constructor of a contract,
           * as the code is not actually created until after the constructor finishes.
           * @param _addr address to check
           * @return whether the target address is a contract
           */
          function isContract(address _addr) internal view returns (bool) {
            uint256 size;
            // XXX Currently there is no better way to check if there is a contract in an address
            // than to check the size of the code at that address.
            // See https://ethereum.stackexchange.com/a/14016/36603
            // for more details about how this works.
            // TODO Check this again before the Serenity release, because all addresses will be
            // contracts then.
            // solium-disable-next-line security/no-inline-assembly
            assembly { size := extcodesize(_addr) }
            return size > 0;
          }
        
        }
        
        contract Proxy {
          /**
           * @dev Fallback function.
           * Implemented entirely in `_fallback`.
           */
          function () payable external {
            _fallback();
          }
        
          /**
           * @return The Address of the implementation.
           */
          function _implementation() internal view returns (address);
        
          /**
           * @dev Delegates execution to an implementation contract.
           * This is a low level function that doesn't return to its internal call site.
           * It will return to the external caller whatever the implementation returns.
           * @param implementation Address to delegate.
           */
          function _delegate(address implementation) internal {
            assembly {
              // Copy msg.data. We take full control of memory in this inline assembly
              // block because it will not return to Solidity code. We overwrite the
              // Solidity scratch pad at memory position 0.
              calldatacopy(0, 0, calldatasize)
        
              // Call the implementation.
              // out and outsize are 0 because we don't know the size yet.
              let result := delegatecall(gas, implementation, 0, calldatasize, 0, 0)
        
              // Copy the returned data.
              returndatacopy(0, 0, returndatasize)
        
              switch result
              // delegatecall returns 0 on error.
              case 0 { revert(0, returndatasize) }
              default { return(0, returndatasize) }
            }
          }
        
          /**
           * @dev Function that is run as the first thing in the fallback function.
           * Can be redefined in derived contracts to add functionality.
           * Redefinitions must call super._willFallback().
           */
          function _willFallback() internal {
          }
        
          /**
           * @dev fallback implementation.
           * Extracted to enable manual triggering.
           */
          function _fallback() internal {
            _willFallback();
            _delegate(_implementation());
          }
        }
        
        contract UpgradeabilityProxy is Proxy {
          /**
           * @dev Emitted when the implementation is upgraded.
           * @param implementation Address of the new implementation.
           */
          event Upgraded(address implementation);
        
          /**
           * @dev Storage slot with the address of the current implementation.
           * This is the keccak-256 hash of "org.zeppelinos.proxy.implementation", and is
           * validated in the constructor.
           */
          bytes32 private constant IMPLEMENTATION_SLOT = 0x7050c9e0f4ca769c69bd3a8ef740bc37934f8e2c036e5a723fd8ee048ed3f8c3;
        
          /**
           * @dev Contract constructor.
           * @param _implementation Address of the initial implementation.
           */
          constructor(address _implementation) public {
            assert(IMPLEMENTATION_SLOT == keccak256("org.zeppelinos.proxy.implementation"));
        
            _setImplementation(_implementation);
          }
        
          /**
           * @dev Returns the current implementation.
           * @return Address of the current implementation
           */
          function _implementation() internal view returns (address impl) {
            bytes32 slot = IMPLEMENTATION_SLOT;
            assembly {
              impl := sload(slot)
            }
          }
        
          /**
           * @dev Upgrades the proxy to a new implementation.
           * @param newImplementation Address of the new implementation.
           */
          function _upgradeTo(address newImplementation) internal {
            _setImplementation(newImplementation);
            emit Upgraded(newImplementation);
          }
        
          /**
           * @dev Sets the implementation address of the proxy.
           * @param newImplementation Address of the new implementation.
           */
          function _setImplementation(address newImplementation) private {
            require(AddressUtils.isContract(newImplementation), "Cannot set a proxy implementation to a non-contract address");
        
            bytes32 slot = IMPLEMENTATION_SLOT;
        
            assembly {
              sstore(slot, newImplementation)
            }
          }
        }
        
        contract AdminUpgradeabilityProxy is UpgradeabilityProxy {
          /**
           * @dev Emitted when the administration has been transferred.
           * @param previousAdmin Address of the previous admin.
           * @param newAdmin Address of the new admin.
           */
          event AdminChanged(address previousAdmin, address newAdmin);
        
          /**
           * @dev Storage slot with the admin of the contract.
           * This is the keccak-256 hash of "org.zeppelinos.proxy.admin", and is
           * validated in the constructor.
           */
          bytes32 private constant ADMIN_SLOT = 0x10d6a54a4754c8869d6886b5f5d7fbfa5b4522237ea5c60d11bc4e7a1ff9390b;
        
          /**
           * @dev Modifier to check whether the `msg.sender` is the admin.
           * If it is, it will run the function. Otherwise, it will delegate the call
           * to the implementation.
           */
          modifier ifAdmin() {
            if (msg.sender == _admin()) {
              _;
            } else {
              _fallback();
            }
          }
        
          /**
           * Contract constructor.
           * It sets the `msg.sender` as the proxy administrator.
           * @param _implementation address of the initial implementation.
           */
          constructor(address _implementation) UpgradeabilityProxy(_implementation) public {
            assert(ADMIN_SLOT == keccak256("org.zeppelinos.proxy.admin"));
        
            _setAdmin(msg.sender);
          }
        
          /**
           * @return The address of the proxy admin.
           */
          function admin() external view ifAdmin returns (address) {
            return _admin();
          }
        
          /**
           * @return The address of the implementation.
           */
          function implementation() external view ifAdmin returns (address) {
            return _implementation();
          }
        
          /**
           * @dev Changes the admin of the proxy.
           * Only the current admin can call this function.
           * @param newAdmin Address to transfer proxy administration to.
           */
          function changeAdmin(address newAdmin) external ifAdmin {
            require(newAdmin != address(0), "Cannot change the admin of a proxy to the zero address");
            emit AdminChanged(_admin(), newAdmin);
            _setAdmin(newAdmin);
          }
        
          /**
           * @dev Upgrade the backing implementation of the proxy.
           * Only the admin can call this function.
           * @param newImplementation Address of the new implementation.
           */
          function upgradeTo(address newImplementation) external ifAdmin {
            _upgradeTo(newImplementation);
          }
        
          /**
           * @dev Upgrade the backing implementation of the proxy and call a function
           * on the new implementation.
           * This is useful to initialize the proxied contract.
           * @param newImplementation Address of the new implementation.
           * @param data Data to send as msg.data in the low level call.
           * It should include the signature and the parameters of the function to be
           * called, as described in
           * https://solidity.readthedocs.io/en/develop/abi-spec.html#function-selector-and-argument-encoding.
           */
          function upgradeToAndCall(address newImplementation, bytes data) payable external ifAdmin {
            _upgradeTo(newImplementation);
            require(address(this).call.value(msg.value)(data));
          }
        
          /**
           * @return The admin slot.
           */
          function _admin() internal view returns (address adm) {
            bytes32 slot = ADMIN_SLOT;
            assembly {
              adm := sload(slot)
            }
          }
        
          /**
           * @dev Sets the address of the proxy admin.
           * @param newAdmin Address of the new proxy admin.
           */
          function _setAdmin(address newAdmin) internal {
            bytes32 slot = ADMIN_SLOT;
        
            assembly {
              sstore(slot, newAdmin)
            }
          }
        
          /**
           * @dev Only fall back when the sender is not the admin.
           */
          function _willFallback() internal {
            require(msg.sender != _admin(), "Cannot call fallback function from the proxy admin");
            super._willFallback();
          }
        }

        File 2 of 3: CardstackToken
        pragma solidity ^0.4.13;
        
        library SafeMath {
        
          /**
          * @dev Multiplies two numbers, throws on overflow.
          */
          function mul(uint256 _a, uint256 _b) internal pure returns (uint256 c) {
            // Gas optimization: this is cheaper than asserting 'a' not being zero, but the
            // benefit is lost if 'b' is also tested.
            // See: https://github.com/OpenZeppelin/openzeppelin-solidity/pull/522
            if (_a == 0) {
              return 0;
            }
        
            c = _a * _b;
            assert(c / _a == _b);
            return c;
          }
        
          /**
          * @dev Integer division of two numbers, truncating the quotient.
          */
          function div(uint256 _a, uint256 _b) internal pure returns (uint256) {
            // assert(_b > 0); // Solidity automatically throws when dividing by 0
            // uint256 c = _a / _b;
            // assert(_a == _b * c + _a % _b); // There is no case in which this doesn't hold
            return _a / _b;
          }
        
          /**
          * @dev Subtracts two numbers, throws on overflow (i.e. if subtrahend is greater than minuend).
          */
          function sub(uint256 _a, uint256 _b) internal pure returns (uint256) {
            assert(_b <= _a);
            return _a - _b;
          }
        
          /**
          * @dev Adds two numbers, throws on overflow.
          */
          function add(uint256 _a, uint256 _b) internal pure returns (uint256 c) {
            c = _a + _b;
            assert(c >= _a);
            return c;
          }
        }
        
        contract Initializable {
        
          /**
           * @dev Indicates if the contract has been initialized.
           */
          bool public initialized;
        
          /**
           * @dev Modifier to use in the initialization function of a contract.
           */
          modifier isInitializer() {
            require(!initialized, "Contract instance has already been initialized");
            _;
            initialized = true;
          }
        }
        
        contract Administratable {
          using SafeMath for uint256;
        
          // zOS requires that the variables are never removed nor order changed
          // Since this is a parent contract, no new variables can be added here
          address internal constant primaryInitializer = 0x0AEaF8c2Fe778797CD5464E7EB8351d28da2E823;
          address internal constant stagingInitializer = 0x1E65F71b024937b988fdba09814d60049e0Fc59d;
          address public owner;
          bool public adminsInitialized;
          address[] public adminsForIndex;
          address[] public superAdminsForIndex;
          mapping (address => bool) public admins;
          mapping (address => bool) public superAdmins;
          mapping (address => bool) private processedAdmin;
          mapping (address => bool) private processedSuperAdmin;
        
          event AddAdmin(address indexed admin);
          event RemoveAdmin(address indexed admin);
          event AddSuperAdmin(address indexed admin);
          event RemoveSuperAdmin(address indexed admin);
          event OwnershipTransferred(
            address indexed previousOwner,
            address indexed newOwner
          );
        
          modifier onlyInitializers() {
            require(msg.sender == primaryInitializer ||
                    msg.sender == stagingInitializer);
            _;
          }
        
          modifier onlyOwner() {
            require(msg.sender == owner);
            _;
          }
        
          modifier onlyAdmins {
            if (msg.sender != owner && !superAdmins[msg.sender] && !admins[msg.sender]) revert();
            _;
          }
        
          modifier onlySuperAdmins {
            if (msg.sender != owner && !superAdmins[msg.sender]) revert();
            _;
          }
        
          constructor() public {
            owner = msg.sender;
          }
        
          function addSuperAdmin(address admin) public onlySuperAdmins {
            _addSuperAdmin(admin);
        
            emit AddSuperAdmin(admin);
          }
        
          function removeSuperAdmin(address admin) public onlySuperAdmins {
            require(admin != address(0));
            superAdmins[admin] = false;
        
            emit RemoveSuperAdmin(admin);
          }
        
          function addAdmin(address admin) public onlySuperAdmins {
            require(admin != address(0));
            admins[admin] = true;
            if (!processedAdmin[admin]) {
              adminsForIndex.push(admin);
              processedAdmin[admin] = true;
            }
        
            emit AddAdmin(admin);
          }
        
          function removeAdmin(address admin) public onlySuperAdmins {
            require(admin != address(0));
            admins[admin] = false;
        
            emit RemoveAdmin(admin);
          }
        
          function transferOwnership(address _newOwner) public onlyOwner {
            require(_newOwner != address(0));
            emit OwnershipTransferred(owner, _newOwner);
            owner = _newOwner;
          }
        
          function totalSuperAdminsMapping() public view returns (uint256) {
            return superAdminsForIndex.length;
          }
        
          function totalAdminsMapping() public view returns (uint256) {
            return adminsForIndex.length;
          }
        
          // Accepting a function param to set initial owner/admins
          // is a potential security vulnerability when using initialize
          // pattern (which is just a public function). So we hard-code
          // our initial admin addresses and use zOS to manage this list.
          function initializeAdmins() internal {
            require(!adminsInitialized, "Contract instance has already initialized the admins");
        
            owner = primaryInitializer;
            _addSuperAdmin(stagingInitializer);
        
            adminsInitialized = true;
          }
        
          function _addSuperAdmin(address admin) internal {
            require(admin != address(0));
            superAdmins[admin] = true;
            if (!processedSuperAdmin[admin]) {
              superAdminsForIndex.push(admin);
              processedSuperAdmin[admin] = true;
            }
          }
        
        
        }
        
        contract Configurable {
          function configureFromStorage() public returns (bool);
        }
        
        library CstLibrary {
          using SafeMath for uint256;
        
          function getTokenName(address _storage) public view returns(bytes32) {
            return ExternalStorage(_storage).getBytes32Value("cstTokenName");
          }
        
          function setTokenName(address _storage, bytes32 tokenName) public {
            ExternalStorage(_storage).setBytes32Value("cstTokenName", tokenName);
          }
        
          function getTokenSymbol(address _storage) public view returns(bytes32) {
            return ExternalStorage(_storage).getBytes32Value("cstTokenSymbol");
          }
        
          function setTokenSymbol(address _storage, bytes32 tokenName) public {
            ExternalStorage(_storage).setBytes32Value("cstTokenSymbol", tokenName);
          }
        
          function getBuyPrice(address _storage) public view returns(uint256) {
            return ExternalStorage(_storage).getUIntValue("cstBuyPrice");
          }
        
          function setBuyPrice(address _storage, uint256 value) public {
            ExternalStorage(_storage).setUIntValue("cstBuyPrice", value);
          }
        
          function getCirculationCap(address _storage) public view returns(uint256) {
            return ExternalStorage(_storage).getUIntValue("cstCirculationCap");
          }
        
          function setCirculationCap(address _storage, uint256 value) public {
            ExternalStorage(_storage).setUIntValue("cstCirculationCap", value);
          }
        
          function getBalanceLimit(address _storage) public view returns(uint256) {
            return ExternalStorage(_storage).getUIntValue("cstBalanceLimit");
          }
        
          function setBalanceLimit(address _storage, uint256 value) public {
            ExternalStorage(_storage).setUIntValue("cstBalanceLimit", value);
          }
        
          function getFoundation(address _storage) public view returns(address) {
            return ExternalStorage(_storage).getAddressValue("cstFoundation");
          }
        
          function setFoundation(address _storage, address value) public {
            ExternalStorage(_storage).setAddressValue("cstFoundation", value);
          }
        
          function getAllowance(address _storage, address account, address spender) public view returns (uint256) {
            return ExternalStorage(_storage).getMultiLedgerValue("cstAllowance", account, spender);
          }
        
          function setAllowance(address _storage, address account, address spender, uint256 allowance) public {
            ExternalStorage(_storage).setMultiLedgerValue("cstAllowance", account, spender, allowance);
          }
        
          function getCustomBuyerLimit(address _storage, address buyer) public view returns (uint256) {
            return ExternalStorage(_storage).getLedgerValue("cstCustomBuyerLimit", buyer);
          }
        
          function setCustomBuyerLimit(address _storage, address buyer, uint256 value) public {
            ExternalStorage(_storage).setLedgerValue("cstCustomBuyerLimit", buyer, value);
          }
        
          function getCustomBuyerForIndex(address _storage, uint256 index) public view returns (address) {
            return ExternalStorage(_storage).ledgerEntryForIndex(keccak256("cstCustomBuyerLimit"), index);
          }
        
          function getCustomBuyerMappingCount(address _storage) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerCount("cstCustomBuyerLimit");
          }
        
          function getApprovedBuyer(address _storage, address buyer) public view returns (bool) {
            return ExternalStorage(_storage).getBooleanMapValue("cstApprovedBuyer", buyer);
          }
        
          function setApprovedBuyer(address _storage, address buyer, bool value) public {
            ExternalStorage(_storage).setBooleanMapValue("cstApprovedBuyer", buyer, value);
          }
        
          function getApprovedBuyerForIndex(address _storage, uint256 index) public view returns (address) {
            return ExternalStorage(_storage).booleanMapEntryForIndex(keccak256("cstApprovedBuyer"), index);
          }
        
          function getApprovedBuyerMappingCount(address _storage) public view returns(uint256) {
            return ExternalStorage(_storage).getBooleanMapCount("cstApprovedBuyer");
          }
        
          function getTotalUnvestedAndUnreleasedTokens(address _storage) public view returns(uint256) {
            return ExternalStorage(_storage).getUIntValue("cstUnvestedAndUnreleasedTokens");
          }
        
          function setTotalUnvestedAndUnreleasedTokens(address _storage, uint256 value) public {
            ExternalStorage(_storage).setUIntValue("cstUnvestedAndUnreleasedTokens", value);
          }
        
          function vestingMappingSize(address _storage) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerCount("cstFullyVestedAmount");
          }
        
          function vestingBeneficiaryForIndex(address _storage, uint256 index) public view returns(address) {
            return ExternalStorage(_storage).ledgerEntryForIndex(keccak256("cstFullyVestedAmount"), index);
          }
        
          function releasableAmount(address _storage, address beneficiary) public view returns (uint256) {
            uint256 releasedAmount = getVestingReleasedAmount(_storage, beneficiary);
            return vestedAvailableAmount(_storage, beneficiary).sub(releasedAmount);
          }
        
          function vestedAvailableAmount(address _storage, address beneficiary) public view returns (uint256) {
            uint256 start = getVestingStart(_storage, beneficiary);
            uint256 fullyVestedAmount = getFullyVestedAmount(_storage, beneficiary);
        
            if (start == 0 || fullyVestedAmount == 0) {
              return 0;
            }
        
            uint256 duration = getVestingDuration(_storage, beneficiary);
            if (duration == 0) {
              return 0;
            }
            uint256 cliff = getVestingCliff(_storage, beneficiary);
            uint256 revokeDate = getVestingRevokeDate(_storage, beneficiary);
        
            if (now < cliff || (revokeDate > 0 && revokeDate < cliff)) {
              return 0;
            } else if (revokeDate > 0 && revokeDate > cliff) {
              return fullyVestedAmount.mul(revokeDate.sub(start)).div(duration);
            } else if (now >= start.add(duration)) {
              return fullyVestedAmount;
            } else {
              return fullyVestedAmount.mul(now.sub(start)).div(duration);
            }
          }
        
          function vestedAmount(address _storage, address beneficiary) public view returns (uint256) {
            uint256 start = getVestingStart(_storage, beneficiary);
            uint256 fullyVestedAmount = getFullyVestedAmount(_storage, beneficiary);
        
            if (start == 0 || fullyVestedAmount == 0) {
              return 0;
            }
        
            uint256 duration = getVestingDuration(_storage, beneficiary);
            if (duration == 0) {
              return 0;
            }
        
            uint256 revokeDate = getVestingRevokeDate(_storage, beneficiary);
        
            if (now <= start) {
              return 0;
            } else if (revokeDate > 0) {
              return fullyVestedAmount.mul(revokeDate.sub(start)).div(duration);
            } else if (now >= start.add(duration)) {
              return fullyVestedAmount;
            } else {
              return fullyVestedAmount.mul(now.sub(start)).div(duration);
            }
          }
        
          function canGrantVestedTokens(address _storage, address beneficiary) public view returns (bool) {
            uint256 existingFullyVestedAmount = getFullyVestedAmount(_storage, beneficiary);
            if (existingFullyVestedAmount == 0) {
              return true;
            }
        
            uint256 existingVestedAmount = vestedAvailableAmount(_storage, beneficiary);
            uint256 existingReleasedAmount = getVestingReleasedAmount(_storage, beneficiary);
            uint256 revokeDate = getVestingRevokeDate(_storage, beneficiary);
        
            if (revokeDate > 0 ||
                (existingVestedAmount == existingFullyVestedAmount &&
                existingReleasedAmount == existingFullyVestedAmount)) {
              return true;
            }
        
            return false;
          }
        
          function canRevokeVesting(address _storage, address beneficiary) public view returns (bool) {
            bool isRevocable = getVestingRevocable(_storage, beneficiary);
            uint256 revokeDate = getVestingRevokeDate(_storage, beneficiary);
            uint256 start = getVestingStart(_storage, beneficiary);
            uint256 duration = getVestingDuration(_storage, beneficiary);
        
            return start > 0 &&
                   isRevocable &&
                   revokeDate == 0 &&
                   now < start.add(duration);
          }
        
          function revokeVesting(address _storage, address beneficiary) public {
            require(canRevokeVesting(_storage, beneficiary));
        
            uint256 totalUnvestedAndUnreleasedAmount = getTotalUnvestedAndUnreleasedTokens(_storage);
            uint256 unvestedAmount = getFullyVestedAmount(_storage, beneficiary).sub(vestedAvailableAmount(_storage, beneficiary));
        
            setVestingRevokeDate(_storage, beneficiary, now);
            setTotalUnvestedAndUnreleasedTokens(_storage, totalUnvestedAndUnreleasedAmount.sub(unvestedAmount));
          }
        
          function getVestingSchedule(address _storage, address _beneficiary) public
                                                                              view returns (uint256 startDate,
                                                                                                uint256 cliffDate,
                                                                                                uint256 durationSec,
                                                                                                uint256 fullyVestedAmount,
                                                                                                uint256 releasedAmount,
                                                                                                uint256 revokeDate,
                                                                                                bool isRevocable) {
            startDate         = getVestingStart(_storage, _beneficiary);
            cliffDate         = getVestingCliff(_storage, _beneficiary);
            durationSec       = getVestingDuration(_storage, _beneficiary);
            fullyVestedAmount = getFullyVestedAmount(_storage, _beneficiary);
            releasedAmount    = getVestingReleasedAmount(_storage, _beneficiary);
            revokeDate        = getVestingRevokeDate(_storage, _beneficiary);
            isRevocable       = getVestingRevocable(_storage, _beneficiary);
          }
        
          function setVestingSchedule(address _storage,
                                      address beneficiary,
                                      uint256 fullyVestedAmount,
                                      uint256 startDate,
                                      uint256 cliffDate,
                                      uint256 duration,
                                      bool isRevocable) public {
            require(canGrantVestedTokens(_storage, beneficiary));
        
            uint256 totalUnvestedAndUnreleasedAmount = getTotalUnvestedAndUnreleasedTokens(_storage);
            setTotalUnvestedAndUnreleasedTokens(_storage, totalUnvestedAndUnreleasedAmount.add(fullyVestedAmount));
        
            ExternalStorage(_storage).setLedgerValue("cstVestingStart", beneficiary, startDate);
            ExternalStorage(_storage).setLedgerValue("cstVestingCliff", beneficiary, cliffDate);
            ExternalStorage(_storage).setLedgerValue("cstVestingDuration", beneficiary, duration);
            ExternalStorage(_storage).setLedgerValue("cstFullyVestedAmount", beneficiary, fullyVestedAmount);
            ExternalStorage(_storage).setBooleanMapValue("cstVestingRevocable", beneficiary, isRevocable);
        
            setVestingRevokeDate(_storage, beneficiary, 0);
            setVestingReleasedAmount(_storage, beneficiary, 0);
          }
        
          function releaseVestedTokens(address _storage, address beneficiary) public {
            uint256 unreleased = releasableAmount(_storage, beneficiary);
            uint256 releasedAmount = getVestingReleasedAmount(_storage, beneficiary);
            uint256 totalUnvestedAndUnreleasedAmount = getTotalUnvestedAndUnreleasedTokens(_storage);
        
            releasedAmount = releasedAmount.add(unreleased);
            setVestingReleasedAmount(_storage, beneficiary, releasedAmount);
            setTotalUnvestedAndUnreleasedTokens(_storage, totalUnvestedAndUnreleasedAmount.sub(unreleased));
          }
        
          function getVestingStart(address _storage, address beneficiary) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerValue("cstVestingStart", beneficiary);
          }
        
          function getVestingCliff(address _storage, address beneficiary) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerValue("cstVestingCliff", beneficiary);
          }
        
          function getVestingDuration(address _storage, address beneficiary) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerValue("cstVestingDuration", beneficiary);
          }
        
          function getFullyVestedAmount(address _storage, address beneficiary) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerValue("cstFullyVestedAmount", beneficiary);
          }
        
          function getVestingRevocable(address _storage, address beneficiary) public view returns(bool) {
            return ExternalStorage(_storage).getBooleanMapValue("cstVestingRevocable", beneficiary);
          }
        
          function setVestingReleasedAmount(address _storage, address beneficiary, uint256 value) public {
            ExternalStorage(_storage).setLedgerValue("cstVestingReleasedAmount", beneficiary, value);
          }
        
          function getVestingReleasedAmount(address _storage, address beneficiary) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerValue("cstVestingReleasedAmount", beneficiary);
          }
        
          function setVestingRevokeDate(address _storage, address beneficiary, uint256 value) public {
            ExternalStorage(_storage).setLedgerValue("cstVestingRevokeDate", beneficiary, value);
          }
        
          function getVestingRevokeDate(address _storage, address beneficiary) public view returns(uint256) {
            return ExternalStorage(_storage).getLedgerValue("cstVestingRevokeDate", beneficiary);
          }
        
          function getRewardsContractHash(address _storage) public view returns (bytes32) {
            return ExternalStorage(_storage).getBytes32Value("cstRewardsContractHash");
          }
        
          function setRewardsContractHash(address _storage, bytes32 rewardsContractHash) public {
            ExternalStorage(_storage).setBytes32Value("cstRewardsContractHash", rewardsContractHash);
          }
        
        }
        
        contract Displayable {
          function bytes32ToString(bytes32 x) public pure returns (string) {
            bytes memory bytesString = new bytes(32);
            uint256 charCount = 0;
            for (uint256 j = 0; j < 32; j++) {
              if (x[j] != 0) {
                bytesString[charCount] = x[j];
                charCount++;
              }
            }
            bytes memory bytesStringTrimmed = new bytes(charCount);
            for (j = 0; j < charCount; j++) {
              bytesStringTrimmed[j] = bytesString[j];
            }
            return string(bytesStringTrimmed);
          }
        }
        
        contract ERC20 {
          function allowance(address owner, address spender) public view returns (uint256);
          function transferFrom(address from, address to, uint256 value) public returns (bool);
          function approve(address spender, uint256 value) public returns (bool);
          function totalSupply() public view returns (uint256);
          function balanceOf(address account) public view returns (uint256);
          function transfer(address to, uint256 value) public returns (bool);
          event Transfer(address indexed _from, address indexed _to, uint256 _value);
          event Approval(address indexed _owner, address indexed _spender, uint256 _value);
        }
        
        contract ExternalStorage is Administratable {
          using SafeMath for uint256;
        
          mapping(bytes32 => address[]) public primaryLedgerEntryForIndex;
          mapping(bytes32 => mapping(address => address[])) public secondaryLedgerEntryForIndex;
          mapping(bytes32 => address[]) public ledgerEntryForIndex;
          mapping(bytes32 => address[]) public booleanMapEntryForIndex;
        
          mapping(bytes32 => mapping(address => mapping(address => uint256))) private MultiLedgerStorage;
          mapping(bytes32 => mapping(address => bool)) private ledgerPrimaryEntries;
          mapping(bytes32 => mapping(address => mapping(address => bool))) private ledgerSecondaryEntries;
          mapping(bytes32 => mapping(address => uint256)) private LedgerStorage;
          mapping(bytes32 => mapping(address => bool)) private ledgerAccounts;
          mapping(bytes32 => mapping(address => bool)) private BooleanMapStorage;
          mapping(bytes32 => mapping(address => bool)) private booleanMapAccounts;
          mapping(bytes32 => uint256) private UIntStorage;
          mapping(bytes32 => bytes32) private Bytes32Storage;
          mapping(bytes32 => address) private AddressStorage;
          mapping(bytes32 => bytes) private BytesStorage;
          mapping(bytes32 => bool) private BooleanStorage;
          mapping(bytes32 => int256) private IntStorage;
        
          function getMultiLedgerValue(string record, address primaryAddress, address secondaryAddress) external view returns (uint256) {
            return MultiLedgerStorage[keccak256(abi.encodePacked(record))][primaryAddress][secondaryAddress];
          }
        
          function primaryLedgerCount(string record) external view returns (uint256) {
            return primaryLedgerEntryForIndex[keccak256(abi.encodePacked(record))].length;
          }
        
          function secondaryLedgerCount(string record, address primaryAddress) external view returns (uint256) {
            return secondaryLedgerEntryForIndex[keccak256(abi.encodePacked(record))][primaryAddress].length;
          }
        
          function setMultiLedgerValue(string record, address primaryAddress, address secondaryAddress, uint256 value) external onlyAdmins {
            bytes32 hash = keccak256(abi.encodePacked(record));
            if (!ledgerSecondaryEntries[hash][primaryAddress][secondaryAddress]) {
              secondaryLedgerEntryForIndex[hash][primaryAddress].push(secondaryAddress);
              ledgerSecondaryEntries[hash][primaryAddress][secondaryAddress] = true;
        
              if (!ledgerPrimaryEntries[hash][primaryAddress]) {
                primaryLedgerEntryForIndex[hash].push(primaryAddress);
                ledgerPrimaryEntries[hash][primaryAddress] = true;
              }
            }
        
            MultiLedgerStorage[hash][primaryAddress][secondaryAddress] = value;
          }
        
          function getLedgerValue(string record, address _address) external view returns (uint256) {
            return LedgerStorage[keccak256(abi.encodePacked(record))][_address];
          }
        
          function getLedgerCount(string record) external view returns (uint256) {
            return ledgerEntryForIndex[keccak256(abi.encodePacked(record))].length;
          }
        
          function setLedgerValue(string record, address _address, uint256 value) external onlyAdmins {
            bytes32 hash = keccak256(abi.encodePacked(record));
            if (!ledgerAccounts[hash][_address]) {
              ledgerEntryForIndex[hash].push(_address);
              ledgerAccounts[hash][_address] = true;
            }
        
            LedgerStorage[hash][_address] = value;
          }
        
          function getBooleanMapValue(string record, address _address) external view returns (bool) {
            return BooleanMapStorage[keccak256(abi.encodePacked(record))][_address];
          }
        
          function getBooleanMapCount(string record) external view returns (uint256) {
            return booleanMapEntryForIndex[keccak256(abi.encodePacked(record))].length;
          }
        
          function setBooleanMapValue(string record, address _address, bool value) external onlyAdmins {
            bytes32 hash = keccak256(abi.encodePacked(record));
            if (!booleanMapAccounts[hash][_address]) {
              booleanMapEntryForIndex[hash].push(_address);
              booleanMapAccounts[hash][_address] = true;
            }
        
            BooleanMapStorage[hash][_address] = value;
          }
        
          function getUIntValue(string record) external view returns (uint256) {
            return UIntStorage[keccak256(abi.encodePacked(record))];
          }
        
          function setUIntValue(string record, uint256 value) external onlyAdmins {
            UIntStorage[keccak256(abi.encodePacked(record))] = value;
          }
        
          function getBytes32Value(string record) external view returns (bytes32) {
            return Bytes32Storage[keccak256(abi.encodePacked(record))];
          }
        
          function setBytes32Value(string record, bytes32 value) external onlyAdmins {
            Bytes32Storage[keccak256(abi.encodePacked(record))] = value;
          }
        
          function getAddressValue(string record) external view returns (address) {
            return AddressStorage[keccak256(abi.encodePacked(record))];
          }
        
          function setAddressValue(string record, address value) external onlyAdmins {
            AddressStorage[keccak256(abi.encodePacked(record))] = value;
          }
        
          function getBytesValue(string record) external view returns (bytes) {
            return BytesStorage[keccak256(abi.encodePacked(record))];
          }
        
          function setBytesValue(string record, bytes value) external onlyAdmins {
            BytesStorage[keccak256(abi.encodePacked(record))] = value;
          }
        
          function getBooleanValue(string record) external view returns (bool) {
            return BooleanStorage[keccak256(abi.encodePacked(record))];
          }
        
          function setBooleanValue(string record, bool value) external onlyAdmins {
            BooleanStorage[keccak256(abi.encodePacked(record))] = value;
          }
        
          function getIntValue(string record) external view returns (int256) {
            return IntStorage[keccak256(abi.encodePacked(record))];
          }
        
          function setIntValue(string record, int256 value) external onlyAdmins {
            IntStorage[keccak256(abi.encodePacked(record))] = value;
          }
        }
        
        contract Freezable is Administratable {
          using SafeMath for uint256;
        
          // zOS requires that the variables are never removed nor order changed
          // Since this is a parent contract, no new variables can be added here
          bool public frozenToken;
          // TODO move this into external storage
          address[] public frozenAccountForIndex;
          mapping (address => bool) public frozenAccount;
          mapping (address => bool) private processedAccount;
        
          event FrozenFunds(address indexed target, bool frozen);
          event FrozenToken(bool frozen);
        
          modifier unlessFrozen {
            require(!frozenToken);
            require(!frozenAccount[msg.sender]);
            _;
          }
        
          function freezeAccount(address target, bool freeze) public onlySuperAdmins {
            frozenAccount[target] = freeze;
            if (!processedAccount[target]) {
              frozenAccountForIndex.push(target);
              processedAccount[target] = true;
            }
            emit FrozenFunds(target, freeze);
          }
        
          function freezeToken(bool freeze) public onlySuperAdmins {
            frozenToken = freeze;
            emit FrozenToken(frozenToken);
          }
        
          function totalFrozenAccountsMapping() public view returns(uint256) {
            return frozenAccountForIndex.length;
          }
        
        }
        
        contract IStorable {
          function getLedgerNameHash() external view returns (bytes32);
          function getStorageNameHash() external view returns (bytes32);
        }
        
        contract CardstackToken is ERC20,
                                   Initializable,
                                   Freezable,
                                   Displayable,
                                   Configurable,
                                   IStorable {
        
          using SafeMath for uint256;
          using CstLibrary for address;
        
          /* zOS requires that the variables are never removed nor order changed
          /* BEGIN VARIABLES */
          uint8 public constant decimals = 18;
          string public constant version = "3";
          uint256 public constant tokenMaxCap = 6000000000000000000000000000; // 6 billion * 10^18
        
          ITokenLedger public tokenLedger;
          string public storageName;
          string public ledgerName;
          address public externalStorage;
          address public registry;
          bool public haltPurchase;
          uint256 public contributionMinimum;
          /* END VARIABLES */
        
          event Mint(uint256 amountMinted);
          event Approval(address indexed _owner,
                         address indexed _spender,
                         uint256 _value);
          event Transfer(address indexed _from,
                         address indexed _to,
                         uint256 _value);
          event WhiteList(address indexed buyer, uint256 holdCap);
          event RemoveWhitelistedBuyer(address indexed buyer);
          event ConfigChanged(uint256 buyPrice, uint256 circulationCap, uint256 balanceLimit);
          event VestedTokenGrant(address indexed beneficiary, uint256 startDate, uint256 cliffDate, uint256 durationSec, uint256 fullyVestedAmount, bool isRevocable);
          event VestedTokenRevocation(address indexed beneficiary);
          event VestedTokenRelease(address indexed beneficiary, uint256 amount);
          event StorageUpdated(address storageAddress, address ledgerAddress);
          event FoundationDeposit(uint256 amount);
          event FoundationWithdraw(uint256 amount);
          event PurchaseHalted();
          event PurchaseResumed();
        
          modifier onlyFoundation {
            address foundation = externalStorage.getFoundation();
            require(foundation != address(0));
            if (msg.sender != owner && msg.sender != foundation) revert();
            _;
          }
        
          modifier initStorage {
            address ledgerAddress = Registry(registry).getStorage(ledgerName);
            address storageAddress = Registry(registry).getStorage(storageName);
        
            tokenLedger = ITokenLedger(ledgerAddress);
            externalStorage = storageAddress;
            _;
          }
        
          function buy() external payable unlessFrozen returns (uint256) {
            require(!haltPurchase);
            require(externalStorage.getApprovedBuyer(msg.sender));
        
            uint256 _buyPriceTokensPerWei = externalStorage.getBuyPrice();
            uint256 _circulationCap = externalStorage.getCirculationCap();
            require(msg.value > 0);
            require(_buyPriceTokensPerWei > 0);
            require(_circulationCap > 0);
        
            uint256 amount = msg.value.mul(_buyPriceTokensPerWei);
            require(totalInCirculation().add(amount) <= _circulationCap);
            require(amount <= tokensAvailable());
        
            uint256 balanceLimit;
            uint256 buyerBalance = tokenLedger.balanceOf(msg.sender);
            uint256 customLimit = externalStorage.getCustomBuyerLimit(msg.sender);
            require(contributionMinimum == 0 || buyerBalance.add(amount) >= contributionMinimum);
        
            if (customLimit > 0) {
              balanceLimit = customLimit;
            } else {
              balanceLimit = externalStorage.getBalanceLimit();
            }
        
            require(balanceLimit > 0 && balanceLimit >= buyerBalance.add(amount));
        
            tokenLedger.debitAccount(msg.sender, amount);
            emit Transfer(this, msg.sender, amount);
        
            return amount;
          }
        
          function getLedgerNameHash() external view returns (bytes32) {
            return keccak256(abi.encodePacked(ledgerName));
          }
        
          function getStorageNameHash() external view returns (bytes32) {
            return keccak256(abi.encodePacked(storageName));
          }
        
          function initialize(address _registry, string _storageName, string _ledgerName) public onlyInitializers {
            initializeAdmins();
            _initialize(_registry, _storageName, _ledgerName);
          }
        
          function configure(bytes32 _tokenName,
                             bytes32 _tokenSymbol,
                             uint256 _buyPrice,
                             uint256 _circulationCap,
                             uint256 _balanceLimit,
                             address _foundation) public onlySuperAdmins initStorage returns (bool) {
        
            uint256 __buyPrice = externalStorage.getBuyPrice();
            if (__buyPrice > 0 && __buyPrice != _buyPrice) {
              require(frozenToken);
            }
        
            externalStorage.setTokenName(_tokenName);
            externalStorage.setTokenSymbol(_tokenSymbol);
            externalStorage.setBuyPrice(_buyPrice);
            externalStorage.setCirculationCap(_circulationCap);
            externalStorage.setFoundation(_foundation);
            externalStorage.setBalanceLimit(_balanceLimit);
        
            emit ConfigChanged(_buyPrice, _circulationCap, _balanceLimit);
        
            return true;
          }
        
          function configureFromStorage() public onlySuperAdmins initStorage returns (bool) {
            freezeToken(true);
            return true;
          }
        
          function updateStorage(string newStorageName, string newLedgerName) public onlySuperAdmins returns (bool) {
            require(frozenToken);
        
            storageName = newStorageName;
            ledgerName = newLedgerName;
        
            configureFromStorage();
        
            address ledgerAddress = Registry(registry).getStorage(ledgerName);
            address storageAddress = Registry(registry).getStorage(storageName);
            emit StorageUpdated(storageAddress, ledgerAddress);
            return true;
          }
        
          function transfer(address recipient, uint256 amount) public unlessFrozen returns (bool) {
            require(!frozenAccount[recipient]);
        
            tokenLedger.transfer(msg.sender, recipient, amount);
            emit Transfer(msg.sender, recipient, amount);
        
            return true;
          }
        
          function mintTokens(uint256 mintedAmount) public onlySuperAdmins returns (bool) {
            require(mintedAmount.add(totalSupply()) <= tokenMaxCap);
            require(mintedAmount > 0);
        
            tokenLedger.mintTokens(mintedAmount);
        
            emit Mint(mintedAmount);
            emit Transfer(address(0), this, mintedAmount);
        
            return true;
          }
        
          function grantTokens(address recipient, uint256 amount) public onlySuperAdmins returns (bool) {
            require(haltPurchase);
            require(!frozenAccount[recipient]);
        
            uint256 _circulationCap = externalStorage.getCirculationCap();
            require(totalInCirculation().add(amount) <= _circulationCap);
            require(amount <= tokensAvailable()); // assert the granted tokens doesnt exceed the totalSupply minus the fully vested amount of vesting tokens
        
            tokenLedger.debitAccount(recipient, amount);
            emit Transfer(this, recipient, amount);
        
            return true;
          }
        
          function setHaltPurchase(bool _haltPurchase) public onlySuperAdmins returns (bool) {
            haltPurchase = _haltPurchase;
        
            if (_haltPurchase) {
              emit PurchaseHalted();
            } else {
              emit PurchaseResumed();
            }
            return true;
          }
        
          // intentionally allowing this to work when token is frozen as foundation is a form of a super admin
          function foundationWithdraw(uint256 amount) public onlyFoundation returns (bool) {
            /* UNTRUSTED */
            msg.sender.transfer(amount);
        
            emit FoundationWithdraw(amount);
            return true;
          }
        
          function foundationDeposit() public payable unlessFrozen returns (bool) {
            emit FoundationDeposit(msg.value);
        
            return true;
          }
        
          function transferFrom(address from, address to, uint256 value) public unlessFrozen returns (bool) {
            require(!frozenAccount[from]);
            require(!frozenAccount[to]);
            require(from != msg.sender);
        
            uint256 allowanceValue = allowance(from, msg.sender);
            require(allowanceValue >= value);
        
            tokenLedger.transfer(from, to, value);
            externalStorage.setAllowance(from, msg.sender, allowanceValue.sub(value));
        
            emit Transfer(from, to, value);
            return true;
          }
        
          /* Beware that changing an allowance with this method brings the risk that someone may use both the old
           * and the new allowance by unfortunate transaction ordering. One possible solution to mitigate this
           * race condition is to first reduce the spender's allowance to 0 and set the desired value afterwards:
           * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
           *
           * Please use `increaseApproval` or `decreaseApproval` instead.
           */
          function approve(address spender, uint256 value) public unlessFrozen returns (bool) {
            require(value == 0 || allowance(msg.sender, spender) == 0);
        
            return _approve(spender, value);
          }
        
          function increaseApproval(address spender, uint256 addedValue) public unlessFrozen returns (bool) {
            return _approve(spender, externalStorage.getAllowance(msg.sender, spender).add(addedValue));
          }
        
          function decreaseApproval(address spender, uint256 subtractedValue) public unlessFrozen returns (bool) {
            uint256 oldValue = externalStorage.getAllowance(msg.sender, spender);
        
            if (subtractedValue > oldValue) {
              return _approve(spender, 0);
            } else {
              return _approve(spender, oldValue.sub(subtractedValue));
            }
          }
        
          function grantVestedTokens(address beneficiary,
                                     uint256 fullyVestedAmount,
                                     uint256 startDate, // 0 indicates start "now"
                                     uint256 cliffSec,
                                     uint256 durationSec,
                                     bool isRevocable) public onlySuperAdmins returns(bool) {
        
            uint256 _circulationCap = externalStorage.getCirculationCap();
        
            require(beneficiary != address(0));
            require(!frozenAccount[beneficiary]);
            require(durationSec >= cliffSec);
            require(totalInCirculation().add(fullyVestedAmount) <= _circulationCap);
            require(fullyVestedAmount <= tokensAvailable());
        
            uint256 _startDate = startDate;
            if (_startDate == 0) {
              _startDate = now;
            }
        
            uint256 cliffDate = _startDate.add(cliffSec);
        
            externalStorage.setVestingSchedule(beneficiary,
                                               fullyVestedAmount,
                                               _startDate,
                                               cliffDate,
                                               durationSec,
                                               isRevocable);
        
            emit VestedTokenGrant(beneficiary, _startDate, cliffDate, durationSec, fullyVestedAmount, isRevocable);
        
            return true;
          }
        
          function revokeVesting(address beneficiary) public onlySuperAdmins returns (bool) {
            require(beneficiary != address(0));
            externalStorage.revokeVesting(beneficiary);
        
            releaseVestedTokensForBeneficiary(beneficiary);
        
            emit VestedTokenRevocation(beneficiary);
        
            return true;
          }
        
          function releaseVestedTokens() public unlessFrozen returns (bool) {
            return releaseVestedTokensForBeneficiary(msg.sender);
          }
        
          function releaseVestedTokensForBeneficiary(address beneficiary) public unlessFrozen returns (bool) {
            require(beneficiary != address(0));
            require(!frozenAccount[beneficiary]);
        
            uint256 unreleased = releasableAmount(beneficiary);
        
            if (unreleased == 0) { return true; }
        
            externalStorage.releaseVestedTokens(beneficiary);
        
            tokenLedger.debitAccount(beneficiary, unreleased);
            emit Transfer(this, beneficiary, unreleased);
        
            emit VestedTokenRelease(beneficiary, unreleased);
        
            return true;
          }
        
          function setCustomBuyer(address buyer, uint256 balanceLimit) public onlySuperAdmins returns (bool) {
            require(buyer != address(0));
            externalStorage.setCustomBuyerLimit(buyer, balanceLimit);
            addBuyer(buyer);
        
            return true;
          }
        
          function setContributionMinimum(uint256 _contributionMinimum) public onlySuperAdmins returns (bool) {
            contributionMinimum = _contributionMinimum;
            return true;
          }
        
          function addBuyer(address buyer) public onlySuperAdmins returns (bool) {
            require(buyer != address(0));
            externalStorage.setApprovedBuyer(buyer, true);
        
            uint256 balanceLimit = externalStorage.getCustomBuyerLimit(buyer);
            if (balanceLimit == 0) {
              balanceLimit = externalStorage.getBalanceLimit();
            }
        
            emit WhiteList(buyer, balanceLimit);
        
            return true;
          }
        
          function removeBuyer(address buyer) public onlySuperAdmins returns (bool) {
            require(buyer != address(0));
            externalStorage.setApprovedBuyer(buyer, false);
        
            emit RemoveWhitelistedBuyer(buyer);
            return true;
          }
        
          function name() public view returns(string) {
            return bytes32ToString(externalStorage.getTokenName());
          }
        
          function symbol() public view returns(string) {
            return bytes32ToString(externalStorage.getTokenSymbol());
          }
        
          function totalInCirculation() public view returns(uint256) {
            return tokenLedger.totalInCirculation().add(totalUnvestedAndUnreleasedTokens());
          }
        
          function cstBalanceLimit() public view returns(uint256) {
            return externalStorage.getBalanceLimit();
          }
        
          function buyPrice() public view returns(uint256) {
            return externalStorage.getBuyPrice();
          }
        
          function circulationCap() public view returns(uint256) {
            return externalStorage.getCirculationCap();
          }
        
          // intentionally allowing this to be visible if upgraded so foundation can
          // withdraw funds from contract that has a successor
          function foundation() public view returns(address) {
            return externalStorage.getFoundation();
          }
        
          function totalSupply() public view returns(uint256) {
            return tokenLedger.totalTokens();
          }
        
          function tokensAvailable() public view returns(uint256) {
            return totalSupply().sub(totalInCirculation());
          }
        
          function balanceOf(address account) public view returns (uint256) {
            address thisAddress = this;
            if (thisAddress == account) {
              return tokensAvailable();
            } else {
              return tokenLedger.balanceOf(account);
            }
          }
        
          function allowance(address _owner, address _spender) public view returns (uint256) {
            return externalStorage.getAllowance(_owner, _spender);
          }
        
          function releasableAmount(address beneficiary) public view returns (uint256) {
            return externalStorage.releasableAmount(beneficiary);
          }
        
          function totalUnvestedAndUnreleasedTokens() public view returns (uint256) {
            return externalStorage.getTotalUnvestedAndUnreleasedTokens();
          }
        
          function vestingMappingSize() public view returns (uint256) {
            return externalStorage.vestingMappingSize();
          }
        
          function vestingBeneficiaryForIndex(uint256 index) public view returns (address) {
            return externalStorage.vestingBeneficiaryForIndex(index);
          }
        
          function vestingSchedule(address _beneficiary) public
                                                         view returns (uint256 startDate,
                                                                       uint256 cliffDate,
                                                                       uint256 durationSec,
                                                                       uint256 fullyVestedAmount,
                                                                       uint256 vestedAmount,
                                                                       uint256 vestedAvailableAmount,
                                                                       uint256 releasedAmount,
                                                                       uint256 revokeDate,
                                                                       bool isRevocable) {
            (
              startDate,
              cliffDate,
              durationSec,
              fullyVestedAmount,
              releasedAmount,
              revokeDate,
              isRevocable
            ) =  externalStorage.getVestingSchedule(_beneficiary);
        
            vestedAmount = externalStorage.vestedAmount(_beneficiary);
            vestedAvailableAmount = externalStorage.vestedAvailableAmount(_beneficiary);
          }
        
          function totalCustomBuyersMapping() public view returns (uint256) {
            return externalStorage.getCustomBuyerMappingCount();
          }
        
          function customBuyerLimit(address buyer) public view returns (uint256) {
            return externalStorage.getCustomBuyerLimit(buyer);
          }
        
          function customBuyerForIndex(uint256 index) public view returns (address) {
            return externalStorage.getCustomBuyerForIndex(index);
          }
        
          function totalBuyersMapping() public view returns (uint256) {
            return externalStorage.getApprovedBuyerMappingCount();
          }
        
          function approvedBuyer(address buyer) public view returns (bool) {
            return externalStorage.getApprovedBuyer(buyer);
          }
        
          function approvedBuyerForIndex(uint256 index) public view returns (address) {
            return externalStorage.getApprovedBuyerForIndex(index);
          }
        
          function _initialize(address _registry, string _storageName, string _ledgerName) internal isInitializer {
            require(_registry != address(0));
        
            storageName = _storageName;
            ledgerName = _ledgerName;
            registry = _registry;
        
            addSuperAdmin(registry);
        
            emit Transfer(address(0), this, 0); // create ERC-20 signature for etherscan.io
          }
        
          function _approve(address spender, uint256 value) internal unlessFrozen returns(bool) {
            require(spender != address(0));
            require(!frozenAccount[spender]);
            require(msg.sender != spender);
        
            externalStorage.setAllowance(msg.sender, spender, value);
        
            emit Approval(msg.sender, spender, value);
            return true;
          }
        
        }
        
        interface ITokenLedger {
          function mintTokens(uint256 amount) external;
          function transfer(address sender, address reciever, uint256 amount) external;
          function creditAccount(address account, uint256 amount) external;
          function debitAccount(address account, uint256 amount) external;
          function addAdmin(address admin) external;
          function removeAdmin(address admin) external;
          function totalTokens() external view returns (uint256);
          function totalInCirculation() external view returns (uint256);
          function balanceOf(address account) external view returns (uint256);
        }
        
        contract CstLedger is ITokenLedger, Initializable, Administratable {
        
          using SafeMath for uint256;
        
          /* zOS requires that the variables are never removed nor order changed
          /* BEGIN VARIABLES */
          string public constant version = "2";
        
          uint256 private _totalInCirculation; // warning this does not take into account unvested nor vested-unreleased tokens into consideration
          uint256 private _totalTokens;
          mapping (address => uint256) private _balanceOf;
          mapping (address => bool) private accounts;
          /* END VARIABLES */
        
          function transfer(address sender, address recipient, uint256 amount) external onlyAdmins {
            require(sender != address(0));
            require(recipient != address(0));
            require(_balanceOf[sender] >= amount);
        
            _balanceOf[sender] = _balanceOf[sender].sub(amount);
            _balanceOf[recipient] = _balanceOf[recipient].add(amount);
          }
        
          function creditAccount(address account, uint256 amount) external onlyAdmins { // remove tokens
            require(account != address(0));
            require(_balanceOf[account] >= amount);
        
            _totalInCirculation = _totalInCirculation.sub(amount);
            _balanceOf[account] = _balanceOf[account].sub(amount);
          }
        
          function debitAccount(address account, uint256 amount) external onlyAdmins { // add tokens
            require(account != address(0));
            _totalInCirculation = _totalInCirculation.add(amount);
            _balanceOf[account] = _balanceOf[account].add(amount);
          }
        
          function totalTokens() external view returns (uint256) {
            return _totalTokens;
          }
        
          function totalInCirculation() external view returns (uint256) {
            return _totalInCirculation;
          }
        
          function balanceOf(address account) external view returns (uint256) {
            return _balanceOf[account];
          }
        
          function mintTokens(uint256 amount) external onlyAdmins {
            _totalTokens = _totalTokens.add(amount);
          }
        
          function initialize() public onlyInitializers isInitializer {
            initializeAdmins();
          }
        
        }
        
        contract Registry is Initializable, Administratable {
          using SafeMath for uint256;
        
          /* zOS requires that the variables are never removed nor order changed
          /* BEGIN VARIABLES */
          string public constant version = "2";
          bytes4 private constant INTERFACE_META_ID = 0x01ffc9a7;
          bytes4 private constant ADDR_INTERFACE_ID = 0x3b3b57de;
          bytes32 private constant BARE_DOMAIN_NAMEHASH = 0x794941fae74d6435d1b29ee1c08cc39941ba78470872e6afd0693c7eeb63025c; // namehash for "cardstack.eth"
          mapping(bytes32 => address) public storageForHash;
          mapping(bytes32 => address) public contractForHash;
          mapping(bytes32 => bytes32) public hashForNamehash;
          mapping(bytes32 => bytes32) public namehashForHash;
          string[] public contractNameForIndex;
          /* END VARIABLES */
        
          event ContractRegistered(address indexed _contract, string _name, bytes32 namehash);
          event StorageAdded(address indexed storageAddress, string name);
          event StorageRemoved(address indexed storageAddress, string name);
          event AddrChanged(bytes32 indexed node, address a);
        
          function() public {
            revert();
          }
        
          function setNamehash(string contractName, bytes32 namehash) external onlySuperAdmins returns (bool) {
            require(namehash != 0x0);
        
            bytes32 hash = keccak256(abi.encodePacked(contractName));
            address contractAddress = contractForHash[hash];
        
            require(contractAddress != 0x0);
            require(hashForNamehash[namehash] == 0x0);
        
            hashForNamehash[namehash] = hash;
            namehashForHash[hash] = namehash;
        
            emit AddrChanged(namehash, contractAddress);
          }
        
          function register(string name, address contractAddress, bytes32 namehash) external onlySuperAdmins returns (bool) {
            bytes32 hash = keccak256(abi.encodePacked(name));
            require(bytes(name).length > 0);
            require(contractAddress != 0x0);
            require(contractForHash[hash] == 0x0);
            require(hashForNamehash[namehash] == 0x0);
        
            contractNameForIndex.push(name);
            contractForHash[hash] = contractAddress;
        
            if (namehash != 0x0) {
              hashForNamehash[namehash] = hash;
              namehashForHash[hash] = namehash;
            }
        
            address storageAddress = storageForHash[IStorable(contractAddress).getStorageNameHash()];
            address ledgerAddress = storageForHash[IStorable(contractAddress).getLedgerNameHash()];
        
            if (storageAddress != 0x0) {
              ExternalStorage(storageAddress).addAdmin(contractAddress);
            }
            if (ledgerAddress != 0x0) {
              CstLedger(ledgerAddress).addAdmin(contractAddress);
            }
        
            Configurable(contractAddress).configureFromStorage();
        
            emit ContractRegistered(contractAddress, name, namehash);
        
            if (namehash != 0x0) {
              emit AddrChanged(namehash, contractAddress);
            }
        
            return true;
          }
        
          function addStorage(string name, address storageAddress) external onlySuperAdmins {
            require(storageAddress != address(0));
            bytes32 hash = keccak256(abi.encodePacked(name));
            storageForHash[hash] = storageAddress;
        
            emit StorageAdded(storageAddress, name);
          }
        
          function initialize() public onlyInitializers isInitializer {
            initializeAdmins();
          }
        
          function removeStorage(string name) public onlySuperAdmins {
            address storageAddress = storageForHash[keccak256(abi.encodePacked(name))];
            delete storageForHash[keccak256(abi.encodePacked(name))];
        
            emit StorageRemoved(storageAddress, name);
          }
        
          function getStorage(string name) public view returns (address) {
            return storageForHash[keccak256(abi.encodePacked(name))];
          }
        
          function addr(bytes32 node) public view returns (address) {
            return contractForHash[hashForNamehash[node]];
          }
        
          function numContracts() public view returns(uint256) {
            return contractNameForIndex.length;
          }
        
          function getContractHash(string name) public pure returns (bytes32) {
            return keccak256(abi.encodePacked(name));
          }
        
          function supportsInterface(bytes4 interfaceId) public pure returns (bool) {
            return interfaceId == ADDR_INTERFACE_ID ||
                   interfaceId == INTERFACE_META_ID;
          }
        }

        File 3 of 3: AdminUpgradeabilityProxy
        pragma solidity ^0.4.13;
        
        library AddressUtils {
        
          /**
           * Returns whether the target address is a contract
           * @dev This function will return false if invoked during the constructor of a contract,
           * as the code is not actually created until after the constructor finishes.
           * @param _addr address to check
           * @return whether the target address is a contract
           */
          function isContract(address _addr) internal view returns (bool) {
            uint256 size;
            // XXX Currently there is no better way to check if there is a contract in an address
            // than to check the size of the code at that address.
            // See https://ethereum.stackexchange.com/a/14016/36603
            // for more details about how this works.
            // TODO Check this again before the Serenity release, because all addresses will be
            // contracts then.
            // solium-disable-next-line security/no-inline-assembly
            assembly { size := extcodesize(_addr) }
            return size > 0;
          }
        
        }
        
        contract Proxy {
          /**
           * @dev Fallback function.
           * Implemented entirely in `_fallback`.
           */
          function () payable external {
            _fallback();
          }
        
          /**
           * @return The Address of the implementation.
           */
          function _implementation() internal view returns (address);
        
          /**
           * @dev Delegates execution to an implementation contract.
           * This is a low level function that doesn't return to its internal call site.
           * It will return to the external caller whatever the implementation returns.
           * @param implementation Address to delegate.
           */
          function _delegate(address implementation) internal {
            assembly {
              // Copy msg.data. We take full control of memory in this inline assembly
              // block because it will not return to Solidity code. We overwrite the
              // Solidity scratch pad at memory position 0.
              calldatacopy(0, 0, calldatasize)
        
              // Call the implementation.
              // out and outsize are 0 because we don't know the size yet.
              let result := delegatecall(gas, implementation, 0, calldatasize, 0, 0)
        
              // Copy the returned data.
              returndatacopy(0, 0, returndatasize)
        
              switch result
              // delegatecall returns 0 on error.
              case 0 { revert(0, returndatasize) }
              default { return(0, returndatasize) }
            }
          }
        
          /**
           * @dev Function that is run as the first thing in the fallback function.
           * Can be redefined in derived contracts to add functionality.
           * Redefinitions must call super._willFallback().
           */
          function _willFallback() internal {
          }
        
          /**
           * @dev fallback implementation.
           * Extracted to enable manual triggering.
           */
          function _fallback() internal {
            _willFallback();
            _delegate(_implementation());
          }
        }
        
        contract UpgradeabilityProxy is Proxy {
          /**
           * @dev Emitted when the implementation is upgraded.
           * @param implementation Address of the new implementation.
           */
          event Upgraded(address implementation);
        
          /**
           * @dev Storage slot with the address of the current implementation.
           * This is the keccak-256 hash of "org.zeppelinos.proxy.implementation", and is
           * validated in the constructor.
           */
          bytes32 private constant IMPLEMENTATION_SLOT = 0x7050c9e0f4ca769c69bd3a8ef740bc37934f8e2c036e5a723fd8ee048ed3f8c3;
        
          /**
           * @dev Contract constructor.
           * @param _implementation Address of the initial implementation.
           */
          constructor(address _implementation) public {
            assert(IMPLEMENTATION_SLOT == keccak256("org.zeppelinos.proxy.implementation"));
        
            _setImplementation(_implementation);
          }
        
          /**
           * @dev Returns the current implementation.
           * @return Address of the current implementation
           */
          function _implementation() internal view returns (address impl) {
            bytes32 slot = IMPLEMENTATION_SLOT;
            assembly {
              impl := sload(slot)
            }
          }
        
          /**
           * @dev Upgrades the proxy to a new implementation.
           * @param newImplementation Address of the new implementation.
           */
          function _upgradeTo(address newImplementation) internal {
            _setImplementation(newImplementation);
            emit Upgraded(newImplementation);
          }
        
          /**
           * @dev Sets the implementation address of the proxy.
           * @param newImplementation Address of the new implementation.
           */
          function _setImplementation(address newImplementation) private {
            require(AddressUtils.isContract(newImplementation), "Cannot set a proxy implementation to a non-contract address");
        
            bytes32 slot = IMPLEMENTATION_SLOT;
        
            assembly {
              sstore(slot, newImplementation)
            }
          }
        }
        
        contract AdminUpgradeabilityProxy is UpgradeabilityProxy {
          /**
           * @dev Emitted when the administration has been transferred.
           * @param previousAdmin Address of the previous admin.
           * @param newAdmin Address of the new admin.
           */
          event AdminChanged(address previousAdmin, address newAdmin);
        
          /**
           * @dev Storage slot with the admin of the contract.
           * This is the keccak-256 hash of "org.zeppelinos.proxy.admin", and is
           * validated in the constructor.
           */
          bytes32 private constant ADMIN_SLOT = 0x10d6a54a4754c8869d6886b5f5d7fbfa5b4522237ea5c60d11bc4e7a1ff9390b;
        
          /**
           * @dev Modifier to check whether the `msg.sender` is the admin.
           * If it is, it will run the function. Otherwise, it will delegate the call
           * to the implementation.
           */
          modifier ifAdmin() {
            if (msg.sender == _admin()) {
              _;
            } else {
              _fallback();
            }
          }
        
          /**
           * Contract constructor.
           * It sets the `msg.sender` as the proxy administrator.
           * @param _implementation address of the initial implementation.
           */
          constructor(address _implementation) UpgradeabilityProxy(_implementation) public {
            assert(ADMIN_SLOT == keccak256("org.zeppelinos.proxy.admin"));
        
            _setAdmin(msg.sender);
          }
        
          /**
           * @return The address of the proxy admin.
           */
          function admin() external view ifAdmin returns (address) {
            return _admin();
          }
        
          /**
           * @return The address of the implementation.
           */
          function implementation() external view ifAdmin returns (address) {
            return _implementation();
          }
        
          /**
           * @dev Changes the admin of the proxy.
           * Only the current admin can call this function.
           * @param newAdmin Address to transfer proxy administration to.
           */
          function changeAdmin(address newAdmin) external ifAdmin {
            require(newAdmin != address(0), "Cannot change the admin of a proxy to the zero address");
            emit AdminChanged(_admin(), newAdmin);
            _setAdmin(newAdmin);
          }
        
          /**
           * @dev Upgrade the backing implementation of the proxy.
           * Only the admin can call this function.
           * @param newImplementation Address of the new implementation.
           */
          function upgradeTo(address newImplementation) external ifAdmin {
            _upgradeTo(newImplementation);
          }
        
          /**
           * @dev Upgrade the backing implementation of the proxy and call a function
           * on the new implementation.
           * This is useful to initialize the proxied contract.
           * @param newImplementation Address of the new implementation.
           * @param data Data to send as msg.data in the low level call.
           * It should include the signature and the parameters of the function to be
           * called, as described in
           * https://solidity.readthedocs.io/en/develop/abi-spec.html#function-selector-and-argument-encoding.
           */
          function upgradeToAndCall(address newImplementation, bytes data) payable external ifAdmin {
            _upgradeTo(newImplementation);
            require(address(this).call.value(msg.value)(data));
          }
        
          /**
           * @return The admin slot.
           */
          function _admin() internal view returns (address adm) {
            bytes32 slot = ADMIN_SLOT;
            assembly {
              adm := sload(slot)
            }
          }
        
          /**
           * @dev Sets the address of the proxy admin.
           * @param newAdmin Address of the new proxy admin.
           */
          function _setAdmin(address newAdmin) internal {
            bytes32 slot = ADMIN_SLOT;
        
            assembly {
              sstore(slot, newAdmin)
            }
          }
        
          /**
           * @dev Only fall back when the sender is not the admin.
           */
          function _willFallback() internal {
            require(msg.sender != _admin(), "Cannot call fallback function from the proxy admin");
            super._willFallback();
          }
        }