Source Code
Overview
ETH Balance
0 ETH
ETH Value
$0.00
Cross-Chain Transactions
Loading...
Loading
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.
Contract Name:
Options
Compiler Version
v0.8.8+commit.dddeac2f
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../common/access/AccessControl.sol";
import "./IOptions.sol";
contract Options is IOptions, AccessControl {
using OptionsRepo for OptionsRepo.Repo;
using OptionsRepo for OptionsRepo.Option;
using OptionsRepo for OptionsRepo.Head;
using OptionsRepo for bytes32;
using EnumerableSet for EnumerableSet.Bytes32Set;
using EnumerableSet for EnumerableSet.UintSet;
OptionsRepo.Repo private _repo;
// ################
// ## Write I/O ##
// ################
function createOption(
bytes32 snOfOpt,
bytes32 snOfCond,
uint rightholder,
uint paid,
uint par
) external onlyAttorney returns (OptionsRepo.Head memory head) {
head = _repo.createOption(snOfOpt, snOfCond, rightholder, paid, par);
}
function delOption(uint256 seqOfOpt) external onlyAttorney returns(bool flag){
flag = _repo.removeOption(seqOfOpt);
}
function addObligorIntoOpt(
uint256 seqOfOpt,
uint256 obligor
) external onlyAttorney returns (bool flag) {
if (isOption(seqOfOpt))
flag = _repo.records[seqOfOpt].obligors.add(obligor);
}
function removeObligorFromOpt(
uint256 seqOfOpt,
uint256 obligor
) external onlyAttorney returns (bool flag) {
if (isOption(seqOfOpt))
flag = _repo.records[seqOfOpt].obligors.remove(obligor);
}
// ################
// ## Read I/O ##
// ################
// ==== Option ====
function counterOfOptions() external view returns (uint32) {
return _repo.counterOfOptions();
}
function qtyOfOptions() external view returns (uint) {
return _repo.qtyOfOptions();
}
function isOption(uint256 seqOfOpt) public view returns (bool) {
return _repo.isOption(seqOfOpt);
}
function getOption(uint256 seqOfOpt) external view
returns (OptionsRepo.Option memory option)
{
return _repo.getOption(seqOfOpt);
}
function getAllOptions() external view returns (OptionsRepo.Option[] memory)
{
return _repo.getAllOptions();
}
// ==== Obligor ====
function isObligor(uint256 seqOfOpt, uint256 acct) external
view returns (bool)
{
return _repo.isObligor(seqOfOpt, acct);
}
function getObligorsOfOption(uint256 seqOfOpt) external view
returns (uint256[] memory)
{
return _repo.getObligorsOfOption(seqOfOpt);
}
function getSeqList() external view returns(uint[] memory) {
return _repo.getSeqList();
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20 {
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/
event Transfer(address indexed from, address indexed to, uint256 indexed value);
/**
* @dev Emitted when the allowance of a `spender` for an `owner` is set by
* a call to {approve}. `value` is the new allowance.
*/
event Approval(address indexed owner, address indexed spender, uint256 indexed value);
/**
* @dev Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `to`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address to, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: 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
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `from` to `to` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(
address from,
address to,
uint256 amount
) external returns (bool);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../comps/IGeneralKeeper.sol";
import "../comps/common/access/IAccessControl.sol";
import "../lib/UsersRepo.sol";
import "../lib/DocsRepo.sol";
import "./ERC20/IERC20.sol";
import "./Oracles/IPriceConsumer2.sol";
interface IRegCenter is IERC20, IPriceConsumer2{
enum TypeOfDoc{
ZeroPoint,
ROCKeeper, // 1
RODKeeper, // 2
BMMKeeper, // 3
ROMKeeper, // 4
GMMKeeper, // 5
ROAKeeper, // 6
ROOKeeper, // 7
ROPKeeper, // 8
SHAKeeper, // 9
LOOKeeper, // 10
ROC, // 11
ROD, // 12
MeetingMinutes, // 13
ROM, // 14
ROA, // 15
ROO, // 16
ROP, // 17
ROS, // 18
LOO, // 19
GeneralKeeper, // 20
IA, // 21
SHA, // 22
AntiDilution, // 23
LockUp, // 24
Alongs, // 25
Options // 26
}
// ##################
// ## Event ##
// ##################
// ==== Options ====
event SetPlatformRule(bytes32 indexed snOfRule);
event SetPriceFeed(uint indexed seq, address indexed priceFeed);
event TransferOwnership(address indexed newOwner);
event TurnOverCenterKey(address indexed newKeeper);
// ==== Points ====
event MintPoints(uint256 indexed to, uint256 indexed amt);
event TransferPoints(uint256 indexed from, uint256 indexed to, uint256 indexed amt);
event LockPoints(bytes32 indexed headSn, bytes32 indexed hashLock);
event LockConsideration(bytes32 indexed headSn, address indexed counterLocker, bytes payload, bytes32 indexed hashLock);
event PickupPoints(bytes32 indexed headSn);
event PickupConsideration(bytes32 indexed headSn);
event WithdrawPoints(bytes32 indexed headSn);
// ==== Docs ====
event SetTemplate(uint256 indexed typeOfDoc, uint256 indexed version, address indexed body);
event TransferIPR(uint indexed typeOfDoc, uint indexed version, uint indexed transferee);
event CreateDoc(bytes32 indexed snOfDoc, address indexed body);
// ##################
// ## Write ##
// ##################
// ==== Opts Setting ====
function setPlatformRule(bytes32 snOfRule) external;
function setPriceFeed(uint seq, address feed_ ) external;
// ==== Power transfer ====
function transferOwnership(address newOwner) external;
function handoverCenterKey(address newKeeper) external;
// ==== Mint/Sell Points ====
function mint(uint256 to, uint amt) external;
function burn(uint amt) external;
function mintAndLockPoints(uint to, uint amt, uint expireDate, bytes32 hashLock) external;
// ==== Points Trade ====
function lockPoints(uint to, uint amt, uint expireDate, bytes32 hashLock) external;
function lockConsideration(uint to, uint amt, uint expireDate, address counterLocker, bytes memory payload, bytes32 hashLock) external;
function pickupPoints(bytes32 hashLock, string memory hashKey) external;
function withdrawPoints(bytes32 hashLock) external;
function getLocker(bytes32 hashLock) external view
returns (LockersRepo.Locker memory locker);
function getLocksList() external view
returns (bytes32[] memory);
// ==== User ====
function regUser() external;
function setBackupKey(address bKey) external;
function upgradeBackupToPrime() external;
function setRoyaltyRule(bytes32 snOfRoyalty) external;
// ==== Doc ====
function setTemplate(uint typeOfDoc, address body, uint author) external;
function createDoc(bytes32 snOfDoc, address primeKeyOfOwner) external
returns(DocsRepo.Doc memory doc);
// ==== Comp ====
function createComp(address dk) external;
// #################
// ## Read ##
// #################
// ==== Options ====
function getOwner() external view returns (address);
function getBookeeper() external view returns (address);
function getPlatformRule() external returns(UsersRepo.Rule memory);
// ==== Users ====
function isKey(address key) external view returns (bool);
function counterOfUsers() external view returns(uint40);
function getUser() external view returns (UsersRepo.User memory);
function getRoyaltyRule(uint author)external view returns (UsersRepo.Key memory);
function getUserNo(address targetAddr, uint fee, uint author) external returns (uint40);
function getMyUserNo() external returns (uint40);
// ==== Docs ====
function counterOfTypes() external view returns(uint32);
function counterOfVersions(uint256 typeOfDoc) external view returns(uint32 seq);
function counterOfDocs(uint256 typeOfDoc, uint256 version) external view returns(uint64 seq);
function docExist(address body) external view returns(bool);
function getAuthor(uint typeOfDoc, uint version) external view returns(uint40);
function getAuthorByBody(address body) external view returns(uint40);
function getHeadByBody(address body) external view returns (DocsRepo.Head memory );
function getDoc(bytes32 snOfDoc) external view returns(DocsRepo.Doc memory doc);
function getDocByUserNo(uint acct) external view returns (DocsRepo.Doc memory doc);
function verifyDoc(bytes32 snOfDoc) external view returns(bool flag);
function getVersionsList(uint256 typeOfDoc) external view returns(DocsRepo.Doc[] memory);
function getDocsList(bytes32 snOfDoc) external view returns(DocsRepo.Doc[] memory);
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.8;
interface IPriceConsumer2 {
/**
* Network: Arbitrum One
* ETH/USD (Base_0): 0x639Fe6ab55C921f74e7fac1ee960C0B6293ba612
* GBP/USD (quote_1): 0x9C4424Fd84C6661F97D8d6b3fc3C1aAc2BeDd137
* EUR/USD (quote_2): 0xA14d53bC1F1c0F31B4aA3BD109344E5009051a84
* JPY/USD (quote_3): 0x3dD6e51CB9caE717d5a8778CF79A04029f9cFDF8
* KRW/USD (quote_4): 0x85bb02E0Ae286600d1c68Bb6Ce22Cc998d411916
* CNY/USD (quote_5): 0xcC3370Bde6AFE51e1205a5038947b9836371eCCb
* AUD/USD (quote_6): 0x9854e9a850e7C354c1de177eA953a6b1fba8Fc22
* CAD/USD (quote_7): 0xf6DA27749484843c4F02f5Ad1378ceE723dD61d4
* CHF/USD (quote_8): 0xe32AccC8c4eC03F6E75bd3621BfC9Fbb234E1FC3
* ARS/USD (quote_9): 0x0000000000000000000000000000000000000000
* PHP/USD (quote_10): 0xfF82AAF635645fD0bcc7b619C3F28004cDb58574
* NZD/USD (quote_11): 0x0000000000000000000000000000000000000000
* SGD/USD (quote_12): 0xF0d38324d1F86a176aC727A4b0c43c9F9d9c5EB1
* NGN/USD (quote_13): 0x0000000000000000000000000000000000000000
* ZAR/USD (quote_14): 0x0000000000000000000000000000000000000000
* RUB/USD (quote_15): 0x0000000000000000000000000000000000000000
* INR/USD (quote_16): 0x0000000000000000000000000000000000000000
* BRL/USD (quote_17): 0x04b7384473A2aDF1903E3a98aCAc5D62ba8C2702
*/
function getPriceFeed(uint seq) external view returns (address);
function decimals(address quote) external view returns (uint8);
function getCentPriceInWei(uint seq) external view returns (uint);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/OrdersRepo.sol";
import "../../../lib/GoldChain.sol";
interface IListOfOrders {
//################
//## Events ##
//################
event RegInvestor(uint indexed investor, uint indexed groupRep, bytes32 indexed idHash);
event ApproveInvestor(uint indexed investor, uint indexed verifier);
event RevokeInvestor(uint indexed investor, uint indexed verifier);
event PlaceSellOrder(bytes32 indexed sn);
event WithdrawSellOrder(bytes32 indexed sn);
event PlaceBuyOrder(uint caller, uint indexed classOfShare, uint indexed paid, uint indexed price);
event Deal(bytes32 indexed deal);
event OfferExpired(bytes32 indexed offer);
event GetBalance(bytes32 indexed balance);
//#################
//## Write I/O ##
//#################
function regInvestor(
uint acct,
uint groupRep,
bytes32 idHash
) external;
function approveInvestor(
uint userNo,
uint verifier
) external;
function revokeInvestor(
uint userNo,
uint verifier
) external;
function placeSellOrder(
uint classOfShare,
uint seqOfShare,
uint votingWeight,
uint paid,
uint price,
uint execHours,
bool sortFromHead
) external;
function withdrawSellOrder(
uint classOfShare,
uint seqOfOrder
) external returns(GoldChain.Node memory order);
function placeBuyOrder(
uint offeror,
uint classOfShare,
uint paid,
uint price
) external returns (
OrdersRepo.Deal[] memory deals,
GoldChain.Node[] memory expired
);
//################
//## Read I/O ##
//################
// ==== Investor ====
function isInvestor(
uint userNo
) external view returns(bool);
function getInvestor(
uint userNo
) external view returns(OrdersRepo.Investor memory);
function getQtyOfInvestors()
external view returns(uint);
function investorList()
external view returns(uint[] memory);
function investorInfoList()
external view returns(OrdersRepo.Investor[] memory);
// ==== Deals ====
function counterOfOffers(
uint classOfShare
) external view returns(uint32);
function headOfList(
uint classOfShare
) external view returns (uint32);
function tailOfList(
uint classOfShare
) external view returns (uint32);
function lengthOfList(
uint classOfShare
) external view returns (uint);
function getSeqList(
uint classOfShare
) external view returns (uint[] memory);
function getChain(
uint classOfShare
) external view returns (GoldChain.NodeWrap[] memory);
// ==== Order ====
function isOrder(
uint classOfShare,
uint seqOfOrder
) external view returns (bool);
function getOrder(
uint classOfShare,
uint seqOfOrder
) external view returns (GoldChain.Node memory );
// ==== Class ====
function isClass(uint classOfShare) external view returns(bool);
function getClassesList() external view returns(uint[] memory);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/SwapsRepo.sol";
import "../../../lib/DealsRepo.sol";
import "../../common/components/ISigPage.sol";
interface IInvestmentAgreement is ISigPage {
//##################
//## Event ##
//##################
event RegDeal(uint indexed seqOfDeal);
event ClearDealCP(
uint256 indexed seq,
bytes32 indexed hashLock,
uint indexed closingDeadline
);
event CloseDeal(uint256 indexed seq, string indexed hashKey);
event TerminateDeal(uint256 indexed seq);
event CreateSwap(uint seqOfDeal, bytes32 snOfSwap);
event PayOffSwap(uint seqOfDeal, uint seqOfSwap, uint msgValue);
event TerminateSwap(uint seqOfDeal, uint seqOfSwap);
event PayOffApprovedDeal(uint seqOfDeal, uint msgValue);
//##################
//## Write I/O ##
//##################
// ======== InvestmentAgreement ========
function addDeal(
bytes32 sn,
uint buyer,
uint groupOfBuyer,
uint paid,
uint par
) external;
function regDeal(DealsRepo.Deal memory deal) external returns(uint16 seqOfDeal);
function delDeal(uint256 seq) external;
function lockDealSubject(uint256 seq) external returns (bool flag);
function releaseDealSubject(uint256 seq) external returns (bool flag);
function clearDealCP( uint256 seq, bytes32 hashLock, uint closingDeadline) external;
function closeDeal(uint256 seq, string memory hashKey)
external returns (bool flag);
function directCloseDeal(uint256 seq) external returns (bool flag);
function terminateDeal(uint256 seqOfDeal) external returns(bool);
function takeGift(uint256 seq) external returns(bool);
function finalizeIA() external;
// ==== Swap ====
function createSwap (
uint seqOfMotion,
uint seqOfDeal,
uint paidOfTarget,
uint seqOfPledge,
uint caller
) external returns(SwapsRepo.Swap memory swap);
function payOffSwap(
uint seqOfMotion,
uint seqOfDeal,
uint seqOfSwap,
uint msgValue,
uint centPrice
) external returns(SwapsRepo.Swap memory swap);
function terminateSwap(
uint seqOfMotion,
uint seqOfDeal,
uint seqOfSwap
) external returns (SwapsRepo.Swap memory swap);
function payOffApprovedDeal(
uint seqOfDeal,
uint msgValue,
uint caller
) external returns (bool flag);
function requestPriceDiff(
uint seqOfDeal,
uint seqOfShare
) external;
// #####################
// ## Read I/O ##
// #####################
// ======== InvestmentAgreement ========
function getTypeOfIA() external view returns (uint8);
function getDeal(uint256 seq) external view returns (DealsRepo.Deal memory);
function getSeqList() external view returns (uint[] memory);
// ==== Swap ====
function getSwap(uint seqOfDeal, uint256 seqOfSwap)
external view returns (SwapsRepo.Swap memory);
function getAllSwaps(uint seqOfDeal)
external view returns (SwapsRepo.Swap[] memory);
function allSwapsClosed(uint seqOfDeal)
external view returns (bool);
function checkValueOfSwap(uint seqOfDeal, uint seqOfSwap)
external view returns(uint);
function checkValueOfDeal(uint seqOfDeal)
external view returns (uint);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../common/components/IFilesFolder.sol";
import "./IInvestmentAgreement.sol";
import "../../../lib/DTClaims.sol";
import "../../../lib/FRClaims.sol";
import "../../../lib/TopChain.sol";
interface IRegisterOfAgreements is IFilesFolder {
//#################
//## Event ##
//#################
event ClaimFirstRefusal(address indexed ia, uint256 indexed seqOfDeal, uint256 indexed caller);
event AcceptAlongClaims(address indexed ia, uint indexed seqOfDeal);
event ExecAlongRight(address indexed ia, bytes32 indexed snOfDTClaim, bytes32 sigHash);
event ComputeFirstRefusal(address indexed ia, uint256 indexed seqOfDeal);
//#################
//## Write I/O ##
//#################
// ======== RegisterOfAgreements ========
function claimFirstRefusal(
address ia,
uint256 seqOfDeal,
uint256 caller,
bytes32 sigHash
) external;
function computeFirstRefusal(
address ia,
uint256 seqOfDeal
) external returns (FRClaims.Claim[] memory output);
function execAlongRight(
address ia,
bool dragAlong,
uint256 seqOfDeal,
uint256 seqOfShare,
uint paid,
uint par,
uint256 caller,
bytes32 sigHash
) external;
function acceptAlongClaims(
address ia,
uint seqOfDeal
) external returns(DTClaims.Claim[] memory);
function createMockOfIA(address ia) external;
function mockDealOfSell (address ia, uint seller, uint amount) external;
function mockDealOfBuy (address ia, uint buyer, uint groupRep, uint amount) external;
//################
//## Read ##
//################
// ==== FR Claims ====
function hasFRClaims(address ia, uint seqOfDeal) external view returns (bool);
function isFRClaimer(address ia, uint256 acct) external returns (bool);
function getSubjectDealsOfFR(address ia) external view returns(uint[] memory);
function getFRClaimsOfDeal(address ia, uint256 seqOfDeal)
external view returns(FRClaims.Claim[] memory);
function allFRClaimsAccepted(address ia) external view returns (bool);
// ==== DT Claims ====
function hasDTClaims(address ia, uint256 seqOfDeal)
external view returns(bool);
function getSubjectDealsOfDT(address ia)
external view returns(uint256[] memory);
function getDTClaimsOfDeal(address ia, uint256 seqOfDeal)
external view returns(DTClaims.Claim[] memory);
function getDTClaimForShare(address ia, uint256 seqOfDeal, uint256 seqOfShare)
external view returns(DTClaims.Claim memory);
function allDTClaimsAccepted(address ia) external view returns(bool);
// ==== Mock Results ====
function mockResultsOfIA(address ia)
external view
returns (uint40 controllor, uint16 ratio);
function mockResultsOfAcct(address ia, uint256 acct)
external view
returns (uint40 groupRep, uint16 ratio);
// ==== AllClaimsAccepted ====
function allClaimsAccepted(address ia) external view returns(bool);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../common/components/IFilesFolder.sol";
interface IRegisterOfConstitution is IFilesFolder{
//##############
//## Event ##
//##############
event ChangePointer(address indexed pointer);
//##################
//## Write I/O ##
//##################
function changePointer(address body) external;
//################
//## Read ##
//################
function pointer() external view returns (address);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../common/components/ISigPage.sol";
import "../../../lib/EnumerableSet.sol";
import "../../../lib/DocsRepo.sol";
interface IShareholdersAgreement is ISigPage {
enum TitleOfTerm {
ZeroPoint,
AntiDilution, // 1
LockUp, // 2
DragAlong, // 3
TagAlong, // 4
Options // 5
}
// ==== Rules ========
/*
| Seq | Type | Abb | Description |
| 0 | GovernanceRule | GR | Board Constitution and General Rules of GM |
| 1 | VotingRuleOfGM | CI | VR for Capital Increase |
| 2 | | SText | VR for External Share Transfer |
| 3 | | STint | VR for Internal Share Transfer |
| 4 | | 1+3 | VR for CI & STint |
| 5 | | 2+3 | VR for SText & STint |
| 6 | | 1+2+3 | VR for CI & SText & STint |
| 7 | | 1+2 | VR for CI & SText |
| 8 | | SHA | VR for Update SHA |
| 9 | | O-Issue-GM | VR for Ordinary Issues of GeneralMeeting |
| 10 | | S-Issue-GM | VR for Special Issues Of GeneralMeeting |
| 11 | VotingRuleOfBoard | CI | VR for Capital Increase |
| 12 | | SText | VR for External Share Transfer |
| 13 | | STint | VR for Internal Share Transfer |
| 14 | | 1+3 | VR for CI & STint |
| 15 | | 2+3 | VR for SText & STint |
| 16 | | 1+2+3 | VR for CI & SText & STint |
| 17 | | 1+2 | VR for CI & SText |
| 18 | | SHA | VR for Update SHA |
| 19 | | O-Issue-B | VR for Ordinary Issues Of Board |
| 20 | | S-Issue-B | VR for Special Issues Of Board |
| 21 | UnilateralDecision| UniDecPower | UnilateralDicisionPowerWithoutVoting |
...
| 256 | PositionAllocateRule | PA Rule | Management Positions' Allocation Rules |
...
| 512 | FirstRefusalRule | FR for CI...| FR rule for Investment Deal |
...
| 768 | GroupUpdateOrder | GroupUpdate | Grouping Members as per their relationship |
...
| 1024 | ListingRule | ListingRule | Listing Rule for Share Issue & Transfer |
...
*/
struct TermsRepo {
// title => body
mapping(uint256 => address) terms;
EnumerableSet.UintSet seqList;
}
struct RulesRepo {
// seq => rule
mapping(uint256 => bytes32) rules;
EnumerableSet.UintSet seqList;
}
//##################
//## Write I/O ##
//##################
function createTerm(uint typeOfDoc, uint version) external;
function removeTerm(uint typeOfDoc) external;
function addRule(bytes32 rule) external;
function removeRule(uint256 seq) external;
function initDefaultRules() external;
function finalizeSHA() external;
//################
//## Read ##
//################
// ==== Terms ====
function hasTitle(uint256 title) external view returns (bool);
function qtyOfTerms() external view returns (uint256);
function getTitles() external view returns (uint256[] memory);
function getTerm(uint256 title) external view returns (address);
// ==== Rules ====
function hasRule(uint256 seq) external view returns (bool);
function qtyOfRules() external view returns (uint256);
function getRules() external view returns (uint256[] memory);
function getRule(uint256 seq) external view returns (bytes32);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../../lib/LinksRepo.sol";
interface IAlongs {
// ################
// ## Write ##
// ################
function addDragger(bytes32 rule, uint256 dragger) external;
function removeDragger(uint256 dragger) external;
function addFollower(uint256 dragger, uint256 follower) external;
function removeFollower(uint256 dragger, uint256 follower) external;
// ###############
// ## Read I/O ##
// ###############
function isDragger(uint256 dragger) external view returns (bool);
function getLinkRule(uint256 dragger) external view
returns (RulesParser.LinkRule memory);
function isFollower(uint256 dragger, uint256 follower)
external view returns (bool);
function getDraggers() external view returns (uint256[] memory);
function getFollowers(uint256 dragger) external view returns (uint256[] memory);
function priceCheck(
DealsRepo.Deal memory deal
) external view returns (bool);
function isTriggered(address ia, DealsRepo.Deal memory deal) external view returns (bool);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../../lib/DealsRepo.sol";
interface IAntiDilution {
struct Benchmark{
uint16 classOfShare;
uint32 floorPrice;
EnumerableSet.UintSet obligors;
}
struct Ruler {
// classOfShare => Benchmark
mapping(uint256 => Benchmark) marks;
EnumerableSet.UintSet classes;
}
// ################
// ## Write ##
// ################
function addBenchmark(uint256 class, uint price) external;
function removeBenchmark(uint256 class) external;
function addObligor(uint256 class, uint256 obligor) external;
function removeObligor(uint256 class, uint256 obligor) external;
// ############
// ## read ##
// ############
function isMarked(uint256 class) external view returns (bool flag);
function getClasses() external view returns (uint256[] memory);
function getFloorPriceOfClass(uint256 class) external view
returns (uint32 price);
function getObligorsOfAD(uint256 class)
external view returns (uint256[] memory);
function isObligor(uint256 class, uint256 acct)
external view returns (bool flag);
function getGiftPaid(address ia, uint256 seqOfDeal, uint256 seqOfShare)
external view returns (uint64 gift);
function isTriggered(DealsRepo.Deal memory deal, uint seqOfShare) external view returns (bool);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../../lib/ArrayUtils.sol";
import "../../../../lib/BallotsBox.sol";
import "../../../../lib/DealsRepo.sol";
import "../../../../lib/EnumerableSet.sol";
import "../../../common/components/ISigPage.sol";
interface ILockUp {
struct Locker {
uint48 dueDate;
EnumerableSet.UintSet keyHolders;
}
// ################
// ## Write ##
// ################
function setLocker(uint256 seqOfShare, uint dueDate) external;
function delLocker(uint256 seqOfShare) external;
function addKeyholder(uint256 seqOfShare, uint256 keyholder) external;
function removeKeyholder(uint256 seqOfShare, uint256 keyholder) external;
// ################
// ## Read I/O ##
// ################
function isLocked(uint256 seqOfShare) external view returns (bool);
function getLocker(uint256 seqOfShare)
external
view
returns (uint48 dueDate, uint256[] memory keyHolders);
function lockedShares() external view returns (uint256[] memory);
function isTriggered(DealsRepo.Deal memory deal) external view returns (bool);
function isExempted(address ia, DealsRepo.Deal memory deal) external view returns (bool);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../../lib/OptionsRepo.sol";
import "../../../../lib/EnumerableSet.sol";
interface IOptions {
// ################
// ## Write I/O ##
// ################
function createOption(
bytes32 snOfOpt,
bytes32 snOfCond,
uint rightholder,
uint paid,
uint par
) external returns (OptionsRepo.Head memory head);
function delOption(uint256 seqOfOpt) external returns(bool flag);
function addObligorIntoOpt(
uint256 seqOfOpt,
uint256 obligor
) external returns (bool flag);
function removeObligorFromOpt(
uint256 seqOfOpt,
uint256 obligor
) external returns (bool flag);
// ################
// ## Read I/O ##
// ################
// ==== Option ====
function counterOfOptions() external view returns (uint32);
function qtyOfOptions() external view returns (uint);
function isOption(uint256 seqOfOpt) external view returns (bool);
function getOption(uint256 seqOfOpt) external view
returns (OptionsRepo.Option memory option);
function getAllOptions() external view returns (OptionsRepo.Option[] memory);
// ==== Obligor ====
function isObligor(uint256 seqOfOpt, uint256 acct) external
view returns (bool);
function getObligorsOfOption(uint256 seqOfOpt) external view
returns (uint256[] memory);
// ==== snOfOpt ====
function getSeqList() external view returns(uint[] memory);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../common/components/IMeetingMinutes.sol";
import "../../../lib/OfficersRepo.sol";
interface IRegisterOfDirectors {
//###################
//## events ##
//##################
event AddPosition(bytes32 indexed snOfPos);
event RemovePosition(uint256 indexed seqOfPos);
event TakePosition(uint256 indexed seqOfPos, uint256 indexed caller);
event QuitPosition(uint256 indexed seqOfPos, uint256 indexed caller);
event RemoveOfficer(uint256 indexed seqOfPos);
//#################
//## Write I/O ##
//#################
function createPosition(bytes32 snOfPos) external;
function updatePosition(OfficersRepo.Position memory pos) external;
function removePosition(uint256 seqOfPos) external;
function takePosition (uint256 seqOfPos, uint caller) external;
function quitPosition (uint256 seqOfPos, uint caller) external;
function removeOfficer (uint256 seqOfPos) external;
//################
//## Read ##
//################
// ==== Positions ====
function posExist(uint256 seqOfPos) external view returns (bool);
function isOccupied(uint256 seqOfPos) external view returns (bool);
function getPosition(uint256 seqOfPos) external view
returns (OfficersRepo.Position memory);
// ==== Managers ====
function isManager(uint256 acct) external view returns (bool);
function getNumOfManagers() external view returns (uint256);
function getManagersList() external view returns (uint256[] memory);
function getManagersPosList() external view returns(uint[] memory);
// ==== Directors ====
function isDirector(uint256 acct) external view returns (bool);
function getNumOfDirectors() external view returns (uint256);
function getDirectorsList() external view
returns (uint256[] memory);
function getDirectorsPosList() external view
returns (uint256[] memory);
// ==== Executives ====
function hasPosition(uint256 acct, uint256 seqOfPos)
external view returns(bool);
function getPosInHand(uint256 acct)
external view returns (uint256[] memory);
function getFullPosInfoInHand(uint acct)
external view returns (OfficersRepo.Position[] memory);
function hasTitle(uint acct, uint title)
external returns (bool flag);
function hasNominationRight(uint seqOfPos, uint acct)
external view returns (bool);
// ==== seatsCalculator ====
function getBoardSeatsOccupied(uint acct) external view
returns (uint256);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/Checkpoints.sol";
import "../../../lib/MembersRepo.sol";
import "../../../lib/SharesRepo.sol";
import "../../../lib/TopChain.sol";
interface IRegisterOfMembers {
//##################
//## Event ##
//##################
event SetVoteBase(bool indexed basedOnPar);
event CapIncrease(uint indexed votingWeight, uint indexed paid, uint indexed par);
event CapDecrease(uint indexed votingWeight, uint indexed paid, uint indexed par);
event SetMaxQtyOfMembers(uint indexed max);
event SetMinVoteRatioOnChain(uint indexed min);
event SetAmtBase(bool indexed basedOnPar);
event AddMember(uint256 indexed acct, uint indexed qtyOfMembers);
event AddShareToMember(uint indexed seqOfShare, uint indexed acct);
event RemoveShareFromMember(uint indexed seqOfShare, uint indexed acct);
event ChangeAmtOfMember(
uint indexed acct,
uint indexed paid,
uint indexed par,
uint clean,
bool increase
);
event AddMemberToGroup(uint indexed acct, uint indexed root);
event RemoveMemberFromGroup(uint256 indexed acct, uint256 indexed root);
event ChangeGroupRep(uint256 indexed orgRep, uint256 indexed newRep);
//#################
//## Write I/O ##
//#################
function setMaxQtyOfMembers(uint max) external;
function setMinVoteRatioOnChain(uint min) external;
function setVoteBase(bool _basedOnPar) external;
function capIncrease(
uint votingWeight,
uint paid,
uint par,
bool isIncrease
) external;
function addMember(uint256 acct) external;
function addShareToMember(
SharesRepo.Share memory share
) external;
function removeShareFromMember(
SharesRepo.Share memory share
) external;
function increaseAmtOfMember(
uint acct,
uint votingWeight,
uint deltaPaid,
uint deltaPar,
uint deltaClean,
bool isIncrease
) external ;
function addMemberToGroup(uint acct, uint root) external;
function removeMemberFromGroup(uint256 acct) external;
// ##############
// ## Read ##
// ##############
function isMember(uint256 acct) external view returns (bool);
function qtyOfMembers() external view returns (uint);
function membersList() external view returns (uint256[] memory);
function sortedMembersList() external view returns (uint256[] memory);
function qtyOfTopMembers() external view returns (uint);
function topMembersList() external view returns (uint[] memory);
// ---- Cap & Equity ----
function ownersEquity()
external view
returns(Checkpoints.Checkpoint memory);
function capAtDate(uint date)
external view
returns (Checkpoints.Checkpoint memory);
function equityOfMember(uint256 acct)
external view
returns (Checkpoints.Checkpoint memory);
function equityAtDate(uint acct, uint date)
external view returns(Checkpoints.Checkpoint memory);
function votesInHand(uint256 acct)
external view returns (uint64);
function votesAtDate(uint256 acct, uint date)
external view
returns (uint64);
function votesHistory(uint acct)
external view
returns (Checkpoints.Checkpoint[] memory);
// ---- ShareNum ----
function qtyOfSharesInHand(uint acct)
external view returns(uint);
function sharesInHand(uint256 acct)
external view
returns (uint[] memory);
// ---- Class ----
function qtyOfSharesInClass(uint acct, uint class)
external view returns(uint);
function sharesInClass(uint256 acct, uint class)
external view returns (uint[] memory);
function isClassMember(uint256 acct, uint class)
external view returns(bool);
function classesBelonged(uint acct)
external view returns(uint[] memory);
function qtyOfClassMember(uint class)
external view returns(uint);
function getMembersOfClass(uint class)
external view returns(uint256[] memory);
// ---- TopChain ----
function basedOnPar() external view returns (bool);
function maxQtyOfMembers() external view returns (uint32);
function minVoteRatioOnChain() external view returns (uint32);
function totalVotes() external view returns (uint64);
function controllor() external view returns (uint40);
function tailOfChain() external view returns (uint40);
function headOfQueue() external view returns (uint40);
function tailOfQueue() external view returns (uint40);
// ==== group ====
function groupRep(uint256 acct) external view returns (uint40);
function votesOfGroup(uint256 acct) external view returns (uint64);
function deepOfGroup(uint256 acct) external view returns (uint256);
function membersOfGroup(uint256 acct)
external view
returns (uint256[] memory);
function qtyOfGroupsOnChain() external view returns (uint32);
function qtyOfGroups() external view returns (uint256);
function affiliated(uint256 acct1, uint256 acct2)
external view
returns (bool);
// ==== snapshot ====
function getSnapshot() external view returns (TopChain.Node[] memory, TopChain.Para memory);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
import "../roc/terms/IOptions.sol";
import "../../../lib/OptionsRepo.sol";
import "../../../lib/SwapsRepo.sol";
pragma solidity ^0.8.8;
interface IRegisterOfOptions {
// ################
// ## Event ##
// ################
event CreateOpt(uint256 indexed seqOfOpt, bytes32 indexed codeOfOpt);
event IssueOpt(uint256 indexed seqOfOpt, uint indexed issueDate);
event AddObligorIntoOpt(uint256 indexed seqOfOpt, uint256 indexed obligor);
event RemoveObligorFromOpt(uint256 indexed seqOfOpt, uint256 indexed obligor);
event UpdateOracle(uint256 indexed seqOfOpt, uint indexed data1, uint indexed data2, uint data3);
event ExecOpt(uint256 indexed seqOfOpt);
event RegSwap(uint256 indexed seqOfOpt, bytes32 indexed snOfSwap);
event PayOffSwap(uint256 indexed seqOfOpt, bytes32 indexed snOfSwap);
event TerminateSwap(uint256 indexed seqOfOpt, uint indexed seqOfSwap);
// ################
// ## Write ##
// ################
function createOption(
bytes32 sn,
bytes32 snOfCond,
uint rightholder,
uint paid,
uint par
) external returns(OptionsRepo.Head memory head);
function issueOption(OptionsRepo.Option memory opt) external;
function regOptionTerms(address opts) external;
function addObligorIntoOption(uint256 seqOfOpt, uint256 obligor) external;
function removeObligorFromOption(uint256 seqOfOpt, uint256 obligor) external;
function updateOracle(
uint256 seqOfOpt,
uint d1,
uint d2,
uint d3
) external;
function execOption(uint256 seqOfOpt, uint caller) external;
function createSwap(
uint256 seqOfOpt,
uint seqOfTarget,
uint paidOfTarget,
uint seqOfPledge,
uint caller
) external returns (SwapsRepo.Swap memory swap);
function payOffSwap(
uint seqOfOpt,
uint seqOfSwap,
uint msgValue,
uint centPrice
) external returns (SwapsRepo.Swap memory swap);
function terminateSwap(
uint seqOfOpt,
uint seqOfSwap
) external returns (SwapsRepo.Swap memory swap);
// ################
// ## Read I/O ##
// ################
function counterOfOptions() external view returns (uint32);
function qtyOfOptions() external view returns (uint);
function isOption(uint256 seqOfOpt) external view returns (bool);
function getOption(uint256 seqOfOpt) external view
returns (OptionsRepo.Option memory opt);
function getAllOptions() external view returns (OptionsRepo.Option[] memory);
function isRightholder(uint256 seqOfOpt, uint256 acct) external view returns (bool);
function isObligor(uint256 seqOfOpt, uint256 acct) external view returns (bool);
function getObligorsOfOption(uint256 seqOfOpt)
external view returns (uint256[] memory);
function getSeqListOfOptions() external view returns(uint[] memory);
// ==== Swap ====
function counterOfSwaps(uint256 seqOfOpt)
external view returns (uint16);
function sumPaidOfTarget(uint256 seqOfOpt)
external view returns (uint64);
function isSwap(uint256 seqOfOpt, uint256 seqOfSwap)
external view returns (bool);
function getSwap(uint256 seqOfOpt, uint256 seqOfSwap)
external view returns (SwapsRepo.Swap memory swap);
function getAllSwapsOfOption(uint256 seqOfOpt)
external view returns (SwapsRepo.Swap[] memory);
function allSwapsClosed(uint256 seqOfOpt)
external view returns (bool);
// ==== oracles ====
function getOracleAtDate(uint256 seqOfOpt, uint date)
external view returns (Checkpoints.Checkpoint memory);
function getLatestOracle(uint256 seqOfOpt) external
view returns(Checkpoints.Checkpoint memory);
function getAllOraclesOfOption(uint256 seqOfOpt)
external view returns (Checkpoints.Checkpoint[] memory);
// ==== Value ====
function checkValueOfSwap(uint seqOfOpt, uint seqOfSwap)
external view returns (uint);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/PledgesRepo.sol";
interface IRegisterOfPledges {
//##################
//## Event ##
//##################
event CreatePledge(
uint256 indexed seqOfShare,
uint256 indexed seqOfPld,
uint256 creditor,
uint256 indexed paid,
uint256 par
);
event TransferPledge(
uint256 indexed seqOfShare,
uint256 indexed seqOfPld,
uint256 indexed newSeqOfPld,
uint256 buyer,
uint256 paid,
uint256 par
);
event RefundDebt(
uint256 indexed seqOfShare,
uint256 indexed seqOfPld,
uint256 indexed refundAmt
);
event ExtendPledge(
uint256 indexed seqOfShare,
uint256 indexed seqOfPld,
uint256 indexed extDays
);
event LockPledge(uint256 indexed seqOfShare, uint256 indexed seqOfPld, bytes32 indexed hashLock);
event ReleasePledge(uint256 indexed seqOfShare, uint256 indexed seqOfPld, string indexed hashKey);
event ExecPledge(uint256 indexed seqOfShare, uint256 indexed seqOfPld);
event RevokePledge(uint256 indexed seqOfShare, uint256 indexed seqOfPld);
//##################
//## Write I/O ##
//##################
function createPledge(
bytes32 snOfPld,
uint paid,
uint par,
uint guaranteedAmt,
uint execDays
) external returns(PledgesRepo.Head memory head);
function issuePledge(
PledgesRepo.Head memory head,
uint paid,
uint par,
uint guaranteedAmt,
uint execDays
) external returns(PledgesRepo.Head memory regHead);
function regPledge(
PledgesRepo.Pledge memory pld
) external returns(PledgesRepo.Head memory head);
function transferPledge(
uint256 seqOfShare,
uint256 seqOfPld,
uint buyer,
uint amt,
uint caller
) external returns (PledgesRepo.Pledge memory newPld);
function refundDebt(
uint256 seqOfShare,
uint256 seqOfPld,
uint amt,
uint caller
) external returns (PledgesRepo.Pledge memory newPld);
function extendPledge(
uint256 seqOfShare,
uint256 seqOfPld,
uint extDays,
uint caller
) external;
// ==== Lock/Release/Exec/Revoke ====
function lockPledge(
uint256 seqOfShare,
uint256 seqOfPld,
bytes32 hashLock,
uint caller
) external;
function releasePledge(uint256 seqOfShare, uint256 seqOfPld, string memory hashKey)
external returns (uint64);
function execPledge(
uint seqOfShare,
uint256 seqOfPld,
uint caller
) external;
function revokePledge(uint256 seqOfShare, uint256 seqOfPld, uint caller)
external;
//################
//## Read ##
//################
function counterOfPledges(uint256 seqOfShare)
external view returns (uint16);
function isPledge(uint256 seqOfShare, uint256 seqOfPld)
external view returns (bool);
function getSNList() external view
returns(bytes32[] memory);
function getPledge(uint256 seqOfShare, uint256 seqOfPld)
external view returns (PledgesRepo.Pledge memory);
function getPledgesOfShare(uint256 seqOfShare)
external view returns (PledgesRepo.Pledge[] memory);
function getAllPledges() external view
returns (PledgesRepo.Pledge[] memory);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/SharesRepo.sol";
import "../../../lib/LockersRepo.sol";
import "../rom/IRegisterOfMembers.sol";
interface IRegisterOfShares {
//##################
//## Event ##
//##################
event IssueShare(bytes32 indexed shareNumber, uint indexed paid, uint indexed par);
event PayInCapital(uint256 indexed seqOfShare, uint indexed amount);
event SubAmountFromShare(uint256 indexed seqOfShare, uint indexed paid, uint indexed par);
event DeregisterShare(uint256 indexed seqOfShare);
event UpdatePriceOfPaid(uint indexed seqOfShare, uint indexed newPrice);
event UpdatePaidInDeadline(uint256 indexed seqOfShare, uint indexed paidInDeadline);
event DecreaseCleanPaid(uint256 indexed seqOfShare, uint indexed paid);
event IncreaseCleanPaid(uint256 indexed seqOfShare, uint indexed paid);
event SetPayInAmt(bytes32 indexed headSn, bytes32 indexed hashLock);
event WithdrawPayInAmt(uint indexed seqOfShare, uint indexed amount);
event IncreaseEquityOfClass(bool indexed isIncrease, uint indexed class, uint indexed amt);
//##################
//## Write I/O ##
//##################
function issueShare(bytes32 shareNumber, uint payInDeadline, uint paid, uint par) external;
function addShare(SharesRepo.Share memory share) external;
function setPayInAmt(uint seqOfShare, uint amt, uint expireDate, bytes32 hashLock) external;
function requestPaidInCapital(bytes32 hashLock, string memory hashKey) external;
function withdrawPayInAmt(bytes32 hashLock, uint seqOfShare) external;
function payInCapital(uint seqOfShare, uint amt) external;
function transferShare(
uint256 seqOfShare,
uint paid,
uint par,
uint to,
uint priceOfPaid,
uint priceOfPar
) external;
function decreaseCapital(uint256 seqOfShare, uint paid, uint par) external;
// ==== CleanPaid ====
function decreaseCleanPaid(uint256 seqOfShare, uint paid) external;
function increaseCleanPaid(uint256 seqOfShare, uint paid) external;
// ==== State & PaidInDeadline ====
function updatePriceOfPaid(uint seqOfShare, uint newPrice) external;
function updatePaidInDeadline(uint256 seqOfShare, uint paidInDeadline) external;
// ==== EquityOfClass ====
function increaseEquityOfClass(
bool isIncrease,
uint classOfShare,
uint deltaPaid,
uint deltaPar,
uint deltaCleanPaid
) external;
// ##################
// ## Read I/O ##
// ##################
function counterOfShares() external view returns (uint32);
function counterOfClasses() external view returns (uint16);
// ==== SharesRepo ====
function isShare(
uint256 seqOfShare
) external view returns (bool);
function getShare(
uint256 seqOfShare
) external view returns (
SharesRepo.Share memory
);
function getQtyOfShares() external view returns (uint);
function getSeqListOfShares() external view returns (uint[] memory);
function getSharesList() external view returns (SharesRepo.Share[] memory);
// ---- Class ----
function getQtyOfSharesInClass(
uint classOfShare
) external view returns (uint);
function getSeqListOfClass(
uint classOfShare
) external view returns (uint[] memory);
function getInfoOfClass(
uint classOfShare
) external view returns (SharesRepo.Share memory);
function getSharesOfClass(
uint classOfShare
) external view returns (SharesRepo.Share[] memory);
// ==== PayInCapital ====
function getLocker(
bytes32 hashLock
) external view returns (LockersRepo.Locker memory);
function getLocksList() external view returns (bytes32[] memory);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./IAccessControl.sol";
contract AccessControl is IAccessControl {
using RolesRepo for RolesRepo.Repo;
bytes32 private constant _ATTORNEYS = bytes32("Attorneys");
RolesRepo.Repo private _roles;
address private _dk;
IRegCenter internal _rc;
IGeneralKeeper internal _gk;
// ################
// ## Modifier ##
// ################
modifier onlyOwner {
require(
_roles.getOwner() == msg.sender,
"AC.onlyOwner: NOT"
);
_;
}
modifier onlyDK {
require(
_dk == msg.sender,
"AC.onlyDK: NOT"
);
_;
}
modifier onlyGC {
require(
_roles.getRoleAdmin(_ATTORNEYS) == msg.sender,
"AC.onlyGC: NOT"
);
_;
}
modifier onlyKeeper {
require(
_gk.isKeeper(msg.sender) ||
_dk == msg.sender,
"AC.onlyKeeper: NOT"
);
_;
}
modifier onlyAttorney {
require(
_roles.hasRole(_ATTORNEYS, msg.sender),
"AC.onlyAttorney: NOT"
);
_;
}
modifier attorneyOrKeeper {
require(
_roles.hasRole(_ATTORNEYS, msg.sender) ||
_gk.isKeeper(msg.sender),
"AC.md.attorneyOrKeeper: NOT"
);
_;
}
// #################
// ## Write ##
// #################
function init(
address owner,
address directKeeper,
address regCenter,
address generalKeeper
) external {
_roles.initDoc(owner);
_dk = directKeeper;
_rc = IRegCenter(regCenter);
_gk = IGeneralKeeper(generalKeeper);
emit Init(owner, directKeeper, regCenter, generalKeeper);
}
function setOwner(address acct) external {
_roles.setOwner(acct, msg.sender);
emit SetOwner(acct);
}
function setDirectKeeper(address acct) external onlyDK {
_dk = acct;
emit SetDirectKeeper(acct);
}
function takeBackKeys (address target) external onlyDK {
IAccessControl(target).setDirectKeeper(msg.sender);
}
function setRoleAdmin(bytes32 role, address acct) external {
_roles.setRoleAdmin(role, acct, msg.sender);
emit SetRoleAdmin(role, acct);
}
function grantRole(bytes32 role, address acct) external {
_roles.grantRole(role, acct, msg.sender);
}
function revokeRole(bytes32 role, address acct) external {
_roles.revokeRole(role, acct, msg.sender);
}
function renounceRole(bytes32 role) external {
_roles.renounceRole(role, msg.sender);
}
function abandonRole(bytes32 role) external {
_roles.abandonRole(role, msg.sender);
}
function lockContents() public {
require(_roles.state == 1, "AC.lockContents: wrong state");
address owner = msg.sender;
_roles.abandonRole(_ATTORNEYS, owner);
_roles.setOwner(address(0), owner);
_roles.state = 2;
emit LockContents();
}
// ##############
// ## Read ##
// ##############
function getOwner() public view returns (address) {
return _roles.getOwner();
}
function getDK() external view returns (address) {
return _dk;
}
function isFinalized() public view returns (bool) {
return _roles.state == 2;
}
function getRoleAdmin(bytes32 role) public view returns (address) {
return _roles.getRoleAdmin(role);
}
function hasRole(bytes32 role, address acct) public view returns (bool) {
return _roles.hasRole(role, acct);
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/RolesRepo.sol";
import "../../../center/IRegCenter.sol";
import "../../IGeneralKeeper.sol";
interface IAccessControl {
// ##################
// ## Event ##
// ##################
event Init(
address indexed owner,
address indexed directKeeper,
address regCenter,
address indexed generalKeeper
);
event SetOwner(address indexed acct);
event SetDirectKeeper(address indexed keeper);
event SetRoleAdmin(bytes32 indexed role, address indexed acct);
event LockContents();
// ##################
// ## Write ##
// ##################
function init(
address owner,
address directKeeper,
address regCenter,
address generalKeeper
) external;
function setOwner(address acct) external;
function setDirectKeeper(address keeper) external;
function takeBackKeys(address target) external;
function setRoleAdmin(bytes32 role, address acct) external;
function grantRole(bytes32 role, address acct) external;
function revokeRole(bytes32 role, address acct) external;
function renounceRole(bytes32 role) external;
function abandonRole(bytes32 role) external;
function lockContents() external;
// ##################
// ## Read I/O ##
// ##################
function getOwner() external view returns (address);
function getDK() external view returns (address);
function isFinalized() external view returns (bool);
function getRoleAdmin(bytes32 role) external view returns (address);
function hasRole(bytes32 role, address acct) external view returns (bool);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/FilesRepo.sol";
import "../../../lib/RulesParser.sol";
interface IFilesFolder {
//#############
//## Event ##
//#############
event UpdateStateOfFile(address indexed body, uint indexed state);
//#################
//## Write I/O ##
//#################
function regFile(bytes32 snOfDoc, address body) external;
function circulateFile(
address body,
uint16 signingDays,
uint16 closingDays,
RulesParser.VotingRule memory vr,
bytes32 docUrl,
bytes32 docHash
) external;
function proposeFile(address body, uint64 seqOfMotion) external;
function voteCountingForFile(address body, bool approved) external;
function execFile(address body) external;
function terminateFile(address body) external;
function setStateOfFile(address body, uint state) external;
//##################
//## read I/O ##
//##################
function signingDeadline(address body) external view returns (uint48);
function closingDeadline(address body) external view returns (uint48);
function frExecDeadline(address body) external view returns (uint48);
function dtExecDeadline(address body) external view returns (uint48);
function terminateStartpoint(address body) external view returns (uint48);
function votingDeadline(address body) external view returns (uint48);
function isRegistered(address body) external view
returns (bool);
function qtyOfFiles() external view
returns (uint256);
function getFilesList() external view
returns (address[] memory);
function getFile(address body) external view
returns (FilesRepo.File memory);
function getHeadOfFile(address body) external view
returns (FilesRepo.Head memory head);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/BallotsBox.sol";
import "../../../lib/MotionsRepo.sol";
import "../../../lib/RulesParser.sol";
import "../../../lib/DelegateMap.sol";
interface IMeetingMinutes {
//##################
//## events ##
//##################
event CreateMotion(bytes32 indexed snOfMotion, uint256 indexed contents);
event ProposeMotionToGeneralMeeting(uint256 indexed seqOfMotion, uint256 indexed proposer);
event ProposeMotionToBoard(uint256 indexed seqOfMotion, uint256 indexed proposer);
event EntrustDelegate(uint256 indexed seqOfMotion, uint256 indexed delegate, uint256 indexed principal);
event CastVoteInGeneralMeeting(uint256 indexed seqOfMotion, uint256 indexed caller, uint indexed attitude, bytes32 sigHash);
event CastVoteInBoardMeeting(uint256 indexed seqOfMotion, uint256 indexed caller, uint indexed attitude, bytes32 sigHash);
event VoteCounting(uint256 indexed seqOfMotion, uint8 indexed result);
event ExecResolution(uint256 indexed seqOfMotion, uint256 indexed caller);
//#################
//## Write I/O ##
//#################
function nominateOfficer(
uint256 seqOfPos,
uint seqOfVR,
uint canidate,
uint nominator
) external returns(uint64);
function createMotionToRemoveOfficer(
uint256 seqOfPos,
uint seqOfVR,
uint nominator
) external returns(uint64);
function createMotionToApproveDoc(
uint doc,
uint seqOfVR,
uint executor,
uint proposer
) external returns(uint64);
function createMotionToDistributeProfits(
uint amt,
uint expireDate,
uint seqOfVR,
uint executor,
uint proposer
) external returns (uint64);
function createMotionToTransferFund(
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfVR,
uint executor,
uint proposer
) external returns (uint64);
function createAction(
uint seqOfVR,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint executor,
uint proposer
) external returns(uint64);
function proposeMotionToGeneralMeeting(
uint256 seqOfMotion,
uint proposer
) external;
function proposeMotionToBoard (
uint seqOfMotion,
uint caller
) external;
function entrustDelegate(
uint256 seqOfMotion,
uint delegate,
uint principal
) external;
// ==== Vote ====
function castVoteInGeneralMeeting(
uint256 seqOfMotion,
uint attitude,
bytes32 sigHash,
uint256 caller
) external;
function castVoteInBoardMeeting(
uint256 seqOfMotion,
uint attitude,
bytes32 sigHash,
uint256 caller
) external;
// ==== UpdateVoteResult ====
function voteCounting(bool flag0, uint256 seqOfMotion, MotionsRepo.VoteCalBase memory base)
external returns(uint8);
// ==== ExecResolution ====
function execResolution(uint256 seqOfMotion, uint256 contents, uint caller)
external;
function distributeProfits(
uint amt,
uint expireDate,
uint seqOfMotion,
uint caller
) external;
function transferFund(
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfMotion,
uint caller
) external;
function execAction(
uint seqOfVR,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint256 seqOfMotion,
uint caller
) external returns(uint contents);
//################
//## Read ##
//################
// ==== Motions ====
function isProposed(uint256 seqOfMotion) external view returns (bool);
function voteStarted(uint256 seqOfMotion) external view returns (bool);
function voteEnded(uint256 seqOfMotion) external view returns (bool);
// ==== Delegate ====
function getVoterOfDelegateMap(uint256 seqOfMotion, uint256 acct)
external view returns (DelegateMap.Voter memory);
function getDelegateOf(uint256 seqOfMotion, uint acct)
external view returns (uint);
// ==== motion ====
function getMotion(uint256 seqOfMotion)
external view returns (MotionsRepo.Motion memory motion);
// ==== voting ====
function isVoted(uint256 seqOfMotion, uint256 acct) external view returns (bool);
function isVotedFor(
uint256 seqOfMotion,
uint256 acct,
uint atti
) external view returns (bool);
function getCaseOfAttitude(uint256 seqOfMotion, uint atti)
external view returns (BallotsBox.Case memory );
function getBallot(uint256 seqOfMotion, uint256 acct)
external view returns (BallotsBox.Ballot memory);
function isPassed(uint256 seqOfMotion) external view returns (bool);
function getSeqList() external view returns (uint[] memory);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../../lib/ArrayUtils.sol";
import "../../../lib/EnumerableSet.sol";
import "../../../lib/SigsRepo.sol";
interface ISigPage {
event CirculateDoc();
//##################
//## Write I/O ##
//##################
function circulateDoc() external;
function setTiming(bool initPage, uint signingDays, uint closingDays) external;
function addBlank(bool initPage, bool beBuyer, uint256 seqOfDeal, uint256 acct)
external;
function removeBlank(bool initPage, uint256 seqOfDeal, uint256 acct)
external;
function signDoc(bool initPage, uint256 caller, bytes32 sigHash)
external;
function regSig(uint256 signer, uint sigDate, bytes32 sigHash)
external returns(bool flag);
//##################
//## read I/O ##
//##################
function getParasOfPage(bool initPage) external view
returns (SigsRepo.Signature memory);
function circulated() external view returns(bool);
function established() external view
returns (bool flag);
function getCirculateDate() external view returns(uint48);
function getSigningDays() external view returns(uint16);
function getClosingDays() external view returns(uint16);
function getSigDeadline() external view returns(uint48);
function getClosingDeadline() external view returns(uint48);
function isBuyer(bool initPage, uint256 acct)
external view returns(bool flag);
function isSeller(bool initPage, uint256 acct)
external view returns(bool flag);
function isParty(uint256 acct)
external view returns(bool flag);
function isInitSigner(uint256 acct)
external view returns (bool flag);
function isSigner(uint256 acct)
external view returns (bool flag);
function getBuyers(bool initPage)
external view returns (uint256[] memory buyers);
function getSellers(bool initPage)
external view returns (uint256[] memory sellers);
function getParties() external view
returns (uint256[] memory parties);
function getSigOfParty(bool initParty, uint256 acct) external view
returns (
uint256[] memory seqOfDeals,
SigsRepo.Signature memory sig,
bytes32 sigHash
);
function getSigsOfPage(bool initPage) external view
returns (
SigsRepo.Signature[] memory sigsOfBuyer,
SigsRepo.Signature[] memory sigsOfSeller
);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./common/components/IMeetingMinutes.sol";
import "../lib/RolesRepo.sol";
import "../lib/UsersRepo.sol";
import "./keepers/IROCKeeper.sol";
import "./keepers/IRODKeeper.sol";
import "./keepers/IBMMKeeper.sol";
import "./keepers/IROMKeeper.sol";
import "./keepers/IGMMKeeper.sol";
import "./keepers/IROAKeeper.sol";
import "./keepers/IROOKeeper.sol";
import "./keepers/IROPKeeper.sol";
import "./keepers/ISHAKeeper.sol";
import "./keepers/ILOOKeeper.sol";
import "./books/roa/IRegisterOfAgreements.sol";
import "./books/roc/IRegisterOfConstitution.sol";
import "./books/rod/IRegisterOfDirectors.sol";
import "./books/rom/IRegisterOfMembers.sol";
import "./books/roo/IRegisterOfOptions.sol";
import "./books/rop/IRegisterOfPledges.sol";
import "./books/ros/IRegisterOfShares.sol";
import "./books/loo/IListOfOrders.sol";
interface IGeneralKeeper {
struct CompInfo {
uint40 regNum;
uint48 regDate;
uint8 currency;
bytes20 symbol;
string name;
}
// ###############
// ## Event ##
// ###############
event ExecAction(uint256 indexed contents, bool indexed result);
// ######################
// ## AccessControl ##
// ######################
function setCompInfo (
uint8 _currency,
bytes20 _symbol,
string memory _name
) external;
function createCorpSeal() external;
function getCompInfo() external view returns(CompInfo memory);
function getCompUser() external view returns (UsersRepo.User memory);
function regKeeper(uint256 title, address keeper) external;
function isKeeper(address caller) external view returns (bool flag);
function getKeeper(uint256) external view returns(address keeper);
function regBook(uint256 title, address keeper) external;
function getBook(uint256 title) external view returns (address);
// ##################
// ## ROCKeeper ##
// ##################
function createSHA(uint version) external;
function circulateSHA(address body, bytes32 docUrl, bytes32 docHash) external;
function signSHA(address sha, bytes32 sigHash) external;
function activateSHA(address body) external;
function acceptSHA(bytes32 sigHash) external;
// ###################
// ## RODKeeper ##
// ###################
function takeSeat(uint256 seqOfMotion, uint256 seqOfPos) external;
function removeDirector (uint256 seqOfMotion, uint256 seqOfPos) external;
function takePosition(uint256 seqOfMotion, uint256 seqOfPos) external;
function removeOfficer (uint256 seqOfMotion, uint256 seqOfPos) external;
function quitPosition(uint256 seqOfPos) external;
// ###################
// ## BMMKeeper ##
// ###################
function nominateOfficer(uint256 seqOfPos, uint candidate) external;
function createMotionToRemoveOfficer(uint256 seqOfPos) external;
function createMotionToApproveDoc(uint doc, uint seqOfVR, uint executor) external;
function createAction(
uint seqOfVR,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint executor
) external;
function entrustDelegaterForBoardMeeting(uint256 seqOfMotion, uint delegate) external;
function proposeMotionToBoard (uint seqOfMotion) external;
function castVote(uint256 seqOfMotion, uint attitude, bytes32 sigHash) external;
function voteCounting(uint256 seqOfMotion) external;
function execAction(
uint typeOfAction,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint256 seqOfMotion
) external;
// ###################
// ## ROMKeeper ##
// ###################
function setMaxQtyOfMembers(uint max) external;
function setPayInAmt(uint seqOfShare, uint amt, uint expireDate, bytes32 hashLock) external;
function requestPaidInCapital(bytes32 hashLock, string memory hashKey) external;
function withdrawPayInAmt(bytes32 hashLock, uint seqOfShare) external;
function payInCapital(uint seqOfShare, uint amt) external payable;
// ###################
// ## GMMKeeper ##
// ###################
function nominateDirector(uint256 seqOfPos, uint candidate) external;
function createMotionToRemoveDirector(uint256 seqOfPos) external;
function proposeDocOfGM(uint doc, uint seqOfVR, uint executor) external;
function createActionOfGM(
uint seqOfVR,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint executor
) external;
function entrustDelegaterForGeneralMeeting(uint256 seqOfMotion, uint delegate) external;
function proposeMotionToGeneralMeeting(uint256 seqOfMotion) external;
function castVoteOfGM(
uint256 seqOfMotion,
uint attitude,
bytes32 sigHash
) external;
function voteCountingOfGM(uint256 seqOfMotion) external;
function execActionOfGM(
uint seqOfVR,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint256 seqOfMotion
) external;
// ###################
// ## ROAKeeper ##
// ###################
function createIA(uint256 snOfIA) external;
function circulateIA(address body, bytes32 docUrl, bytes32 docHash) external;
function signIA(address ia, bytes32 sigHash) external;
// ======== Deal Closing ========
function pushToCoffer(address ia, uint256 seqOfDeal, bytes32 hashLock, uint closingDeadline) external;
function closeDeal(address ia, uint256 seqOfDeal, string memory hashKey) external;
function issueNewShare(address ia, uint256 seqOfDeal) external;
function transferTargetShare(address ia, uint256 seqOfDeal) external;
function terminateDeal(address ia, uint256 seqOfDeal) external;
function requestToBuy(address ia, uint seqOfDeal, uint paidOfTarget, uint seqOfPledge) external;
function payOffRejectedDeal(
address ia,
uint seqOfDeal,
uint seqOfSwap
) external payable;
function pickupPledgedShare(
address ia,
uint seqOfDeal,
uint seqOfSwap
) external;
function payOffApprovedDeal(
address ia,
uint seqOfDeal
) external payable;
// #################
// ## ROOKeeper ##
// #################
function updateOracle(
uint256 seqOfOpt,
uint d1,
uint d2,
uint d3
) external;
function execOption(uint256 seqOfOpt) external;
function payOffSwap(
uint256 seqOfOpt,
uint256 seqOfSwap
) external payable;
function terminateSwap(
uint256 seqOfOpt,
uint256 seqOfSwap
) external;
// ###################
// ## ROPKeeper ##
// ###################
function createPledge(bytes32 snOfPld, uint paid, uint par, uint guaranteedAmt, uint execDays) external;
function transferPledge(uint256 seqOfShare, uint256 seqOfPld, uint buyer, uint amt)
external;
function refundDebt(uint256 seqOfShare, uint256 seqOfPld, uint amt) external;
function extendPledge(uint256 seqOfShare, uint256 seqOfPld, uint extDays) external;
function lockPledge(uint256 seqOfShare, uint256 seqOfPld, bytes32 hashLock) external;
function releasePledge(uint256 seqOfShare, uint256 seqOfPld, string memory hashKey) external;
function execPledge(bytes32 snOfDeal, uint256 seqOfPld, uint version, uint buyer, uint groupOfBuyer) external;
function revokePledge(uint256 seqOfShare, uint256 seqOfPld) external;
// ###################
// ## SHAKeeper ##
// ###################
// ======= TagAlong ========
function execTagAlong(
address ia,
uint256 seqOfDeal,
uint256 seqOfShare,
uint paid,
uint par,
bytes32 sigHash
) external;
// ======= DragAlong ========
function execDragAlong(
address ia,
uint256 seqOfDeal,
uint256 seqOfShare,
uint paid,
uint par,
bytes32 sigHash
) external;
function acceptAlongDeal(
address ia,
uint256 seqOfDeal,
bytes32 sigHash
) external;
// ======== AntiDilution ========
function execAntiDilution(
address ia,
uint256 seqOfDeal,
uint256 seqOfShare,
bytes32 sigHash
) external;
function takeGiftShares(address ia, uint256 seqOfDeal) external;
// ======== First Refusal ========
function execFirstRefusal(
uint256 seqOfRule,
uint256 seqOfRightholder,
address ia,
uint256 seqOfDeal,
bytes32 sigHash
) external;
function computeFirstRefusal(
address ia,
uint256 seqOfDeal
) external;
// ############
// ## Fund ##
// ############
function getCentPrice() external view returns(uint);
function saveToCoffer(uint acct, uint value) external;
function pickupDeposit() external;
function proposeToDistributeProfits(
uint amt,
uint expireDate,
uint seqOfVR,
uint executor
) external;
function distributeProfits(
uint amt,
uint expireDate,
uint seqOfMotion
) external;
function proposeToTransferFund(
bool toBMM,
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfVR,
uint executor
) external;
function transferFund(
bool fromBMM,
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfMotion
) external;
// #################
// ## LOOKeeper ##
// #################
function regInvestor(uint groupRep, bytes32 idHash) external;
function approveInvestor(uint userNo, uint seqOfLR) external;
function revokeInvestor(uint userNo, uint seqOfLR) external;
function placeInitialOffer(
uint classOfShare,
uint execHours,
uint paid,
uint price,
uint seqOfLR
) external;
function withdrawInitialOffer(
uint classOfShare,
uint seqOfOrder,
uint seqOfLR
) external;
function placeSellOrder(
uint seqOfClass,
uint execHours,
uint paid,
uint price,
uint seqOfLR,
bool sortFromHead
) external;
function withdrawSellOrder(
uint classOfShare,
uint seqOfOrder
) external;
function placeBuyOrder(uint classOfShare, uint paid, uint price) external payable;
// ###############
// ## Routing ##
// ###############
function getROC() external view returns (IRegisterOfConstitution );
function getSHA() external view returns (IShareholdersAgreement);
function getROD() external view returns (IRegisterOfDirectors);
function getBMM() external view returns (IMeetingMinutes);
function getROM() external view returns (IRegisterOfMembers);
function getGMM() external view returns (IMeetingMinutes);
function getROA() external view returns (IRegisterOfAgreements);
function getROO() external view returns (IRegisterOfOptions);
function getROP() external view returns (IRegisterOfPledges);
function getROS() external view returns (IRegisterOfShares);
function getLOO() external view returns (IListOfOrders);
function depositOfMine(uint user) external view returns(uint);
function totalDeposits() external view returns(uint);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../books/rod/IRegisterOfDirectors.sol";
import "../../lib/RulesParser.sol";
interface IBMMKeeper {
function nominateOfficer(
uint256 seqOfPos,
uint candidate,
uint nominator
) external;
function createMotionToRemoveOfficer(
uint256 seqOfPos,
uint nominator
) external;
// ---- Docs ----
function createMotionToApproveDoc(
uint doc,
uint seqOfVR,
uint executor,
uint proposer
) external;
// ---- TransferFund ----
function proposeToTransferFund(
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfVR,
uint executor,
uint proposer
) external;
// ---- Action ----
function createAction(
uint seqOfVR,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint executor,
uint proposer
) external;
// ==== Cast Vote ====
function entrustDelegaterForBoardMeeting(
uint256 seqOfMotion,
uint delegate,
uint caller
) external;
function proposeMotionToBoard (
uint seqOfMotion,
uint caller
) external;
function castVote(
uint256 seqOfMotion,
uint attitude,
bytes32 sigHash,
uint256 caller
) external;
// ==== Vote Counting ====
function voteCounting(uint256 seqOfMotion) external;
// ==== Exec Motion ====
function transferFund(
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfMotion,
uint caller
) external;
function execAction(
uint typeOfAction,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint256 seqOfMotion,
uint caller
) external returns (uint);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../books/roa/IInvestmentAgreement.sol";
import "../../lib/MotionsRepo.sol";
import "../../lib/OfficersRepo.sol";
import "../common/components/ISigPage.sol";
interface IGMMKeeper {
// ################
// ## Motion ##
// ################
function nominateDirector(
uint256 seqOfPos,
uint candidate,
uint nominator
) external;
function createMotionToRemoveDirector(
uint256 seqOfPos,
uint caller
) external;
function proposeDocOfGM(uint doc, uint seqOfVR, uint executor, uint proposer) external;
function proposeToDistributeProfits(
uint amt,
uint expireDate,
uint seqOfVR,
uint executor,
uint caller
) external;
function proposeToTransferFund(
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfVR,
uint executor,
uint proposer
) external;
function createActionOfGM(
uint seqOfVR,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint executor,
uint proposer
) external;
function entrustDelegaterForGeneralMeeting(uint256 seqOfMotion, uint delegate, uint caller) external;
function proposeMotionToGeneralMeeting(uint256 seqOfMotion,uint caller) external;
function castVoteOfGM(
uint256 seqOfMotion,
uint attitude,
bytes32 sigHash,
uint256 caller
) external;
function voteCountingOfGM(uint256 seqOfMotion) external;
function distributeProfits(
uint amt,
uint expireDate,
uint seqOfMotion,
uint caller
) external;
function transferFund(
address to,
bool isCBP,
uint amt,
uint expireDate,
uint seqOfMotion,
uint caller
) external;
function execActionOfGM(
uint typeOfAction,
address[] memory targets,
uint256[] memory values,
bytes[] memory params,
bytes32 desHash,
uint256 seqOfMotion,
uint caller
) external returns(uint);
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../lib/SharesRepo.sol";
interface ILOOKeeper {
//###############
//## Write ##
//###############
function regInvestor(
uint userNo,
uint groupRep,
bytes32 idHash
) external;
function approveInvestor(
uint userNo,
uint caller,
uint seqOfLR
) external;
function revokeInvestor(
uint userNo,
uint caller,
uint seqOfLR
) external;
function placeInitialOffer(
uint caller,
uint classOfShare,
uint execHours,
uint paid,
uint price,
uint seqOfLR
) external;
function withdrawInitialOffer(
uint caller,
uint classOfShare,
uint seqOfOrder,
uint seqOfLR
) external;
function placeSellOrder(
uint caller,
uint seqOfClass,
uint execHours,
uint paid,
uint price,
uint seqOfLR,
bool sortFromHead
) external;
function withdrawSellOrder(
uint caller,
uint classOfShare,
uint seqOfOrder
) external;
function placeBuyOrder(
uint caller,
uint classOfShare,
uint paid,
uint price,
uint msgValue
) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../books/roa/IInvestmentAgreement.sol";
import "../common/components/IFilesFolder.sol";
import "../common/components/ISigPage.sol";
import "../../lib/DocsRepo.sol";
import "../../lib/RulesParser.sol";
import "../../lib/SharesRepo.sol";
interface IROAKeeper {
// #################
// ## Write IO ##
// #################
function createIA(uint256 version, address primeKeyOfCaller, uint caller) external;
function circulateIA(
address ia,
bytes32 docUrl,
bytes32 docHash,
uint256 caller
) external;
function signIA(
address ia,
uint256 caller,
bytes32 sigHash
) external;
// ==== Deal & IA ====
function pushToCoffer(
address ia,
uint256 seqOfDeal,
bytes32 hashLock,
uint closingDeadline,
uint256 caller
) external;
function closeDeal(
address ia,
uint256 seqOfDeal,
string memory hashKey
) external;
function transferTargetShare(
address ia,
uint256 seqOfDeal,
uint256 caller
) external;
function issueNewShare(address ia, uint256 seqOfDeal, uint caller) external;
function terminateDeal(
address ia,
uint256 seqOfDeal,
uint256 caller
) external;
function payOffApprovedDeal(
address ia,
uint seqOfDeal,
uint msgValue,
uint caller
) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../books/roc/IShareholdersAgreement.sol";
import "../books/roc/terms/ILockUp.sol";
import "../common/components/ISigPage.sol";
import "../../lib/OfficersRepo.sol";
import "../../lib/RulesParser.sol";
import "../../lib/DocsRepo.sol";
interface IROCKeeper {
// ############
// ## SHA ##
// ############
function createSHA(uint version, address primeKeyOfCaller, uint caller) external;
function circulateSHA(
address sha,
bytes32 docUrl,
bytes32 docHash,
uint256 caller
) external;
function signSHA(
address sha,
bytes32 sigHash,
uint256 caller
) external;
function activateSHA(address sha, uint256 caller) external;
function acceptSHA(bytes32 sigHash, uint256 caller) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../books/rod/IRegisterOfDirectors.sol";
interface IRODKeeper {
// ==== Directors ====
function takeSeat(
uint256 seqOfMotion,
uint256 seqOfPos,
uint caller
) external;
function removeDirector (
uint256 seqOfMotion,
uint256 seqOfPos,
uint caller
) external;
// ==== Officers ====
function takePosition(
uint256 seqOfMotion,
uint256 seqOfPos,
uint caller
) external;
function removeOfficer (
uint256 seqOfMotion,
uint256 seqOfPos,
uint caller
) external;
// ==== Quit ====
function quitPosition(uint256 seqOfPos, uint caller) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
interface IROMKeeper {
// #################
// ## Write IO ##
// #################
// ==== BOS funcs ====
function setMaxQtyOfMembers(uint max) external;
function setPayInAmt(uint seqOfShare, uint amt, uint expireDate, bytes32 hashLock) external;
function requestPaidInCapital(bytes32 hashLock, string memory hashKey) external;
function withdrawPayInAmt(bytes32 hashLock, uint seqOfShare) external;
function payInCapital(
uint seqOfShare,
uint amt,
uint msgValue,
uint caller
) external;
function decreaseCapital(
uint256 seqOfShare,
uint paid,
uint par
) external;
function updatePaidInDeadline(uint256 seqOfShare, uint line) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../lib/OptionsRepo.sol";
import "../../lib/SwapsRepo.sol";
interface IROOKeeper {
// #################
// ## ROOKeeper ##
// #################
function updateOracle(
uint256 seqOfOpt,
uint d1,
uint d2,
uint d3
) external;
function execOption(uint256 seqOfOpt, uint256 caller)
external;
function createSwap(
uint256 seqOfOpt,
uint seqOfTarget,
uint paidOfTarget,
uint seqOfPledge,
uint256 caller
) external;
function payOffSwap(
uint256 seqOfOpt,
uint256 seqOfSwap,
uint msgValue,
uint caller
) external;
function terminateSwap(
uint256 seqOfOpt,
uint256 seqOfSwap,
uint caller
) external;
// ==== Swap ====
function requestToBuy(
address ia,
uint seqOfDeal,
uint paidOfTarget,
uint seqOfPledge,
uint caller
) external;
function payOffRejectedDeal(
address ia,
uint seqOfDeal,
uint seqOfSwap,
uint msgValue,
uint caller
) external;
function pickupPledgedShare(
address ia,
uint seqOfDeal,
uint seqOfSwap,
uint caller
) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../../lib/PledgesRepo.sol";
import "../../lib/DealsRepo.sol";
import "../../lib/DocsRepo.sol";
import "../../lib/RulesParser.sol";
interface IROPKeeper {
// ###################
// ## ROPKeeper ##
// ###################
function createPledge(
bytes32 snOfPld,
uint paid,
uint par,
uint guaranteedAmt,
uint execDays,
uint256 caller
) external;
function transferPledge(
uint256 seqOfShare,
uint256 seqOfPld,
uint buyer,
uint amt,
uint256 caller
) external;
function refundDebt(
uint256 seqOfShare,
uint256 seqOfPld,
uint amt,
uint256 caller
) external;
function extendPledge(
uint256 seqOfShare,
uint256 seqOfPld,
uint extDays,
uint256 caller
) external;
function lockPledge(
uint256 seqOfShare,
uint256 seqOfPld,
bytes32 hashLock,
uint256 caller
) external;
function releasePledge(
uint256 seqOfShare,
uint256 seqOfPld,
string memory hashKey
) external;
function execPledge(
bytes32 snOfDeal,
uint256 seqOfPld,
uint version,
address primeKeyOfCaller,
uint buyer,
uint groupOfBuyer,
uint256 caller
) external;
function revokePledge(
uint256 seqOfShare,
uint256 seqOfPld,
uint256 caller
) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../books/roc/terms/IAntiDilution.sol";
import "../books/roc/terms/IAlongs.sol";
import "../books/roc/IShareholdersAgreement.sol";
import "../books/roa/IInvestmentAgreement.sol";
import "../common/components/IFilesFolder.sol";
import "../common/components/ISigPage.sol";
import "../../lib/RulesParser.sol";
import "../../lib/SharesRepo.sol";
import "../../lib/FRClaims.sol";
interface ISHAKeeper {
// ======== TagAlong & DragAlong ========
function execAlongRight(
address ia,
uint256 seqOfDeal,
bool dragAlong,
uint256 seqOfShare,
uint paid,
uint par,
uint256 caller,
bytes32 sigHash
) external;
function acceptAlongDeal(
address ia,
uint256 seqOfDeal,
uint256 caller,
bytes32 sigHash
) external;
// ======== AntiDilution ========
function execAntiDilution(
address ia,
uint256 seqOfDeal,
uint256 seqOfShare,
uint256 caller,
bytes32 sigHash
) external;
function takeGiftShares(
address ia,
uint256 seqOfDeal,
uint caller
) external;
// ======== FirstRefusal ========
function execFirstRefusal(
uint256 seqOfFRRule,
uint256 seqOfRightholder,
address ia,
uint256 seqOfDeal,
uint256 caller,
bytes32 sigHash
) external;
function computeFirstRefusal(
address ia,
uint256 seqOfDeal,
uint256 caller
) external;
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library ArrayUtils {
function merge(uint256[] memory arrA, uint256[] memory arrB)
public pure returns(uint256[] memory)
{
uint256[] memory arrC = new uint256[](arrA.length + arrB.length);
uint256 lenC;
(arrC, lenC) = filter(arrA, arrC, 0);
(arrC, lenC) = filter(arrB, arrC, lenC);
return resize(arrC, lenC);
}
function filter(uint256[] memory arrA, uint256[] memory arrC, uint256 lenC)
public pure returns(uint256[] memory, uint256)
{
uint256 lenA = arrA.length;
uint256 i;
while (i < lenA) {
uint256 j;
while (j < lenC){
if (arrA[i] == arrC[j]) break;
j++;
}
if (j == lenC) {
arrC[lenC] = arrA[i];
lenC++;
}
i++;
}
return (arrC, lenC);
}
function refine(uint256[] memory arrA)
public pure returns(uint256[] memory)
{
uint256[] memory arrB = new uint256[](arrA.length);
uint256 lenB;
(arrB, lenB) = filter(arrA, arrB, 0);
return resize(arrB, lenB);
}
function resize(uint256[] memory arrA, uint256 len)
public pure returns(uint256[] memory)
{
uint256[] memory output = new uint256[](len);
while (len > 0) {
output[len - 1] = arrA[len - 1];
len--;
}
return output;
}
function combine(uint256[] memory arrA, uint256[] memory arrB)
public pure returns (uint256[] memory)
{
uint256 lenA = arrA.length;
uint256 lenB = arrB.length;
uint256 i;
uint256[] memory arrC = new uint256[](lenA + lenB);
for (i = 0; i < lenA; i++) arrC[i] = arrA[i];
for (i = 0; i < lenB; i++) arrC[lenA + i] = arrB[i];
return arrC;
}
function minus(uint256[] memory arrA, uint256[] memory arrB)
public pure returns (uint256[] memory)
{
uint256 lenA = arrA.length;
uint256 lenB = arrB.length;
uint256[] memory arrC = new uint256[](lenA);
uint256 pointer;
while (lenA > 0) {
bool flag = false;
lenB = arrB.length;
while (lenB > 0) {
if (arrB[lenB - 1] == arrA[lenA - 1]) {
flag = true;
break;
}
lenB--;
}
if (!flag) {
arrC[pointer] = arrA[lenA - 1];
pointer++;
}
lenA--;
}
return resize(arrC, pointer);
}
function fullyCoveredBy(uint256[] memory arrA, uint256[] memory arrB)
public pure returns (bool)
{
uint256[] memory arrAr = refine(arrA);
uint256[] memory arrBr = refine(arrB);
uint256 lenA = arrAr.length;
uint256 lenB = arrBr.length;
while (lenA > 0) {
uint256 i;
while (i < lenB) {
if (arrBr[i] == arrAr[lenA-1]) break;
i++;
}
if (i==lenB) return false;
lenA--;
}
return true;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library BallotsBox {
enum AttitudeOfVote {
All,
Support,
Against,
Abstain
}
struct Ballot {
uint40 acct;
uint8 attitude;
uint32 head;
uint64 weight;
uint48 sigDate;
uint64 blocknumber;
bytes32 sigHash;
uint[] principals;
}
struct Case {
uint32 sumOfHead;
uint64 sumOfWeight;
uint256[] voters;
uint256[] principals;
}
struct Box {
mapping(uint256 => Case) cases;
mapping(uint256 => Ballot) ballots;
}
// #################
// ## Write ##
// #################
function castVote(
Box storage box,
uint acct,
uint attitude,
uint head,
uint weight,
bytes32 sigHash,
uint[] memory principals
) public returns (bool flag) {
require(
attitude == uint8(AttitudeOfVote.Support) ||
attitude == uint8(AttitudeOfVote.Against) ||
attitude == uint8(AttitudeOfVote.Abstain),
"BB.CV: attitude overflow"
);
Ballot storage b = box.ballots[acct];
if (b.sigDate == 0) {
box.ballots[acct] = Ballot({
acct: uint40(acct),
attitude: uint8(attitude),
head: uint32(head),
weight: uint64(weight),
sigDate: uint48(block.timestamp),
blocknumber: uint64(block.number),
sigHash: sigHash,
principals: principals
});
_pushToCase(box.cases[attitude], b);
_pushToCase(box.cases[uint8(AttitudeOfVote.All)], b);
flag = true;
}
}
function _pushToCase(Case storage c, Ballot memory b) private {
c.sumOfHead += b.head;
c.sumOfWeight += b.weight;
c.voters.push(b.acct);
uint len = b.principals.length;
while (len > 0) {
c.principals.push(b.principals[len - 1]);
len--;
}
}
// #################
// ## Read ##
// #################
function isVoted(Box storage box, uint256 acct)
public view returns (bool)
{
return box.ballots[acct].sigDate > 0;
}
function isVotedFor(
Box storage box,
uint256 acct,
uint256 atti
) public view returns (bool) {
return box.ballots[acct].attitude == atti;
}
function getCaseOfAttitude(Box storage box, uint256 atti)
public view returns (Case memory )
{
return box.cases[atti];
}
function getBallot(Box storage box, uint256 acct)
public view returns (Ballot memory)
{
return box.ballots[acct];
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library Checkpoints {
struct Checkpoint {
uint48 timestamp;
uint16 votingWeight;
uint64 paid;
uint64 par;
uint64 cleanPaid;
}
struct History {
// checkpoints[0].timestamp : counter
mapping (uint256 => Checkpoint) checkpoints;
}
//##################
//## Write I/O ##
//##################
function push(
History storage self,
uint weight,
uint paid,
uint par,
uint cleanPaid
) public {
uint256 pos = counterOfPoints(self);
uint48 timestamp = uint48 (block.timestamp);
Checkpoint memory point = Checkpoint({
timestamp: timestamp,
votingWeight: uint16(weight),
paid: uint64(paid),
par: uint64(par),
cleanPaid: uint64(cleanPaid)
});
if (self.checkpoints[pos].timestamp == timestamp) {
self.checkpoints[pos] = point;
} else {
self.checkpoints[pos+1] = point;
_increaseCounter(self);
}
}
function _increaseCounter(History storage self)
public
{
self.checkpoints[0].timestamp++;
}
//################
//## Read ##
//################
function counterOfPoints(History storage self)
public view returns (uint256)
{
return self.checkpoints[0].timestamp;
}
function latest(History storage self)
public view returns (Checkpoint memory point)
{
point = self.checkpoints[counterOfPoints(self)];
}
function _average(uint256 a, uint256 b) private pure returns (uint256) {
return (a & b) + ((a ^ b) >> 1);
}
function getAtDate(History storage self, uint256 timestamp)
public view returns (Checkpoint memory point)
{
require(
timestamp <= block.timestamp,
"Checkpoints: block not yet mined"
);
uint256 high = counterOfPoints(self) + 1;
uint256 low = 1;
while (low < high) {
uint256 mid = _average(low, high);
if (self.checkpoints[mid].timestamp > timestamp) {
high = mid;
} else {
low = mid + 1;
}
}
if (high > 1) point = self.checkpoints[high - 1];
}
function pointsOfHistory(History storage self)
public view returns (Checkpoint[] memory)
{
uint256 len = counterOfPoints(self);
Checkpoint[] memory output = new Checkpoint[](len);
while (len > 0) {
output[len-1] = self.checkpoints[len];
len--;
}
return output;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
library CondsRepo {
using EnumerableSet for EnumerableSet.Bytes32Set;
enum LogOps {
ZeroPoint, // 0
And,
Or,
Equal,
NotEqual, // 4
AndAnd,
OrOr,
AndOr,
OrAnd,
EqEq,
NeNe,
EqNe,
NeEq,
AndEq,
EqAnd,
OrEq,
EqOr,
AndNe,
NeAnd,
OrNe,
NeOr
}
enum ComOps {
ZeroPoint,
Equal,
NotEqual,
Bigger,
Smaller,
BiggerOrEqual,
SmallerOrEqual
}
struct Cond {
uint32 seqOfCond;
uint8 logicOpr;
uint8 compOpr1;
uint64 para1;
uint8 compOpr2;
uint64 para2;
uint8 compOpr3;
uint64 para3;
}
struct Repo {
mapping(uint256 => Cond) conds;
EnumerableSet.Bytes32Set seqList;
}
// ###############
// ## Write I/O ##
// ###############
// ==== codify / parser ====
function snParser(bytes32 sn) public pure returns(Cond memory cond)
{
uint _sn = uint(sn);
cond = Cond({
seqOfCond: uint32(_sn >> 224),
logicOpr: uint8(_sn >> 216),
compOpr1: uint8(_sn >> 208),
para1: uint64(_sn >> 144),
compOpr2: uint8(_sn >> 136),
para2: uint64(_sn >> 72),
compOpr3: uint8(_sn >> 64),
para3: uint64(_sn)
});
}
function codifyCond(Cond memory cond) public pure returns(bytes32 sn)
{
bytes memory _sn = abi.encodePacked(
cond.seqOfCond,
cond.logicOpr,
cond.compOpr1,
cond.para1,
cond.compOpr2,
cond.para2,
cond.compOpr3,
cond.para3);
assembly {
sn := mload(add(_sn, 0x20))
}
}
// ==== create / reg ====
function createCond(Repo storage repo, bytes32 sn) public returns(uint32 seqOfCond)
{
seqOfCond = regCond(repo, snParser(sn));
}
function regCond(Repo storage repo, Cond memory cond) public returns(uint32 seqOfCond)
{
cond.seqOfCond = _increaseCounterOfConds(repo);
repo.conds[cond.seqOfCond] = cond;
repo.seqList.add(codifyCond(cond));
seqOfCond = cond.seqOfCond;
}
function _increaseCounterOfConds(Repo storage repo) private returns(uint32)
{
repo.conds[0].seqOfCond++;
return repo.conds[0].seqOfCond;
}
function removeCond(Repo storage repo, uint256 seqOfCond) public returns(bool flag)
{
if (repo.seqList.remove(codifyCond(repo.conds[seqOfCond])))
{
delete repo.conds[seqOfCond];
flag = true;
}
}
// ##################
// ## Write I/O ##
// ##################
function counterOfConds(Repo storage repo) public view returns(uint32 seqOfCond) {
seqOfCond = repo.conds[0].seqOfCond;
}
function getConds(Repo storage repo) public view returns(Cond[] memory)
{
uint256 len = repo.seqList.length();
Cond[] memory output = new Cond[](len);
while (len > 0) {
output[len -1] = repo.conds[len];
len--;
}
return output;
}
function checkCond(
uint compOpr,
uint para,
uint data
) public pure returns (bool flag) {
if (compOpr == uint8(ComOps.Equal)) flag = data == para;
else if (compOpr == uint8(ComOps.NotEqual)) flag = data != para;
else if (compOpr == uint8(ComOps.Bigger)) flag = data > para;
else if (compOpr == uint8(ComOps.Smaller)) flag = data < para;
else if (compOpr == uint8(ComOps.BiggerOrEqual)) flag = data >= para;
else if (compOpr == uint8(ComOps.SmallerOrEqual)) flag = data <= para;
else revert ("CR.CSC: compOpr overflow");
}
function checkSoleCond(
Cond memory cond,
uint data
) public pure returns (bool flag) {
flag = checkCond(cond.compOpr1, cond.para1, data);
}
function checkCondsOfTwo(
Cond memory cond,
uint data1,
uint data2
) public pure returns (bool flag) {
bool flag1;
bool flag2;
flag1 = checkCond(cond.compOpr1, cond.para1, data1);
flag2 = checkCond(cond.compOpr2, cond.para2, data2);
if (cond.logicOpr == uint8(LogOps.And)) flag = flag1 && flag2;
else if (cond.logicOpr == uint8(LogOps.Or)) flag = flag1 || flag2;
else if (cond.logicOpr == uint8(LogOps.Equal)) flag = flag1 == flag2;
else if (cond.logicOpr == uint8(LogOps.NotEqual)) flag = flag1 != flag2;
else revert("CR.CCO2: logicOpr overflow");
}
function checkCondsOfThree(
Cond memory cond,
uint data1,
uint data2,
uint data3
) public pure returns (bool flag) {
bool flag1;
bool flag2;
bool flag3;
flag1 = checkCond(cond.compOpr1, cond.para1, data1);
flag2 = checkCond(cond.compOpr2, cond.para2, data2);
flag3 = checkCond(cond.compOpr3, cond.para3, data3);
if (cond.logicOpr == uint8(LogOps.AndAnd)) flag = flag1 && flag2 && flag3;
else if (cond.logicOpr == uint8(LogOps.OrOr)) flag = flag1 || flag2 || flag3;
else if (cond.logicOpr == uint8(LogOps.AndOr)) flag = flag1 && flag2 || flag3;
else if (cond.logicOpr == uint8(LogOps.OrAnd)) flag = flag1 || flag2 && flag3;
else if (cond.logicOpr == uint8(LogOps.EqEq)) flag = flag1 == flag2 == flag3;
else if (cond.logicOpr == uint8(LogOps.NeNe)) flag = flag1 != flag2 != flag3;
else if (cond.logicOpr == uint8(LogOps.EqNe)) flag = flag1 == flag2 != flag3;
else if (cond.logicOpr == uint8(LogOps.NeEq)) flag = flag1 != flag2 == flag3;
else if (cond.logicOpr == uint8(LogOps.AndEq)) flag = flag1 && flag2 == flag3;
else if (cond.logicOpr == uint8(LogOps.EqAnd)) flag = flag1 == flag2 && flag3;
else if (cond.logicOpr == uint8(LogOps.OrEq)) flag = flag1 || flag2 == flag3;
else if (cond.logicOpr == uint8(LogOps.EqOr)) flag = flag1 == flag2 || flag3;
else if (cond.logicOpr == uint8(LogOps.AndNe)) flag = flag1 && flag2 != flag3;
else if (cond.logicOpr == uint8(LogOps.NeAnd)) flag = flag1 != flag2 && flag3;
else if (cond.logicOpr == uint8(LogOps.OrNe)) flag = flag1 || flag2 != flag3;
else if (cond.logicOpr == uint8(LogOps.NeOr)) flag = flag1 != flag2 || flag3;
else revert("CR.CCO3: logicOpr overflow");
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
import "./MotionsRepo.sol";
import "./SwapsRepo.sol";
import "./SharesRepo.sol";
import "../comps/common/components/IMeetingMinutes.sol";
import "../comps/books/ros/IRegisterOfShares.sol";
library DealsRepo {
using EnumerableSet for EnumerableSet.UintSet;
using SwapsRepo for SwapsRepo.Repo;
// _deals[0].head {
// seqOfDeal: counterOfClosedDeal;
// preSeq: counterOfDeal;
// typeOfDeal: typeOfIA;
// }
enum TypeOfDeal {
ZeroPoint,
CapitalIncrease,
ShareTransferExt,
ShareTransferInt,
PreEmptive,
TagAlong,
DragAlong,
FirstRefusal,
FreeGift
}
enum TypeOfIA {
ZeroPoint,
CapitalIncrease,
ShareTransferExt,
ShareTransferInt,
CI_STint,
SText_STint,
CI_SText_STint,
CI_SText
}
enum StateOfDeal {
Drafting,
Locked,
Cleared,
Closed,
Terminated
}
struct Head {
uint8 typeOfDeal;
uint16 seqOfDeal;
uint16 preSeq;
uint16 classOfShare;
uint32 seqOfShare;
uint40 seller;
uint32 priceOfPaid;
uint32 priceOfPar;
uint48 closingDeadline;
uint16 votingWeight;
}
struct Body {
uint40 buyer;
uint40 groupOfBuyer;
uint64 paid;
uint64 par;
uint8 state;
uint16 para;
uint16 argu;
bool flag;
}
struct Deal {
Head head;
Body body;
bytes32 hashLock;
}
struct Repo {
mapping(uint256 => Deal) deals;
mapping(uint256 => SwapsRepo.Repo) swaps;
//seqOfDeal => seqOfShare => bool
mapping(uint => mapping(uint => bool)) priceDiffRequested;
EnumerableSet.UintSet seqList;
}
//##################
//## Modifier ##
//##################
modifier onlyCleared(Repo storage repo, uint256 seqOfDeal) {
require(
repo.deals[seqOfDeal].body.state == uint8(StateOfDeal.Cleared),
"DR.mf.OC: wrong stateOfDeal"
);
_;
}
modifier dealExist(Repo storage repo, uint seqOfDeal) {
require(isDeal(repo, seqOfDeal), "DR.mf.dealExist: not");
_;
}
//#################
//## Write I/O ##
//#################
function snParser(bytes32 sn) public pure returns(Head memory head) {
uint _sn = uint(sn);
head = Head({
typeOfDeal: uint8(_sn >> 248),
seqOfDeal: uint16(_sn >> 232),
preSeq: uint16(_sn >> 216),
classOfShare: uint16(_sn >> 200),
seqOfShare: uint32(_sn >> 168),
seller: uint40(_sn >> 128),
priceOfPaid: uint32(_sn >> 96),
priceOfPar: uint32(_sn >> 64),
closingDeadline: uint48(_sn >> 16),
votingWeight: uint16(_sn)
});
}
function codifyHead(Head memory head) public pure returns(bytes32 sn) {
bytes memory _sn = abi.encodePacked(
head.typeOfDeal,
head.seqOfDeal,
head.preSeq,
head.classOfShare,
head.seqOfShare,
head.seller,
head.priceOfPaid,
head.priceOfPaid,
head.closingDeadline,
head.votingWeight);
assembly {
sn := mload(add(_sn, 0x20))
}
}
function addDeal(
Repo storage repo,
bytes32 sn,
uint buyer,
uint groupOfBuyer,
uint paid,
uint par
) public returns (uint16 seqOfDeal) {
Deal memory deal;
deal.head = snParser(sn);
deal.body.buyer = uint40(buyer);
deal.body.groupOfBuyer = uint40(groupOfBuyer);
deal.body.paid = uint64(paid);
deal.body.par = uint64(par);
seqOfDeal = regDeal(repo, deal);
}
function regDeal(Repo storage repo, Deal memory deal)
public returns(uint16 seqOfDeal)
{
require(deal.body.par > 0, "DR.RD: zero par");
require(deal.body.par >= deal.body.paid, "DR.RD: paid overflow");
deal.head.seqOfDeal = _increaseCounterOfDeal(repo);
repo.seqList.add(deal.head.seqOfDeal);
repo.deals[deal.head.seqOfDeal] = Deal({
head: deal.head,
body: deal.body,
hashLock: bytes32(0)
});
seqOfDeal = deal.head.seqOfDeal;
}
function _increaseCounterOfDeal(Repo storage repo) private returns(uint16 seqOfDeal){
repo.deals[0].head.preSeq++;
seqOfDeal = repo.deals[0].head.preSeq;
}
function delDeal(Repo storage repo, uint256 seqOfDeal) public returns (bool flag) {
if (repo.seqList.remove(seqOfDeal)) {
delete repo.deals[seqOfDeal];
repo.deals[0].head.preSeq--;
flag = true;
}
}
function lockDealSubject(Repo storage repo, uint256 seqOfDeal) public returns (bool flag) {
if (repo.deals[seqOfDeal].body.state == uint8(StateOfDeal.Drafting)) {
repo.deals[seqOfDeal].body.state = uint8(StateOfDeal.Locked);
flag = true;
}
}
function releaseDealSubject(Repo storage repo, uint256 seqOfDeal) public returns (bool flag)
{
uint8 state = repo.deals[seqOfDeal].body.state;
if ( state < uint8(StateOfDeal.Closed) ) {
repo.deals[seqOfDeal].body.state = uint8(StateOfDeal.Drafting);
flag = true;
} else if (state == uint8(StateOfDeal.Terminated)) {
flag = true;
}
}
function clearDealCP(
Repo storage repo,
uint256 seqOfDeal,
bytes32 hashLock,
uint closingDeadline
) public {
Deal storage deal = repo.deals[seqOfDeal];
require(deal.body.state == uint8(StateOfDeal.Locked),
"IA.CDCP: wrong Deal state");
deal.body.state = uint8(StateOfDeal.Cleared);
deal.hashLock = hashLock;
if (closingDeadline > 0) {
if (block.timestamp < closingDeadline)
deal.head.closingDeadline = uint48(closingDeadline);
else revert ("IA.clearDealCP: updated closingDeadline not FUTURE time");
}
}
function closeDeal(Repo storage repo, uint256 seqOfDeal, string memory hashKey)
public onlyCleared(repo, seqOfDeal) returns (bool flag)
{
require(
repo.deals[seqOfDeal].hashLock == keccak256(bytes(hashKey)),
"IA.closeDeal: hashKey NOT correct"
);
return _closeDeal(repo, seqOfDeal);
}
function directCloseDeal(Repo storage repo, uint seqOfDeal)
public returns (bool flag)
{
require(repo.deals[seqOfDeal].body.state == uint8(StateOfDeal.Locked),
"IA.directCloseDeal: wrong state of deal");
return _closeDeal(repo, seqOfDeal);
}
function _closeDeal(Repo storage repo, uint seqOfDeal)
private returns(bool flag)
{
Deal storage deal = repo.deals[seqOfDeal];
require(
block.timestamp < deal.head.closingDeadline,
"IA.closeDeal: MISSED closing date"
);
deal.body.state = uint8(StateOfDeal.Closed);
_increaseCounterOfClosedDeal(repo);
flag = (counterOfDeal(repo) == counterOfClosedDeal(repo));
}
function terminateDeal(Repo storage repo, uint256 seqOfDeal) public returns(bool flag){
Body storage body = repo.deals[seqOfDeal].body;
require(body.state == uint8(StateOfDeal.Locked) ||
body.state == uint8(StateOfDeal.Cleared)
, "DR.TD: wrong stateOfDeal");
body.state = uint8(StateOfDeal.Terminated);
_increaseCounterOfClosedDeal(repo);
flag = (counterOfDeal(repo) == counterOfClosedDeal(repo));
}
function takeGift(Repo storage repo, uint256 seqOfDeal)
public returns (bool flag)
{
Deal storage deal = repo.deals[seqOfDeal];
require(
deal.head.typeOfDeal == uint8(TypeOfDeal.FreeGift),
"not a gift deal"
);
require(
repo.deals[deal.head.preSeq].body.state == uint8(StateOfDeal.Closed),
"Capital Increase not closed"
);
require(deal.body.state == uint8(StateOfDeal.Locked), "wrong state");
deal.body.state = uint8(StateOfDeal.Closed);
_increaseCounterOfClosedDeal(repo);
flag = (counterOfDeal(repo) == counterOfClosedDeal(repo));
}
function _increaseCounterOfClosedDeal(Repo storage repo) private {
repo.deals[0].head.seqOfDeal++;
}
function calTypeOfIA(Repo storage repo) public {
uint[3] memory types;
uint[] memory seqList = repo.seqList.values();
uint len = seqList.length;
while (len > 0) {
uint typeOfDeal = repo.deals[seqList[len-1]].head.typeOfDeal;
len--;
if (typeOfDeal == 1) {
if (types[0] == 0) types[0] = 1;
continue;
} else if (typeOfDeal == 2) {
if (types[1] == 0) types[1] = 2;
continue;
} else if (typeOfDeal == 3) {
if (types[2] == 0) types[2] = 3;
continue;
}
}
uint8 sum = uint8(types[0] + types[1] + types[2]);
repo.deals[0].head.typeOfDeal = (sum == 3)
? (types[2] == 0)
? 7
: 3
: sum;
}
// ==== Swap ====
function createSwap(
Repo storage repo,
uint seqOfMotion,
uint seqOfDeal,
uint paidOfTarget,
uint seqOfPledge,
uint caller,
IRegisterOfShares _ros,
IMeetingMinutes _gmm
) public returns(SwapsRepo.Swap memory swap) {
Deal storage deal = repo.deals[seqOfDeal];
require(caller == deal.head.seller,
"DR.createSwap: not seller");
require(deal.body.state == uint8(StateOfDeal.Terminated),
"DR.createSwap: wrong state");
MotionsRepo.Motion memory motion =
_gmm.getMotion(seqOfMotion);
require(
motion.body.state == uint8(MotionsRepo.StateOfMotion.Rejected_ToBuy),
"DR.createSwap: NO need to buy"
);
require(block.timestamp < motion.body.voteEndDate +
uint48(motion.votingRule.execDaysForPutOpt) * 86400,
"DR.createSwap: missed deadline");
swap = SwapsRepo.Swap({
seqOfSwap: 0,
seqOfPledge: uint32(seqOfPledge),
paidOfPledge: 0,
seqOfTarget: deal.head.seqOfShare,
paidOfTarget: uint64(paidOfTarget),
priceOfDeal: deal.head.priceOfPaid,
isPutOpt: true,
state: uint8(SwapsRepo.StateOfSwap.Issued)
});
SharesRepo.Head memory headOfPledge = _ros.getShare(swap.seqOfPledge).head;
require(_gmm.getBallot(seqOfMotion, _gmm.getDelegateOf(seqOfMotion,
headOfPledge.shareholder)).attitude == 2,
"DR.createSwap: not vetoer");
require (deal.body.paid >= repo.swaps[seqOfDeal].sumPaidOfTarget() +
swap.paidOfTarget, "DR.createSwap: paidOfTarget overflow");
swap.paidOfPledge = (swap.priceOfDeal - _ros.getShare(swap.seqOfTarget).head.priceOfPaid) *
swap.paidOfTarget / headOfPledge.priceOfPaid;
return repo.swaps[seqOfDeal].regSwap(swap);
}
function payOffSwap(
Repo storage repo,
uint seqOfMotion,
uint seqOfDeal,
uint seqOfSwap,
uint msgValue,
uint centPrice,
IMeetingMinutes _gmm
) public returns(SwapsRepo.Swap memory){
MotionsRepo.Motion memory motion = _gmm.getMotion(seqOfMotion);
require(block.timestamp < motion.body.voteEndDate +
uint48(motion.votingRule.execDaysForPutOpt) * 86400,
"DR.payOffSwap: missed deadline");
return repo.swaps[seqOfDeal].payOffSwap(seqOfSwap, msgValue, centPrice);
}
function terminateSwap(
Repo storage repo,
uint seqOfMotion,
uint seqOfDeal,
uint seqOfSwap,
IMeetingMinutes _gmm
) public returns (SwapsRepo.Swap memory){
MotionsRepo.Motion memory motion = _gmm.getMotion(seqOfMotion);
require(block.timestamp >= motion.body.voteEndDate +
uint48(motion.votingRule.execDaysForPutOpt) * 86400,
"DR.terminateSwap: still in exec period");
return repo.swaps[seqOfDeal].terminateSwap(seqOfSwap);
}
function payOffApprovedDeal(
Repo storage repo,
uint seqOfDeal,
uint caller
) public returns (bool flag){
Deal storage deal = repo.deals[seqOfDeal];
require(deal.head.typeOfDeal != uint8(TypeOfDeal.FreeGift),
"DR.payApprDeal: free gift");
require(caller == deal.body.buyer,
"DR.payApprDeal: not buyer");
require(deal.body.state == uint8(StateOfDeal.Locked) ||
deal.body.state == uint8(StateOfDeal.Cleared) ,
"DR.payApprDeal: wrong state");
require(block.timestamp < deal.head.closingDeadline,
"DR.payApprDeal: missed closingDeadline");
deal.body.state = uint8(StateOfDeal.Closed);
_increaseCounterOfClosedDeal(repo);
flag = (counterOfDeal(repo) == counterOfClosedDeal(repo));
}
function requestPriceDiff(
Repo storage repo,
uint seqOfDeal,
uint seqOfShare
) public dealExist(repo, seqOfDeal) {
require(!repo.priceDiffRequested[seqOfDeal][seqOfShare],
"DR.requestPriceDiff: already requested");
repo.priceDiffRequested[seqOfDeal][seqOfShare] = true;
}
// ##########################
// ## Read I/O ##
// ##########################
function getTypeOfIA(Repo storage repo) external view returns (uint8) {
return repo.deals[0].head.typeOfDeal;
}
function counterOfDeal(Repo storage repo) public view returns (uint16) {
return repo.deals[0].head.preSeq;
}
function counterOfClosedDeal(Repo storage repo) public view returns (uint16) {
return repo.deals[0].head.seqOfDeal;
}
function isDeal(Repo storage repo, uint256 seqOfDeal) public view returns (bool) {
return repo.seqList.contains(seqOfDeal);
}
function getDeal(Repo storage repo, uint256 seq)
external view dealExist(repo, seq) returns (Deal memory)
{
return repo.deals[seq];
}
function getSeqList(Repo storage repo) external view returns (uint[] memory) {
return repo.seqList.values();
}
// ==== Swap ====
function counterOfSwaps(Repo storage repo, uint seqOfDeal)
public view returns (uint16)
{
return repo.swaps[seqOfDeal].counterOfSwaps();
}
function sumPaidOfTarget(Repo storage repo, uint seqOfDeal)
public view returns (uint64)
{
return repo.swaps[seqOfDeal].sumPaidOfTarget();
}
function isSwap(Repo storage repo, uint seqOfDeal, uint256 seqOfSwap)
public view returns (bool)
{
return repo.swaps[seqOfDeal].isSwap(seqOfSwap);
}
function getSwap(Repo storage repo, uint seqOfDeal, uint256 seqOfSwap)
public view returns (SwapsRepo.Swap memory)
{
return repo.swaps[seqOfDeal].getSwap(seqOfSwap);
}
function getAllSwaps(Repo storage repo, uint seqOfDeal)
public view returns (SwapsRepo.Swap[] memory )
{
return repo.swaps[seqOfDeal].getAllSwaps();
}
function allSwapsClosed(Repo storage repo, uint seqOfDeal)
public view returns (bool)
{
return repo.swaps[seqOfDeal].allSwapsClosed();
}
// ==== Value Calculation ====
function checkValueOfSwap(
Repo storage repo,
uint seqOfDeal,
uint seqOfSwap,
uint centPrice
) public view dealExist(repo, seqOfDeal) returns (uint) {
return repo.swaps[seqOfDeal].checkValueOfSwap(seqOfSwap, centPrice);
}
function checkValueOfDeal(
Repo storage repo,
uint seqOfDeal,
uint centPrice
) public view returns (uint) {
Deal memory deal = repo.deals[seqOfDeal];
return (uint(deal.body.paid * deal.head.priceOfPaid) +
uint((deal.body.par - deal.body.paid) * deal.head.priceOfPar)) *
centPrice / 100;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../comps/books/rom/IRegisterOfMembers.sol";
import "../comps/books/rod/IRegisterOfDirectors.sol";
library DelegateMap {
struct LeavesInfo {
uint64 weight;
uint32 emptyHead;
}
struct Voter {
uint40 delegate;
uint64 weight;
uint64 repWeight;
uint32 repHead;
uint[] principals;
}
struct Map {
mapping(uint256 => Voter) voters;
}
// #################
// ## Write ##
// #################
function entrustDelegate(
Map storage map,
uint principal,
uint delegate,
uint weight
) public returns (bool flag) {
require(principal != 0, "DM.ED: zero principal");
require(delegate != 0, "DM.ED: zero delegate");
require(principal != delegate,"DM.ED: self delegate");
if (map.voters[principal].delegate == 0 &&
map.voters[delegate].delegate == 0)
{
Voter storage p = map.voters[principal];
Voter storage d = map.voters[delegate];
p.delegate = uint40(delegate);
p.weight = uint64(weight);
d.repHead += (p.repHead + 1);
d.repWeight += (p.repWeight + p.weight);
d.principals.push(uint40(principal));
_consolidatePrincipals(p.principals, d);
flag = true;
}
}
function _consolidatePrincipals(uint[] memory principals, Voter storage d) private {
uint len = principals.length;
while (len > 0) {
d.principals.push(principals[len-1]);
len--;
}
}
// #################
// ## Read ##
// #################
function getDelegateOf(Map storage map, uint acct)
public
view
returns (uint d)
{
while (acct > 0) {
d = acct;
acct = map.voters[d].delegate;
}
}
function updateLeavesWeightAtDate(Map storage map, uint256 acct, uint baseDate, IRegisterOfMembers _rom)
public
{
LeavesInfo memory info;
Voter storage voter = map.voters[acct];
uint[] memory leaves = voter.principals;
uint256 len = leaves.length;
while (len > 0) {
uint64 w = _rom.votesAtDate(leaves[len-1], baseDate);
if (w > 0) {
info.weight += w;
} else {
info.emptyHead++;
}
len--;
}
voter.weight = _rom.votesAtDate(acct, baseDate);
voter.repWeight = info.weight;
voter.repHead = uint32(leaves.length) - info.emptyHead;
}
function updateLeavesHeadcountOfDirectors(Map storage map, uint256 acct, IRegisterOfDirectors _rod)
public
{
uint[] memory leaves = map.voters[acct].principals;
uint256 len = leaves.length;
uint32 repHead;
while (len > 0) {
if (_rod.isDirector(leaves[len-1])) repHead++;
len--;
}
map.voters[acct].repHead = repHead;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library DocsRepo {
struct Head {
uint32 typeOfDoc;
uint32 version;
uint64 seqOfDoc;
uint40 author;
uint40 creator;
uint48 createDate;
}
struct Body {
uint64 seq;
address addr;
}
struct Doc {
Head head;
address body;
}
struct Repo {
// typeOfDoc => version => seqOfDoc => Body
mapping(uint256 => mapping(uint256 => mapping(uint256 => Body))) bodies;
mapping(address => Head) heads;
}
//##################
//## Write I/O ##
//##################
function snParser(bytes32 sn) public pure returns(Head memory head) {
uint _sn = uint(sn);
head.typeOfDoc = uint32(_sn >> 224);
head.version = uint32(_sn >> 192);
head.seqOfDoc = uint64(_sn >> 128);
head.author = uint40(_sn >> 88);
}
function codifyHead(Head memory head) public pure returns(bytes32 sn) {
bytes memory _sn = abi.encodePacked(
head.typeOfDoc,
head.version,
head.seqOfDoc,
head.author,
head.creator,
head.createDate);
assembly {
sn := mload(add(_sn, 0x20))
}
}
function setTemplate(
Repo storage repo,
uint typeOfDoc,
address body,
uint author,
uint caller
) public returns (Head memory head) {
head.typeOfDoc = uint32(typeOfDoc);
head.author = uint40(author);
head.creator = uint40(caller);
require(body != address(0), "DR.setTemplate: zero address");
require(head.typeOfDoc > 0, "DR.setTemplate: zero typeOfDoc");
if (head.typeOfDoc > counterOfTypes(repo))
head.typeOfDoc = _increaseCounterOfTypes(repo);
require(head.author > 0, "DR.setTemplate: zero author");
require(head.creator > 0, "DR.setTemplate: zero creator");
head.version = _increaseCounterOfVersions(repo, head.typeOfDoc);
head.createDate = uint48(block.timestamp);
repo.bodies[head.typeOfDoc][head.version][0].addr = body;
repo.heads[body] = head;
}
function createDoc(
Repo storage repo,
bytes32 snOfDoc,
uint creator
) public returns (Doc memory doc)
{
doc.head = snParser(snOfDoc);
doc.head.creator = uint40(creator);
require(doc.head.typeOfDoc > 0, "DR.createDoc: zero typeOfDoc");
require(doc.head.version > 0, "DR.createDoc: zero version");
require(doc.head.creator > 0, "DR.createDoc: zero creator");
address temp = repo.bodies[doc.head.typeOfDoc][doc.head.version][0].addr;
require(temp != address(0), "DR.createDoc: template not ready");
doc.head.author = repo.heads[temp].author;
doc.head.seqOfDoc = _increaseCounterOfDocs(repo, doc.head.typeOfDoc, doc.head.version);
doc.head.createDate = uint48(block.timestamp);
doc.body = _createClone(temp);
repo.bodies[doc.head.typeOfDoc][doc.head.version][doc.head.seqOfDoc].addr = doc.body;
repo.heads[doc.body] = doc.head;
}
function transferIPR(
Repo storage repo,
uint typeOfDoc,
uint version,
uint transferee,
uint caller
) public {
require (caller == getAuthor(repo, typeOfDoc, version),
"DR.transferIPR: not author");
repo.heads[repo.bodies[typeOfDoc][version][0].addr].author = uint40(transferee);
}
function _increaseCounterOfTypes(Repo storage repo)
private returns(uint32)
{
repo.bodies[0][0][0].seq++;
return uint32(repo.bodies[0][0][0].seq);
}
function _increaseCounterOfVersions(
Repo storage repo,
uint256 typeOfDoc
) private returns(uint32) {
repo.bodies[typeOfDoc][0][0].seq++;
return uint32(repo.bodies[typeOfDoc][0][0].seq);
}
function _increaseCounterOfDocs(
Repo storage repo,
uint256 typeOfDoc,
uint256 version
) private returns(uint64) {
repo.bodies[typeOfDoc][version][0].seq++;
return repo.bodies[typeOfDoc][version][0].seq;
}
// ==== CloneFactory ====
/*
The MIT License (MIT)
Copyright (c) 2018 Murray Software, LLC.
Permission is hereby granted, free of charge, to any person obtaining
a copy of this software and associated documentation files (the
"Software"), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject to
the following conditions:
The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
//solhint-disable max-line-length
//solhint-disable no-inline-assembly
function _createClone(address temp) private returns (address result) {
bytes20 tempBytes = bytes20(temp);
assembly {
let clone := mload(0x40)
mstore(
clone,
0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000
)
mstore(add(clone, 0x14), tempBytes)
mstore(
add(clone, 0x28),
0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000
)
result := create(0, clone, 0x37)
}
}
function _isClone(address temp, address query)
private view returns (bool result)
{
bytes20 tempBytes = bytes20(temp);
assembly {
let clone := mload(0x40)
mstore(
clone,
0x363d3d373d3d3d363d7300000000000000000000000000000000000000000000
)
mstore(add(clone, 0xa), tempBytes)
mstore(
add(clone, 0x1e),
0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000
)
let other := add(clone, 0x40)
extcodecopy(query, other, 0, 0x2d)
result := and(
eq(mload(clone), mload(other)),
eq(mload(add(clone, 0xd)), mload(add(other, 0xd)))
)
}
}
//##################
//## read I/O ##
//##################
function counterOfTypes(Repo storage repo) public view returns(uint32) {
return uint32(repo.bodies[0][0][0].seq);
}
function counterOfVersions(Repo storage repo, uint typeOfDoc) public view returns(uint32) {
return uint32(repo.bodies[uint32(typeOfDoc)][0][0].seq);
}
function counterOfDocs(Repo storage repo, uint typeOfDoc, uint version) public view returns(uint64) {
return repo.bodies[uint32(typeOfDoc)][uint32(version)][0].seq;
}
function getAuthor(
Repo storage repo,
uint typeOfDoc,
uint version
) public view returns(uint40) {
address temp = repo.bodies[typeOfDoc][version][0].addr;
require(temp != address(0), "getAuthor: temp not exist");
return repo.heads[temp].author;
}
function getAuthorByBody(
Repo storage repo,
address body
) public view returns(uint40) {
Head memory head = getHeadByBody(repo, body);
return getAuthor(repo, head.typeOfDoc, head.version);
}
function docExist(Repo storage repo, address body) public view returns(bool) {
Head memory head = repo.heads[body];
if ( body == address(0)
|| head.typeOfDoc == 0
|| head.version == 0
|| head.seqOfDoc == 0
) return false;
return repo.bodies[head.typeOfDoc][head.version][head.seqOfDoc].addr == body;
}
function getHeadByBody(
Repo storage repo,
address body
) public view returns (Head memory ) {
return repo.heads[body];
}
function getDoc(
Repo storage repo,
bytes32 snOfDoc
) public view returns(Doc memory doc) {
doc.head = snParser(snOfDoc);
doc.body = repo.bodies[doc.head.typeOfDoc][doc.head.version][doc.head.seqOfDoc].addr;
doc.head = repo.heads[doc.body];
}
function getVersionsList(
Repo storage repo,
uint typeOfDoc
) public view returns(Doc[] memory)
{
uint32 len = counterOfVersions(repo, typeOfDoc);
Doc[] memory out = new Doc[](len);
while (len > 0) {
Head memory head;
head.typeOfDoc = uint32(typeOfDoc);
head.version = len;
out[len - 1] = getDoc(repo, codifyHead(head));
len--;
}
return out;
}
function getDocsList(
Repo storage repo,
bytes32 snOfDoc
) public view returns(Doc[] memory) {
Head memory head = snParser(snOfDoc);
uint64 len = counterOfDocs(repo, head.typeOfDoc, head.version);
Doc[] memory out = new Doc[](len);
while (len > 0) {
head.seqOfDoc = len;
out[len - 1] = getDoc(repo, codifyHead(head));
len--;
}
return out;
}
function verifyDoc(
Repo storage repo,
bytes32 snOfDoc
) public view returns(bool) {
Head memory head = snParser(snOfDoc);
address temp = repo.bodies[head.typeOfDoc][head.version][0].addr;
address target = repo.bodies[head.typeOfDoc][head.version][head.seqOfDoc].addr;
return _isClone(temp, target);
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
library DTClaims {
using EnumerableSet for EnumerableSet.UintSet;
struct Head {
uint16 seqOfDeal;
bool dragAlong;
uint32 seqOfShare;
uint64 paid;
uint64 par;
uint40 caller;
uint16 para;
uint16 argu;
}
struct Claim {
uint8 typeOfClaim;
uint32 seqOfShare;
uint64 paid;
uint64 par;
uint40 claimer;
uint48 sigDate;
bytes32 sigHash;
}
struct Pack {
//seqOfShare => Claim
mapping(uint256 => Claim) claims;
EnumerableSet.UintSet shares;
}
struct Claims {
// seqOfDeal => drag/tag/merged => Pack
mapping(uint256 => mapping(uint256 => Pack)) packs;
EnumerableSet.UintSet deals;
}
modifier dealExist(Claims storage cls, uint seqOfDeal) {
require (hasClaim(cls, seqOfDeal), "DTClaims.mf.dealExist: not");
_;
}
//#################
//## Write I/O ##
//#################
function snParser(bytes32 sn) public pure returns(Head memory head) {
uint _sn = uint(sn);
head = Head({
seqOfDeal: uint16(_sn >> 240),
dragAlong: bool(uint8(_sn >> 232) == 1),
seqOfShare: uint32(_sn >> 200),
paid: uint64(_sn >> 136),
par: uint64(_sn >> 72),
caller: uint40(_sn >> 32),
para: uint16(_sn >> 16),
argu: uint16(_sn)
});
}
function codifyHead(Head memory head) public pure returns(bytes32 sn) {
bytes memory _sn = abi.encodePacked(
head.seqOfDeal,
head.dragAlong,
head.seqOfShare,
head.paid,
head.par,
head.caller,
head.para,
head.argu
);
assembly {
sn := mload(add(_sn, 0x20))
}
}
function execAlongRight(
Claims storage cls,
bool dragAlong,
uint256 seqOfDeal,
uint256 seqOfShare,
uint paid,
uint par,
uint256 claimer,
bytes32 sigHash
) public {
uint16 intSeqOfDeal = uint16(seqOfDeal);
require(intSeqOfDeal > 0, "DTClaims.exec: zero seqOfDeal");
Claim memory newClaim = Claim({
typeOfClaim: dragAlong ? 0 : 1,
seqOfShare: uint32(seqOfShare),
paid: uint64(paid),
par: uint64(par),
claimer: uint40(claimer),
sigDate: uint48(block.timestamp),
sigHash: sigHash
});
require(newClaim.seqOfShare > 0, "DTClaims.exec: zero seqOfShare");
Pack storage p = cls.packs[intSeqOfDeal][newClaim.typeOfClaim];
if (p.shares.add(newClaim.seqOfShare)){
p.claims[newClaim.seqOfShare] = newClaim;
cls.deals.add(intSeqOfDeal);
_consolidateClaimsOfShare(cls, intSeqOfDeal, newClaim);
}
}
function _consolidateClaimsOfShare(
Claims storage cls,
uint intSeqOfDeal,
Claim memory newClaim
) private {
Pack storage m = cls.packs[intSeqOfDeal][2];
if (m.shares.add(newClaim.seqOfShare)) {
m.claims[newClaim.seqOfShare] = newClaim;
} else {
Claim storage mClaim = m.claims[newClaim.seqOfShare];
mClaim.paid = newClaim.paid > mClaim.paid ? newClaim.paid : mClaim.paid;
mClaim.par = newClaim.par > mClaim.par ? newClaim.par : mClaim.par;
if (mClaim.typeOfClaim == 0){
Claim memory tClaim = cls.packs[intSeqOfDeal][1].claims[newClaim.seqOfShare];
mClaim.typeOfClaim = 1;
mClaim.claimer = tClaim.claimer;
mClaim.sigDate = tClaim.sigDate;
mClaim.sigHash = tClaim.sigHash;
}
}
}
function acceptAlongClaims(
Claims storage cls,
uint seqOfDeal
) public returns (Claim[] memory) {
cls.packs[seqOfDeal][2].claims[0].typeOfClaim = 1;
return getClaimsOfDeal(cls, seqOfDeal);
}
// ################################
// ## Read I/O ##
// ################################
function hasClaim(Claims storage cls, uint seqOfDeal) public view returns(bool) {
return cls.deals.contains(seqOfDeal);
}
function getDeals(Claims storage cls) public view returns(uint[] memory) {
return cls.deals.values();
}
function getClaimsOfDeal(
Claims storage cls,
uint seqOfDeal
) public view dealExist(cls, seqOfDeal) returns(Claim[] memory) {
Pack storage m = cls.packs[seqOfDeal][2];
uint[] memory sharesList = m.shares.values();
uint len = sharesList.length;
Claim[] memory output = new Claim[](len);
while (len > 0) {
output[len - 1] = m.claims[sharesList[len - 1]];
len --;
}
return output;
}
function hasShare(
Claims storage cls,
uint seqOfDeal,
uint seqOfShare
) public view dealExist(cls, seqOfDeal) returns(bool) {
return cls.packs[seqOfDeal][2].shares.contains(seqOfShare);
}
function getClaimForShare(
Claims storage cls,
uint seqOfDeal,
uint seqOfShare
) public view returns (Claim memory) {
require (hasShare(cls, seqOfDeal, seqOfShare), "DTClaims.getClaimsForShare: not exist");
return cls.packs[seqOfDeal][2].claims[seqOfShare];
}
function allAccepted(Claims storage cls) public view returns(bool flag) {
uint[] memory dealsList = cls.deals.values();
uint len = dealsList.length;
flag = true;
while(len > 0) {
if (cls.packs[dealsList[len - 1]][2].claims[0].typeOfClaim == 0) {
flag = false;
break;
}
len--;
}
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/structs/EnumerableSet.sol)
pragma solidity ^0.8.8;
library EnumerableSet {
struct Set {
bytes32[] _values;
mapping(bytes32 => uint256) _indexes;
}
function _add(Set storage set, bytes32 value) private returns (bool) {
if (!_contains(set, value)) {
set._values.push(value);
set._indexes[value] = set._values.length;
return true;
} else {
return false;
}
}
function _remove(Set storage set, bytes32 value) private returns (bool) {
uint256 valueIndex = set._indexes[value];
if (valueIndex != 0) {
uint256 toDeleteIndex = valueIndex - 1;
uint256 lastIndex = set._values.length - 1;
if (lastIndex != toDeleteIndex) {
bytes32 lastValue = set._values[lastIndex];
set._values[toDeleteIndex] = lastValue;
set._indexes[lastValue] = valueIndex;
}
delete set._values[lastIndex];
set._values.pop();
delete set._indexes[value];
return true;
} else {
return false;
}
}
function _contains(Set storage set, bytes32 value)
private
view
returns (bool)
{
return set._indexes[value] != 0;
}
function _length(Set storage set) private view returns (uint256) {
return set._values.length;
}
function _at(Set storage set, uint256 index)
private
view
returns (bytes32)
{
return set._values[index];
}
function _values(Set storage set) private view returns (bytes32[] memory) {
return set._values;
}
//======== Bytes32Set ========
struct Bytes32Set {
Set _inner;
}
function add(Bytes32Set storage set, bytes32 value)
public
returns (bool)
{
return _add(set._inner, value);
}
function remove(Bytes32Set storage set, bytes32 value)
public
returns (bool)
{
return _remove(set._inner, value);
}
function contains(Bytes32Set storage set, bytes32 value)
public
view
returns (bool)
{
return _contains(set._inner, value);
}
function length(Bytes32Set storage set) public view returns (uint256) {
return _length(set._inner);
}
function at(Bytes32Set storage set, uint256 index)
public
view
returns (bytes32)
{
return _at(set._inner, index);
}
function values(Bytes32Set storage set)
public
view
returns (bytes32[] memory)
{
return _values(set._inner);
}
//======== AddressSet ========
struct AddressSet {
Set _inner;
}
function add(AddressSet storage set, address value)
public
returns (bool)
{
return _add(set._inner, bytes32(uint256(uint160(value))));
}
function remove(AddressSet storage set, address value)
public
returns (bool)
{
return _remove(set._inner, bytes32(uint256(uint160(value))));
}
function contains(AddressSet storage set, address value)
public
view
returns (bool)
{
return _contains(set._inner, bytes32(uint256(uint160(value))));
}
function length(AddressSet storage set) public view returns (uint256) {
return _length(set._inner);
}
function at(AddressSet storage set, uint256 index)
public
view
returns (address)
{
return address(uint160(uint256(_at(set._inner, index))));
}
function values(AddressSet storage set)
public
view
returns (address[] memory)
{
bytes32[] memory store = _values(set._inner);
address[] memory result;
assembly {
result := store
}
return result;
}
//======== UintSet ========
struct UintSet {
Set _inner;
}
function add(UintSet storage set, uint256 value) public returns (bool) {
return _add(set._inner, bytes32(value));
}
function remove(UintSet storage set, uint256 value)
public
returns (bool)
{
return _remove(set._inner, bytes32(value));
}
/**
* @dev Returns true if the value is in the set. O(1).
*/
function contains(UintSet storage set, uint256 value)
public
view
returns (bool)
{
return _contains(set._inner, bytes32(value));
}
/**
* @dev Returns the number of values on the set. O(1).
*/
function length(UintSet storage set) public view returns (uint256) {
return _length(set._inner);
}
function at(UintSet storage set, uint256 index)
public
view
returns (uint256)
{
return uint256(_at(set._inner, index));
}
function values(UintSet storage set)
public
view
returns (uint256[] memory)
{
bytes32[] memory store = _values(set._inner);
uint256[] memory result;
assembly {
result := store
}
return result;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
import "./RulesParser.sol";
library FilesRepo {
using EnumerableSet for EnumerableSet.AddressSet;
enum StateOfFile {
ZeroPoint, // 0
Created, // 1
Circulated, // 2
Proposed, // 3
Approved, // 4
Rejected, // 5
Closed, // 6
Revoked // 7
}
struct Head {
uint48 circulateDate;
uint8 signingDays;
uint8 closingDays;
uint16 seqOfVR;
uint8 frExecDays;
uint8 dtExecDays;
uint8 dtConfirmDays;
uint48 proposeDate;
uint8 invExitDays;
uint8 votePrepareDays;
uint8 votingDays;
uint8 execDaysForPutOpt;
uint64 seqOfMotion;
uint8 state;
}
struct Ref {
bytes32 docUrl;
bytes32 docHash;
}
struct File {
bytes32 snOfDoc;
Head head;
Ref ref;
}
struct Repo {
mapping(address => File) files;
EnumerableSet.AddressSet filesList;
}
//####################
//## modifier ##
//####################
modifier onlyRegistered(Repo storage repo, address body) {
require(repo.filesList.contains(body),
"FR.md.OR: doc NOT registered");
_;
}
//##################
//## Write I/O ##
//##################
function regFile(Repo storage repo, bytes32 snOfDoc, address body)
public returns (bool flag)
{
if (repo.filesList.add(body)) {
File storage file = repo.files[body];
file.snOfDoc = snOfDoc;
file.head.state = uint8(StateOfFile.Created);
flag = true;
}
}
function circulateFile(
Repo storage repo,
address body,
uint16 signingDays,
uint16 closingDays,
RulesParser.VotingRule memory vr,
bytes32 docUrl,
bytes32 docHash
) public onlyRegistered(repo, body) returns (Head memory head){
require(
repo.files[body].head.state == uint8(StateOfFile.Created),
"FR.CF: Doc not pending"
);
head = Head({
circulateDate: uint48(block.timestamp),
signingDays: uint8(signingDays),
closingDays: uint8(closingDays),
seqOfVR: vr.seqOfRule,
frExecDays: vr.frExecDays,
dtExecDays: vr.dtExecDays,
dtConfirmDays: vr.dtConfirmDays,
proposeDate: 0,
invExitDays: vr.invExitDays,
votePrepareDays: vr.votePrepareDays,
votingDays: vr.votingDays,
execDaysForPutOpt: vr.execDaysForPutOpt,
seqOfMotion: 0,
state: uint8(StateOfFile.Circulated)
});
require(head.signingDays > 0, "FR.CF: zero signingDays");
require(head.closingDays >= signingDays + vr.frExecDays + vr.dtExecDays + vr.dtConfirmDays +
vr.invExitDays + vr.votePrepareDays + vr.votingDays,
"FR.CF: insufficient closingDays");
File storage file = repo.files[body];
file.head = head;
if (docUrl != bytes32(0) || docHash != bytes32(0)){
file.ref = Ref({
docUrl: docUrl,
docHash: docHash
});
}
return file.head;
}
function proposeFile(
Repo storage repo,
address body,
uint64 seqOfMotion
) public onlyRegistered(repo, body) returns(Head memory){
require(repo.files[body].head.state == uint8(StateOfFile.Circulated),
"FR.PF: Doc not circulated");
uint48 timestamp = uint48(block.timestamp);
require(timestamp >= dtExecDeadline(repo, body),
"FR.proposeFile: still in dtExecPeriod");
File storage file = repo.files[body];
require(timestamp < terminateStartpoint(repo, body) || (file.head.frExecDays
+ file.head.dtExecDays + file.head.dtConfirmDays) == 0,
"FR.proposeFile: missed proposeDeadline");
file.head.proposeDate = timestamp;
file.head.seqOfMotion = seqOfMotion;
file.head.state = uint8(StateOfFile.Proposed);
return file.head;
}
function voteCountingForFile(
Repo storage repo,
address body,
bool approved
) public onlyRegistered(repo, body) {
require(repo.files[body].head.state == uint8(StateOfFile.Proposed),
"FR.VCFF: Doc not proposed");
uint48 timestamp = uint48(block.timestamp);
require(timestamp >= votingDeadline(repo, body),
"FR.voteCounting: still in votingPeriod");
File storage file = repo.files[body];
file.head.state = approved ?
uint8(StateOfFile.Approved) : uint8(StateOfFile.Rejected);
}
function execFile(
Repo storage repo,
address body
) public onlyRegistered(repo, body) {
File storage file = repo.files[body];
require(file.head.state == uint8(StateOfFile.Approved),
"FR.EF: Doc not approved");
uint48 timestamp = uint48(block.timestamp);
require(timestamp < closingDeadline(repo, body),
"FR.EF: missed closingDeadline");
file.head.state = uint8(StateOfFile.Closed);
}
function terminateFile(
Repo storage repo,
address body
) public onlyRegistered(repo, body) {
File storage file = repo.files[body];
require(file.head.state != uint8(StateOfFile.Closed),
"FR.terminateFile: Doc is closed");
file.head.state = uint8(StateOfFile.Revoked);
}
function setStateOfFile(Repo storage repo, address body, uint state)
public onlyRegistered(repo, body)
{
repo.files[body].head.state = uint8(state);
}
//##################
//## read I/O ##
//##################
function signingDeadline(Repo storage repo, address body)
public view returns (uint48) {
File storage file = repo.files[body];
return file.head.circulateDate + uint48(file.head.signingDays) * 86400;
}
function closingDeadline(Repo storage repo, address body)
public view returns (uint48) {
File storage file = repo.files[body];
return file.head.circulateDate + uint48(file.head.closingDays) * 86400;
}
function frExecDeadline(Repo storage repo, address body)
public view returns (uint48) {
File storage file = repo.files[body];
return file.head.circulateDate + uint48(file.head.signingDays +
file.head.frExecDays) * 86400;
}
function dtExecDeadline(Repo storage repo, address body)
public view returns (uint48) {
File storage file = repo.files[body];
return file.head.circulateDate + uint48(file.head.signingDays +
file.head.frExecDays + file.head.dtExecDays) * 86400;
}
function terminateStartpoint(Repo storage repo, address body)
public view returns (uint48) {
File storage file = repo.files[body];
return file.head.circulateDate + (uint48(file.head.signingDays +
file.head.frExecDays + file.head.dtExecDays + file.head.dtConfirmDays)) * 86400;
}
function votingDeadline(Repo storage repo, address body)
public view returns (uint48) {
File storage file = repo.files[body];
return file.head.proposeDate + (uint48(file.head.invExitDays +
file.head.votePrepareDays + file.head.votingDays)) * 86400;
}
function isRegistered(Repo storage repo, address body) public view returns (bool) {
return repo.filesList.contains(body);
}
function qtyOfFiles(Repo storage repo) public view returns (uint256) {
return repo.filesList.length();
}
function getFilesList(Repo storage repo) public view returns (address[] memory) {
return repo.filesList.values();
}
function getFile(Repo storage repo, address body) public view returns (File memory) {
return repo.files[body];
}
function getHeadOfFile(Repo storage repo, address body)
public view onlyRegistered(repo, body) returns (Head memory)
{
return repo.files[body].head;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../comps/books/rom/IRegisterOfMembers.sol";
library FRClaims {
struct Claim {
uint16 seqOfDeal;
uint40 claimer;
uint64 weight;
uint64 ratio;
uint48 sigDate;
bytes32 sigHash;
}
struct Package {
uint64 sumOfWeight;
Claim[] claims;
mapping(uint256 => bool) isClaimer;
}
// packages[0] {
// claims: deals;
// isClaimer: isClaimer;
// }
struct Claims {
// seqOfDeal => Package
mapping(uint256 => Package) packages;
}
//##################
//## Write I/O ##
//##################
function claimFirstRefusal(
Claims storage cls,
uint256 seqOfDeal,
uint256 acct,
bytes32 sigHash
) public {
Claim memory cl = Claim({
seqOfDeal: uint16(seqOfDeal),
claimer: uint40(acct),
weight: 0,
ratio: 0,
sigDate: uint48(block.timestamp),
sigHash: sigHash
});
require(cl.seqOfDeal > 0, "FRClaims.exec: zero seqOfDeal");
require(!isClaimerOfDeal(cls, cl.seqOfDeal, cl.claimer),
"FRClaims.exec: double claim");
if (!isDeal(cls, cl.seqOfDeal))
cls.packages[0].claims.push(cl);
Package storage p = cls.packages[cl.seqOfDeal];
p.isClaimer[cl.claimer] = true;
p.claims.push(cl);
cls.packages[0].isClaimer[cl.claimer] = true;
}
function computeFirstRefusal(
Claims storage cls,
uint256 seqOfDeal,
IRegisterOfMembers rom
) public returns (Claim[] memory output) {
require(isDeal(cls, seqOfDeal), "FRClaims.accept: no claims received");
Package storage p = cls.packages[seqOfDeal];
if (p.sumOfWeight == 0) {
uint256 len = p.claims.length;
uint256 i;
while (i < len) {
Claim storage cl = p.claims[i];
uint64 weight = rom.votesInHand(cl.claimer);
cl.weight = weight;
p.sumOfWeight += weight;
i++;
}
i = 0;
while(i < len) {
Claim storage cl = p.claims[i];
cl.ratio = cl.weight * 10000 / p.sumOfWeight;
i++;
}
} else revert("FRClaims: already created");
output = p.claims;
}
// ################################
// ## Read I/O ##
// ################################
function isClaimer(Claims storage cls, uint acct) public view returns(bool) {
return cls.packages[0].isClaimer[acct];
}
function isClaimerOfDeal(
Claims storage cls,
uint seqOfDeal,
uint acct
) public view returns(bool) {
return cls.packages[seqOfDeal].isClaimer[acct];
}
function isDeal(
Claims storage cls,
uint seqOfDeal
) public view returns(bool) {
return cls.packages[seqOfDeal].claims.length > 0;
}
function getDeals(Claims storage cls) public view returns(uint[] memory) {
Claim[] memory claims = cls.packages[0].claims;
uint len = claims.length;
uint[] memory deals = new uint[](len);
while (len > 0) {
deals[len - 1] = claims[len - 1].seqOfDeal;
len--;
}
return deals;
}
function getClaimsOfDeal(Claims storage cls, uint256 seqOfDeal)
public view returns (Claim[] memory)
{
require(isDeal(cls, seqOfDeal), "FRD.COFR: not a targetDeal");
return cls.packages[seqOfDeal].claims;
}
function allAccepted(Claims storage cls) public view returns (bool) {
uint[] memory deals = getDeals(cls);
uint len = deals.length;
while (len > 0) {
if (cls.packages[deals[len - 1]].sumOfWeight == 0)
return false;
len--;
}
return true;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library GoldChain {
struct Node {
uint32 prev;
uint32 next;
uint32 seqOfShare;
uint64 paid;
uint32 price;
uint48 expireDate;
uint16 votingWeight;
}
struct NodeWrap {
uint32 seq;
Node node;
}
/* nodes[0] {
prev: tail;
next: head;
seqOfShare: counter;
price: length;
} */
struct Chain {
mapping (uint => Node) nodes;
}
//#################
//## Modifier ##
//#################
modifier nodeExist(
Chain storage chain,
uint seq
) {
require(isNode(chain, seq),
"GC.nodeExist: not");
_;
}
//#################
//## Write I/O ##
//#################
function parseSn(
bytes32 sn
) public pure returns(Node memory node) {
uint _sn = uint(sn);
node.prev = uint32(_sn >> 224);
node.next = uint32(_sn >> 192);
node.seqOfShare = uint32(_sn >> 160);
node.paid = uint64(_sn >> 96);
node.price = uint32(_sn >> 64);
node.expireDate = uint48(_sn >> 16);
node.votingWeight = uint16(_sn);
}
function codifyNode(
Node memory node
) public pure returns(bytes32 sn) {
bytes memory _sn =
abi.encodePacked(
node.prev,
node.next,
node.seqOfShare,
node.paid,
node.price,
node.expireDate,
node.votingWeight
);
assembly {
sn := mload(add(_sn, 0x20))
}
}
function createNode(
Chain storage chain,
uint seqOfShare,
uint votingWeight,
uint paid,
uint price,
uint execHours,
bool sortFromHead
) public returns (bytes32 sn) {
require (uint64(paid) > 0, 'GC.createOffer: zero paid');
uint32 seq = _increaseCounter(chain);
Node memory node = Node({
prev: 0,
next: 0,
seqOfShare: uint32(seqOfShare),
paid: uint64(paid),
price: uint32(price),
expireDate: uint48(block.timestamp) + uint48(execHours) * 3600,
votingWeight: uint16(votingWeight)
});
_increaseLength(chain);
chain.nodes[seq] = node;
_upChain(chain, seq, sortFromHead);
sn = codifyNode(node);
}
function _upChain(
Chain storage chain,
uint32 seq,
bool sortFromHead
) private {
Node storage n = chain.nodes[seq];
(uint prev, uint next) =
_getPos(
chain,
n.price,
sortFromHead ? 0 : tail(chain),
sortFromHead ? head(chain) : 0,
sortFromHead
);
n.prev = uint32(prev);
n.next = uint32(next);
chain.nodes[prev].next = seq;
chain.nodes[next].prev = seq;
}
function _getPos(
Chain storage chain,
uint price,
uint prev,
uint next,
bool sortFromHead
) public view returns(uint, uint) {
if (sortFromHead) {
while(next > 0 && chain.nodes[next].price <= price) {
prev = next;
next = chain.nodes[next].next;
}
} else {
while(prev > 0 && chain.nodes[prev].price > price) {
next = prev;
prev = chain.nodes[prev].prev;
}
}
return (prev, next);
}
function offChain(
Chain storage chain,
uint seq
) public nodeExist(chain, seq) returns(Node memory node) {
node = chain.nodes[seq];
chain.nodes[node.prev].next = node.next;
chain.nodes[node.next].prev = node.prev;
delete chain.nodes[seq];
_decreaseLength(chain);
}
function _increaseCounter(
Chain storage chain
) private returns (uint32) {
Node storage n = chain.nodes[0];
do {
unchecked {
n.seqOfShare++;
}
} while(isNode(chain, n.seqOfShare) ||
n.seqOfShare == 0);
return n.seqOfShare;
}
function _increaseLength(
Chain storage chain
) private {
chain.nodes[0].price++;
}
function _decreaseLength(
Chain storage chain
) private {
chain.nodes[0].price--;
}
//#################
//## Read I/O ##
//#################
// ==== Node[0] ====
function counter(
Chain storage chain
) public view returns (uint32) {
return chain.nodes[0].seqOfShare;
}
function length(
Chain storage chain
) public view returns (uint32) {
return chain.nodes[0].price;
}
function head(
Chain storage chain
) public view returns (uint32) {
return chain.nodes[0].next;
}
function tail(
Chain storage chain
) public view returns (uint32) {
return chain.nodes[0].prev;
}
// ==== Node ====
function isNode(
Chain storage chain,
uint seq
) public view returns(bool) {
return chain.nodes[seq].expireDate > 0;
}
function getNode(
Chain storage chain,
uint seq
) public view nodeExist(chain, seq) returns(
Node memory
) {
return chain.nodes[seq];
}
// ==== Chain ====
function getSeqList(
Chain storage chain
) public view returns (uint[] memory) {
uint len = length(chain);
uint[] memory list = new uint[](len);
Node memory node = chain.nodes[0];
while (len > 0) {
list[len-1] = node.prev;
node = chain.nodes[node.prev];
len--;
}
return list;
}
function getChain(
Chain storage chain
) public view returns (NodeWrap[] memory) {
uint len = length(chain);
NodeWrap[] memory list = new NodeWrap[](len);
Node memory node = chain.nodes[0];
while (len > 0) {
list[len-1].seq = node.prev;
node = chain.nodes[node.prev];
list[len-1].node = node;
len--;
}
return list;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../comps/books/rom/IRegisterOfMembers.sol";
import "../comps/books/ros/IRegisterOfShares.sol";
import "./DealsRepo.sol";
import "./RulesParser.sol";
import "./SharesRepo.sol";
library LinksRepo {
using EnumerableSet for EnumerableSet.UintSet;
using RulesParser for bytes32;
enum TriggerTypeOfAlongs {
NoConditions,
ControlChanged,
ControlChangedWithHigherPrice,
ControlChangedWithHigherROE
}
struct Link {
RulesParser.LinkRule linkRule;
EnumerableSet.UintSet followersList;
}
struct Repo {
// dragger => Link
mapping(uint256 => Link) links;
EnumerableSet.UintSet draggersList;
}
modifier draggerExist(Repo storage repo, uint dragger, IRegisterOfMembers _rom) {
require(isDragger(repo, dragger, _rom), "LR.mf.draggerExist: not");
_;
}
// ###############
// ## Write I/O ##
// ###############
function addDragger(Repo storage repo, bytes32 rule, uint256 dragger, IRegisterOfMembers _rom) public {
uint40 groupRep = _rom.groupRep(dragger);
if (repo.draggersList.add(groupRep))
repo.links[groupRep].linkRule = rule.linkRuleParser();
}
function removeDragger(Repo storage repo, uint256 dragger) public {
if (repo.draggersList.remove(dragger))
delete repo.links[dragger];
}
function addFollower(Repo storage repo, uint256 dragger, uint256 follower) public {
repo.links[dragger].followersList.add(uint40(follower));
}
function removeFollower(Repo storage repo, uint256 dragger, uint256 follower) public {
repo.links[dragger].followersList.remove(follower);
}
// ################
// ## Read I/O ##
// ################
function isDragger(Repo storage repo, uint256 dragger, IRegisterOfMembers _rom)
public view returns (bool)
{
uint40 groupRep = _rom.groupRep(dragger);
return repo.draggersList.contains(groupRep);
}
function getLinkRule(Repo storage repo, uint256 dragger, IRegisterOfMembers _rom)
public view draggerExist(repo, dragger, _rom)
returns (RulesParser.LinkRule memory)
{
uint40 groupRep = _rom.groupRep(dragger);
return repo.links[groupRep].linkRule;
}
function isFollower(
Repo storage repo,
uint256 dragger,
uint256 follower,
IRegisterOfMembers _rom
) public view draggerExist(repo, dragger, _rom)
returns (bool)
{
uint40 groupRep = _rom.groupRep(dragger);
return repo.links[groupRep].followersList.contains(uint40(follower));
}
function getDraggers(Repo storage repo) public view returns (uint256[] memory) {
return repo.draggersList.values();
}
function getFollowers(Repo storage repo, uint256 dragger, IRegisterOfMembers _rom)
public view draggerExist(repo, dragger, _rom) returns (uint256[] memory)
{
uint40 groupRep = _rom.groupRep(dragger);
return repo.links[groupRep].followersList.values();
}
function priceCheck(
Repo storage repo,
DealsRepo.Deal memory deal,
IRegisterOfShares _ros,
IRegisterOfMembers _rom
) public view returns (bool) {
RulesParser.LinkRule memory lr =
getLinkRule(repo, deal.head.seller, _rom);
if (lr.triggerType == uint8(TriggerTypeOfAlongs.ControlChangedWithHigherPrice))
return (deal.head.priceOfPaid >= lr.rate);
SharesRepo.Share memory share =
_ros.getShare(deal.head.seqOfShare);
if (lr.triggerType == uint8(TriggerTypeOfAlongs.ControlChangedWithHigherROE))
return (_roeOfDeal(
deal.head.priceOfPaid,
share.head.priceOfPaid,
deal.head.closingDeadline,
share.head.issueDate) >= lr.rate);
return true;
}
function _roeOfDeal(
uint32 dealPrice,
uint32 issuePrice,
uint48 closingDeadline,
uint48 issueDateOfShare
) private pure returns (uint32 roe) {
require(dealPrice > issuePrice, "ROE: NEGATIVE selling price");
require(closingDeadline > issueDateOfShare, "ROE: NEGATIVE holding period");
uint deltaPrice = uint(dealPrice - issuePrice);
uint deltaDate = uint(closingDeadline - issueDateOfShare);
roe = uint32(deltaPrice * 10000 / uint(issuePrice) * 31536000 / deltaDate);
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
library LockersRepo {
using EnumerableSet for EnumerableSet.Bytes32Set;
struct Head {
uint40 from;
uint40 to;
uint48 expireDate;
uint128 value;
}
struct Body {
address counterLocker;
bytes payload;
}
struct Locker {
Head head;
Body body;
}
struct Repo {
// hashLock => locker
mapping (bytes32 => Locker) lockers;
EnumerableSet.Bytes32Set snList;
}
//#################
//## Write ##
//#################
function headSnParser(bytes32 sn) public pure returns (Head memory head) {
uint _sn = uint(sn);
head = Head({
from: uint40(_sn >> 216),
to: uint40(_sn >> 176),
expireDate: uint48(_sn >> 128),
value: uint128(_sn)
});
}
function codifyHead(Head memory head) public pure returns (bytes32 headSn) {
bytes memory _sn = abi.encodePacked(
head.from,
head.to,
head.expireDate,
head.value);
assembly {
headSn := mload(add(_sn, 0x20))
}
}
function lockPoints(
Repo storage repo,
Head memory head,
bytes32 hashLock
) public {
Body memory body;
lockConsideration(repo, head, body, hashLock);
}
function lockConsideration(
Repo storage repo,
Head memory head,
Body memory body,
bytes32 hashLock
) public {
if (repo.snList.add(hashLock)) {
Locker storage locker = repo.lockers[hashLock];
locker.head = head;
locker.body = body;
} else revert ("LR.lockConsideration: occupied");
}
function pickupPoints(
Repo storage repo,
bytes32 hashLock,
string memory hashKey,
uint caller
) public returns(Head memory head) {
bytes memory key = bytes(hashKey);
require(hashLock == keccak256(key),
"LR.pickupPoints: wrong key");
Locker storage locker = repo.lockers[hashLock];
require(block.timestamp < locker.head.expireDate,
"LR.pickupPoints: locker expired");
bool flag = true;
if (locker.body.counterLocker != address(0)) {
require(locker.head.to == caller,
"LR.pickupPoints: wrong caller");
uint len = key.length;
bytes memory zero = new bytes(32 - (len % 32));
bytes memory payload = abi.encodePacked(locker.body.payload, len, key, zero);
(flag, ) = locker.body.counterLocker.call(payload);
}
if (flag) {
head = locker.head;
delete repo.lockers[hashLock];
repo.snList.remove(hashLock);
}
}
function withdrawDeposit(
Repo storage repo,
bytes32 hashLock,
uint256 caller
) public returns(Head memory head) {
Locker memory locker = repo.lockers[hashLock];
require(block.timestamp >= locker.head.expireDate,
"LR.withdrawDeposit: locker not expired");
require(locker.head.from == caller,
"LR.withdrawDeposit: wrong caller");
if (repo.snList.remove(hashLock)) {
head = locker.head;
delete repo.lockers[hashLock];
} revert ("LR.withdrawDeposit: locker not exist");
}
//#################
//## Read ##
//#################
function getHeadOfLocker(
Repo storage repo,
bytes32 hashLock
) public view returns (Head memory head) {
return repo.lockers[hashLock].head;
}
function getLocker(
Repo storage repo,
bytes32 hashLock
) public view returns (Locker memory) {
return repo.lockers[hashLock];
}
function getSnList(
Repo storage repo
) public view returns (bytes32[] memory ) {
return repo.snList.values();
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./Checkpoints.sol";
import "./EnumerableSet.sol";
import "./SharesRepo.sol";
import "./TopChain.sol";
library MembersRepo {
using Checkpoints for Checkpoints.History;
using EnumerableSet for EnumerableSet.UintSet;
using TopChain for TopChain.Chain;
struct Member {
Checkpoints.History votesInHand;
// class => seqList
mapping(uint => EnumerableSet.UintSet) sharesOfClass;
EnumerableSet.UintSet classesBelonged;
}
/*
members[0] {
votesInHand: ownersEquity;
}
*/
/* Node[0] {
prev: tail;
next: head;
ptr: pending;
amt: pending;
sum: totalVotes;
cat: basedOnPar;
} */
struct Repo {
TopChain.Chain chain;
mapping(uint256 => Member) members;
// class => membersList
mapping(uint => EnumerableSet.UintSet) membersOfClass;
}
//###############
//## Modifer ##
//###############
modifier memberExist(
Repo storage repo,
uint acct
) {
require(isMember(repo, acct),
"MR.memberExist: not");
_;
}
//##################
//## Write I/O ##
//##################
// ==== Zero Node Setting ====
function setVoteBase(
Repo storage repo,
bool _basedOnPar
) public returns (bool flag) {
if (repo.chain.basedOnPar() != _basedOnPar) {
uint256[] memory members =
repo.membersOfClass[0].values();
uint256 len = members.length;
while (len > 0) {
uint256 cur = members[len - 1];
Checkpoints.Checkpoint memory cp =
repo.members[cur].votesInHand.latest();
if (cp.paid != cp.par) {
if (_basedOnPar)
repo.chain.increaseAmt(cur, (cp.par - cp.paid) * cp.votingWeight / 100, true);
else repo.chain.increaseAmt(cur, (cp.par - cp.paid) * cp.votingWeight / 100, false);
}
len--;
}
repo.chain.setVoteBase(_basedOnPar);
flag = true;
}
}
// ==== Member ====
function addMember(
Repo storage repo,
uint acct
) public returns (bool flag) {
if (repo.membersOfClass[0].add(acct)) {
repo.chain.addNode(acct);
flag = true;
}
}
function delMember(
Repo storage repo,
uint acct
) public {
repo.chain.delNode(acct);
uint[] memory classes =
repo.members[acct].classesBelonged.values();
uint len = classes.length;
while (len > 0) {
repo.membersOfClass[classes[len - 1]].remove(acct);
len--;
}
repo.membersOfClass[0].remove(acct);
delete repo.members[acct];
}
function addShareToMember(
Repo storage repo,
SharesRepo.Head memory head
) public {
Member storage member = repo.members[head.shareholder];
if (member.sharesOfClass[0].add(head.seqOfShare)
&& member.sharesOfClass[head.class].add(head.seqOfShare)
&& member.classesBelonged.add(head.class))
repo.membersOfClass[head.class].add(head.shareholder);
}
function removeShareFromMember(
Repo storage repo,
SharesRepo.Head memory head
) public {
Member storage member =
repo.members[head.shareholder];
if (member.sharesOfClass[head.class].remove(head.seqOfShare)
&& member.sharesOfClass[0].remove(head.seqOfShare)) {
if(member.sharesOfClass[head.class].length() == 0) {
repo.membersOfClass[head.class].remove(head.shareholder);
member.classesBelonged.remove(head.class);
}
}
}
function increaseAmtOfMember(
Repo storage repo,
uint acct,
uint votingWeight,
uint deltaPaid,
uint deltaPar,
uint deltaClean,
bool isIncrease
) public {
if (deltaPaid > 0 || deltaPar > 0 ) {
uint deltaAmt = repo.chain.basedOnPar()
? deltaPar
: deltaPaid;
repo.chain.increaseAmt(
acct,
deltaAmt * votingWeight / 100,
isIncrease
);
}
Checkpoints.Checkpoint memory cp =
repo.members[acct].votesInHand.latest();
if (cp.votingWeight != votingWeight)
cp.votingWeight = _calWeight(
repo,
cp,
votingWeight,
deltaPaid,
deltaPar,
isIncrease
);
if (isIncrease) {
cp.paid += uint64(deltaPaid);
cp.par += uint64(deltaPar);
cp.cleanPaid += uint64(deltaClean);
} else {
cp.paid -= uint64(deltaPaid);
cp.par -= uint64(deltaPar);
cp.cleanPaid -= uint64(deltaClean);
}
repo.members[acct].votesInHand.push(
cp.votingWeight,
cp.paid,
cp.par,
cp.cleanPaid
);
}
function increaseAmtOfCap(
Repo storage repo,
uint votingWeight,
uint deltaPaid,
uint deltaPar,
bool isIncrease
) public {
Checkpoints.Checkpoint memory cp =
repo.members[0].votesInHand.latest();
if (cp.votingWeight != votingWeight)
cp.votingWeight = _calWeight(
repo,
cp,
votingWeight,
deltaPaid,
deltaPar,
isIncrease
);
if (isIncrease) {
cp.paid += uint64(deltaPaid);
cp.par += uint64(deltaPar);
} else {
cp.paid -= uint64(deltaPaid);
cp.par -= uint64(deltaPar);
}
updateOwnersEquity(repo, cp);
if (repo.chain.basedOnPar() && deltaPar > 0) {
repo.chain.increaseTotalVotes(deltaPar * votingWeight / 100, isIncrease);
} else if (!repo.chain.basedOnPar() && deltaPaid > 0) {
repo.chain.increaseTotalVotes(deltaPaid * votingWeight / 100, isIncrease);
}
}
function _calWeight(
Repo storage repo,
Checkpoints.Checkpoint memory cp,
uint votingWeight,
uint deltaPaid,
uint deltaPar,
bool isIncrease
) private view returns(uint16 output) {
if (isIncrease) {
output = repo.chain.basedOnPar()
? uint16(((cp.votingWeight * cp.par + votingWeight * deltaPar) * 100 / (cp.par + deltaPar) + 50) / 100)
: uint16(((cp.votingWeight * cp.paid + votingWeight * deltaPaid) * 100 / (cp.paid + deltaPaid) + 50) / 100);
} else {
output = repo.chain.basedOnPar()
? uint16(((cp.votingWeight * cp.par - votingWeight * deltaPar) * 100 / (cp.par - deltaPar) + 50) / 100)
: uint16(((cp.votingWeight * cp.paid - votingWeight * deltaPaid) * 100 / (cp.paid - deltaPaid) + 50) / 100);
}
}
// ==== Zero Node Setting ====
function updateOwnersEquity(
Repo storage repo,
Checkpoints.Checkpoint memory cp
) public {
repo.members[0].votesInHand.push(cp.votingWeight, cp.paid, cp.par, cp.cleanPaid);
}
//##################
//## Read ##
//##################
// ==== member ====
function isMember(
Repo storage repo,
uint acct
) public view returns(bool) {
return repo.membersOfClass[0].contains(acct);
}
function qtyOfMembers(
Repo storage repo
) public view returns(uint) {
return repo.membersOfClass[0].length();
}
function membersList(
Repo storage repo
) public view returns(uint[] memory) {
return repo.membersOfClass[0].values();
}
// ---- Votes ----
function ownersEquity(
Repo storage repo
) public view returns(Checkpoints.Checkpoint memory) {
return repo.members[0].votesInHand.latest();
}
function capAtDate(
Repo storage repo,
uint date
) public view returns(Checkpoints.Checkpoint memory) {
return repo.members[0].votesInHand.getAtDate(date);
}
function equityOfMember(
Repo storage repo,
uint acct
) public view memberExist(repo, acct) returns(
Checkpoints.Checkpoint memory
) {
return repo.members[acct].votesInHand.latest();
}
function equityAtDate(
Repo storage repo,
uint acct,
uint date
) public view memberExist(repo, acct) returns(
Checkpoints.Checkpoint memory
) {
return repo.members[acct].votesInHand.getAtDate(date);
}
function votesAtDate(
Repo storage repo,
uint256 acct,
uint date
) public view returns (uint64) {
Checkpoints.Checkpoint memory cp = repo.members[acct].votesInHand.getAtDate(date);
return repo.chain.basedOnPar()
? (cp.par * cp.votingWeight + 50) / 100
: (cp.paid * cp.votingWeight + 50) / 100;
}
function votesHistory(
Repo storage repo,
uint acct
) public view memberExist(repo, acct)
returns (Checkpoints.Checkpoint[] memory)
{
return repo.members[acct].votesInHand.pointsOfHistory();
}
// ---- Class ----
function isClassMember(
Repo storage repo,
uint256 acct,
uint class
) public view memberExist(repo, acct) returns (bool flag) {
return repo.members[acct].classesBelonged.contains(class);
}
function classesBelonged(
Repo storage repo,
uint256 acct
) public view memberExist(repo, acct) returns (uint[] memory) {
return repo.members[acct].classesBelonged.values();
}
function qtyOfClassMember(
Repo storage repo,
uint class
) public view returns(uint256) {
return repo.membersOfClass[class].length();
}
function getMembersOfClass(
Repo storage repo,
uint class
) public view returns(uint256[] memory) {
return repo.membersOfClass[class].values();
}
// ---- Share ----
function qtyOfSharesInHand(
Repo storage repo,
uint acct
) public view memberExist(repo, acct) returns(uint) {
return repo.members[acct].sharesOfClass[0].length();
}
function sharesInHand(
Repo storage repo,
uint acct
) public view memberExist(repo, acct) returns(uint[] memory) {
return repo.members[acct].sharesOfClass[0].values();
}
function qtyOfSharesInClass(
Repo storage repo,
uint acct,
uint class
) public view memberExist(repo, acct) returns(uint) {
require(isClassMember(repo, acct, class),
"MR.qtyOfSharesInClass: not class member");
return repo.members[acct].sharesOfClass[class].length();
}
function sharesInClass(
Repo storage repo,
uint acct,
uint class
) public view memberExist(repo, acct) returns(uint[] memory) {
require(isClassMember(repo, acct, class),
"MR.sharesInClass: not class member");
return repo.members[acct].sharesOfClass[class].values();
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./BallotsBox.sol";
import "./DelegateMap.sol";
import "./EnumerableSet.sol";
import "./RulesParser.sol";
import "../comps/books/roc/IShareholdersAgreement.sol";
library MotionsRepo {
using BallotsBox for BallotsBox.Box;
using DelegateMap for DelegateMap.Map;
using EnumerableSet for EnumerableSet.UintSet;
using RulesParser for bytes32;
enum TypeOfMotion {
ZeroPoint,
ElectOfficer,
RemoveOfficer,
ApproveDoc,
ApproveAction,
TransferFund,
DistributeProfits
}
enum StateOfMotion {
ZeroPoint, // 0
Created, // 1
Proposed, // 2
Passed, // 3
Rejected, // 4
Rejected_NotToBuy, // 5
Rejected_ToBuy, // 6
Executed // 7
}
struct Head {
uint16 typeOfMotion;
uint64 seqOfMotion;
uint16 seqOfVR;
uint40 creator;
uint40 executor;
uint48 createDate;
uint32 data;
}
struct Body {
uint40 proposer;
uint48 proposeDate;
uint48 shareRegDate;
uint48 voteStartDate;
uint48 voteEndDate;
uint16 para;
uint8 state;
}
struct Motion {
Head head;
Body body;
RulesParser.VotingRule votingRule;
uint contents;
}
struct Record {
DelegateMap.Map map;
BallotsBox.Box box;
}
struct VoteCalBase {
uint32 totalHead;
uint64 totalWeight;
uint32 supportHead;
uint64 supportWeight;
uint16 attendHeadRatio;
uint16 attendWeightRatio;
uint16 para;
uint8 state;
bool unaniConsent;
}
struct Repo {
mapping(uint256 => Motion) motions;
mapping(uint256 => Record) records;
EnumerableSet.UintSet seqList;
}
//#################
//## Write I/O ##
//#################
// ==== snParser ====
function snParser (bytes32 sn) public pure returns(Head memory head) {
uint _sn = uint(sn);
head = Head({
typeOfMotion: uint16(_sn >> 240),
seqOfMotion: uint64(_sn >> 176),
seqOfVR: uint16(_sn >> 160),
creator: uint40(_sn >> 120),
executor: uint40(_sn >> 80),
createDate: uint48(_sn >> 32),
data: uint32(_sn)
});
}
function codifyHead(Head memory head) public pure returns(bytes32 sn) {
bytes memory _sn = abi.encodePacked(
head.typeOfMotion,
head.seqOfMotion,
head.seqOfVR,
head.creator,
head.executor,
head.createDate,
head.data);
assembly {
sn := mload(add(_sn, 0x20))
}
}
// ==== addMotion ====
function addMotion(
Repo storage repo,
Head memory head,
uint256 contents
) public returns (Head memory) {
require(head.typeOfMotion > 0, "MR.CM: zero typeOfMotion");
require(head.seqOfVR > 0, "MR.CM: zero seqOfVR");
require(head.creator > 0, "MR.CM: zero caller");
if (!repo.seqList.contains(head.seqOfMotion)) {
head.seqOfMotion = _increaseCounterOfMotion(repo);
head.createDate = uint48(block.timestamp);
repo.seqList.add(head.seqOfMotion);
}
Motion storage m = repo.motions[head.seqOfMotion];
m.head = head;
m.contents = contents;
m.body.state = uint8(StateOfMotion.Created);
return head;
}
function _increaseCounterOfMotion (Repo storage repo) private returns (uint64 seq) {
repo.motions[0].head.seqOfMotion++;
seq = repo.motions[0].head.seqOfMotion;
}
// ==== entrustDelegate ====
function entrustDelegate(
Repo storage repo,
uint256 seqOfMotion,
uint delegate,
uint principal,
IRegisterOfMembers _rom,
IRegisterOfDirectors _rod
) public returns (bool flag) {
Motion storage m = repo.motions[seqOfMotion];
require(m.body.state == uint8(StateOfMotion.Created) ||
m.body.state == uint8(StateOfMotion.Proposed) ,
"MR.EntrustDelegate: wrong state");
if (_rom.isMember(delegate) && _rom.isMember(principal)) {
uint64 weight;
if (m.body.shareRegDate > 0 && block.timestamp >= m.body.shareRegDate)
weight = _rom.votesAtDate(principal, m.body.shareRegDate);
return repo.records[seqOfMotion].map.entrustDelegate(principal, delegate, weight);
} else if (_rod.isDirector(delegate) && _rod.isDirector(principal)) {
return repo.records[seqOfMotion].map.entrustDelegate(principal, delegate, 0);
} else revert ("MR.entrustDelegate: not both Members or Directors");
}
// ==== propose ====
function proposeMotionToGeneralMeeting(
Repo storage repo,
uint256 seqOfMotion,
IShareholdersAgreement _sha,
IRegisterOfMembers _rom,
IRegisterOfDirectors _rod,
uint caller
) public {
RulesParser.GovernanceRule memory gr =
_sha.getRule(0).governanceRuleParser();
require(_memberProposalRightCheck(repo, seqOfMotion, gr, caller, _rom) ||
_directorProposalRightCheck(repo, seqOfMotion, caller, gr.proposeHeadRatioOfDirectorsInGM, _rod),
"MR.PMTGM: has no proposalRight");
_proposeMotion(repo, seqOfMotion, _sha, caller);
}
function _proposeMotion(
Repo storage repo,
uint seqOfMotion,
IShareholdersAgreement _sha,
uint caller
) private {
require(caller > 0, "MR.PM: zero caller");
require(repo.records[seqOfMotion].map.voters[caller].delegate == 0,
"MR.PM: entrused delegate");
Motion storage m = repo.motions[seqOfMotion];
require(m.body.state == uint8(StateOfMotion.Created),
"MR.PM: wrong state");
RulesParser.VotingRule memory vr =
_sha.getRule(m.head.seqOfVR).votingRuleParser();
uint48 timestamp = uint48(block.timestamp);
Body memory body = Body({
proposer: uint40(caller),
proposeDate: timestamp,
shareRegDate: timestamp + uint48(vr.invExitDays) * 86400,
voteStartDate: timestamp + uint48(vr.invExitDays + vr.votePrepareDays) * 86400,
voteEndDate: timestamp + uint48(vr.invExitDays + vr.votePrepareDays + vr.votingDays) * 86400,
para: 0,
state: uint8(StateOfMotion.Proposed)
});
m.body = body;
m.votingRule = vr;
}
function _memberProposalRightCheck(
Repo storage repo,
uint seqOfMotion,
RulesParser.GovernanceRule memory gr,
uint caller,
IRegisterOfMembers _rom
) private returns(bool) {
if (!_rom.isMember(caller)) return false;
Motion memory motion = repo.motions[seqOfMotion];
if (motion.head.typeOfMotion == uint8(TypeOfMotion.ApproveDoc) ||
motion.head.typeOfMotion == uint8(TypeOfMotion.ElectOfficer))
return true;
uint totalVotes = _rom.totalVotes();
if (gr.proposeWeightRatioOfGM > 0 &&
_rom.votesInHand(caller) * 10000 / totalVotes >= gr.proposeWeightRatioOfGM)
return true;
Record storage r = repo.records[seqOfMotion];
r.map.updateLeavesWeightAtDate(caller, uint48(block.timestamp), _rom);
DelegateMap.Voter memory voter = r.map.voters[caller];
if (gr.proposeWeightRatioOfGM > 0 &&
(voter.weight + voter.repWeight) * 10000 / totalVotes >= gr.proposeWeightRatioOfGM)
return true;
if (gr.proposeHeadRatioOfMembers > 0 &&
(voter.repHead + 1) * 10000 / _rom.qtyOfMembers() >=
gr.proposeHeadRatioOfMembers)
return true;
return false;
}
function _directorProposalRightCheck(
Repo storage repo,
uint seqOfMotion,
uint caller,
uint16 proposalThreshold,
IRegisterOfDirectors _rod
) private returns (bool) {
if (!_rod.isDirector(caller)) return false;
uint totalHead = _rod.getNumOfDirectors();
repo.records[seqOfMotion].map.updateLeavesHeadcountOfDirectors(caller, _rod);
if (proposalThreshold > 0 &&
(repo.records[seqOfMotion].map.voters[caller].repHead + 1) * 10000 / totalHead >=
proposalThreshold)
return true;
return false;
}
function proposeMotionToBoard(
Repo storage repo,
uint256 seqOfMotion,
IShareholdersAgreement _sha,
IRegisterOfDirectors _rod,
uint caller
) public {
RulesParser.GovernanceRule memory gr =
_sha.getRule(0).governanceRuleParser();
require(
_directorProposalRightCheck(
repo, seqOfMotion, caller,
gr.proposeHeadRatioOfDirectorsInBoard,
_rod
),
"MR.PMTB: has no proposalRight");
_proposeMotion(repo, seqOfMotion, _sha, caller);
}
// ==== vote ====
function castVoteInGeneralMeeting(
Repo storage repo,
uint256 seqOfMotion,
uint256 acct,
uint attitude,
bytes32 sigHash,
IRegisterOfMembers _rom
) public {
require(_rom.isMember(acct), "MR.castVoteInGM: not Member");
Motion storage m = repo.motions[seqOfMotion];
Record storage r = repo.records[seqOfMotion];
DelegateMap.Voter storage voter = r.map.voters[acct];
r.map.updateLeavesWeightAtDate(acct, m.body.shareRegDate, _rom);
_castVote(repo, seqOfMotion, acct, attitude, voter.repHead + 1, voter.weight + voter.repWeight, sigHash);
}
function castVoteInBoardMeeting(
Repo storage repo,
uint256 seqOfMotion,
uint256 acct,
uint attitude,
bytes32 sigHash,
IRegisterOfDirectors _rod
) public {
require(_rod.isDirector(acct), "MR.CVBM: not Director");
Record storage r = repo.records[seqOfMotion];
DelegateMap.Voter storage voter = r.map.voters[acct];
r.map.updateLeavesHeadcountOfDirectors(acct, _rod);
_castVote(repo, seqOfMotion, acct, attitude, voter.repHead + 1, 0, sigHash);
}
function _castVote(
Repo storage repo,
uint256 seqOfMotion,
uint256 acct,
uint attitude,
uint headcount,
uint weight,
bytes32 sigHash
) private {
require(seqOfMotion > 0, "MR.CV: zero seqOfMotion");
require(voteStarted(repo, seqOfMotion), "MR.CV: vote not started");
require(!voteEnded(repo, seqOfMotion), "MR.CV: vote is Ended");
Record storage r = repo.records[seqOfMotion];
DelegateMap.Voter storage voter = r.map.voters[acct];
require(voter.delegate == 0,
"MR.CV: entrusted delegate");
r.box.castVote(acct, attitude, headcount, weight, sigHash, voter.principals);
}
// ==== counting ====
function voteCounting(
Repo storage repo,
bool flag0,
uint256 seqOfMotion,
VoteCalBase memory base
) public returns (uint8) {
Motion storage m = repo.motions[seqOfMotion];
Record storage r = repo.records[seqOfMotion];
require (m.body.state == uint8(StateOfMotion.Proposed) , "MR.VT: wrong state");
require (voteEnded(repo, seqOfMotion), "MR.VT: vote not ended yet");
bool flag1 = m.votingRule.headRatio == 0;
bool flag2 = m.votingRule.amountRatio == 0;
bool flag = (flag1 && flag2);
if (!flag && flag0 && !_isVetoed(r, m.votingRule.vetoers[0]) &&
!_isVetoed(r, m.votingRule.vetoers[1])) {
flag1 = flag1 ? true : base.totalHead > 0
? ((r.box.cases[uint8(BallotsBox.AttitudeOfVote.Support)]
.sumOfHead + base.supportHead) * 10000) /
base.totalHead >
m.votingRule.headRatio
: base.unaniConsent
? true
: false;
flag2 = flag2 ? true : base.totalWeight > 0
? ((r.box.cases[uint8(BallotsBox.AttitudeOfVote.Support)]
.sumOfWeight + base.supportWeight) * 10000) /
base.totalWeight >
m.votingRule.amountRatio
: base.unaniConsent
? true
: false;
}
m.body.state = flag || (flag0 && flag1 && flag2)
? uint8(MotionsRepo.StateOfMotion.Passed)
: m.votingRule.againstShallBuy
? uint8(MotionsRepo.StateOfMotion.Rejected_ToBuy)
: uint8(MotionsRepo.StateOfMotion.Rejected_NotToBuy);
return m.body.state;
}
function _isVetoed(Record storage r, uint256 vetoer)
private
view
returns (bool)
{
return vetoer > 0 && (r.box.ballots[vetoer].sigDate == 0 ||
r.box.ballots[vetoer].attitude != uint8(BallotsBox.AttitudeOfVote.Support));
}
// ==== ExecResolution ====
function execResolution(
Repo storage repo,
uint256 seqOfMotion,
uint256 contents,
uint executor
) public {
Motion storage m = repo.motions[seqOfMotion];
require (m.contents == contents,
"MR.execResolution: wrong contents");
require (m.body.state == uint8(StateOfMotion.Passed),
"MR.execResolution: wrong state");
require (m.head.executor == uint40(executor), "MR.ER: not executor");
m.body.state = uint8(StateOfMotion.Executed);
}
//#################
//## Read ##
//#################
// ==== VoteState ====
function isProposed(Repo storage repo, uint256 seqOfMotion)
public view returns (bool)
{
return repo.motions[seqOfMotion].body.state == uint8(StateOfMotion.Proposed);
}
function voteStarted(Repo storage repo, uint256 seqOfMotion)
public view returns (bool)
{
return isProposed(repo, seqOfMotion) &&
repo.motions[seqOfMotion].body.voteStartDate <= block.timestamp;
}
function voteEnded(Repo storage repo, uint256 seqOfMotion)
public view returns (bool)
{
return isProposed(repo, seqOfMotion) &&
repo.motions[seqOfMotion].body.voteEndDate <= block.timestamp;
}
// ==== Delegate ====
function getVoterOfDelegateMap(Repo storage repo, uint256 seqOfMotion, uint256 acct)
public view returns (DelegateMap.Voter memory)
{
return repo.records[seqOfMotion].map.voters[acct];
}
function getDelegateOf(Repo storage repo, uint256 seqOfMotion, uint acct)
public view returns (uint)
{
return repo.records[seqOfMotion].map.getDelegateOf(acct);
}
// ==== motion ====
function getMotion(Repo storage repo, uint256 seqOfMotion)
public view returns (Motion memory motion)
{
motion = repo.motions[seqOfMotion];
}
// ==== voting ====
function isVoted(Repo storage repo, uint256 seqOfMotion, uint256 acct)
public view returns (bool)
{
return repo.records[seqOfMotion].box.isVoted(acct);
}
function isVotedFor(
Repo storage repo,
uint256 seqOfMotion,
uint256 acct,
uint256 atti
) public view returns (bool) {
return repo.records[seqOfMotion].box.isVotedFor(acct, atti);
}
function getCaseOfAttitude(Repo storage repo, uint256 seqOfMotion, uint256 atti)
public view returns (BallotsBox.Case memory )
{
return repo.records[seqOfMotion].box.getCaseOfAttitude(atti);
}
function getBallot(Repo storage repo, uint256 seqOfMotion, uint256 acct)
public view returns (BallotsBox.Ballot memory)
{
return repo.records[seqOfMotion].box.getBallot(acct);
}
function isPassed(Repo storage repo, uint256 seqOfMotion) public view returns (bool) {
return repo.motions[seqOfMotion].body.state == uint8(MotionsRepo.StateOfMotion.Passed);
}
// ==== snList ====
function getSeqList(Repo storage repo) public view returns (uint[] memory) {
return repo.seqList.values();
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
import "../comps/books/rom/IRegisterOfMembers.sol";
library OfficersRepo {
using EnumerableSet for EnumerableSet.Bytes32Set;
using EnumerableSet for EnumerableSet.UintSet;
enum TitleOfOfficers {
ZeroPoint,
Shareholder,
Chairman,
ViceChairman,
ManagingDirector,
Director,
CEO,
CFO,
COO,
CTO,
President,
VicePresident,
Supervisor,
SeniorManager,
Manager,
ViceManager
}
struct Position {
uint16 title;
uint16 seqOfPos;
uint40 acct;
uint40 nominator;
uint48 startDate;
uint48 endDate;
uint16 seqOfVR;
uint16 titleOfNominator;
uint16 argu;
}
struct Group {
// seqList
EnumerableSet.UintSet posList;
// acctList
EnumerableSet.UintSet acctList;
}
struct Repo {
//seqOfPos => Position
mapping(uint => Position) positions;
// acct => seqOfPos
mapping(uint => EnumerableSet.UintSet) posInHand;
Group directors;
Group managers;
}
//#################
//## Modifier ##
//#################
modifier isVacant(Repo storage repo, uint256 seqOfPos) {
require(!isOccupied(repo, seqOfPos),
"OR.mf.IV: position occupied");
_;
}
//#################
//## Write ##
//#################
// ==== snParser ====
function snParser(bytes32 sn) public pure returns (Position memory position) {
uint _sn = uint(sn);
position = Position({
title: uint16(_sn >> 240),
seqOfPos: uint16(_sn >> 224),
acct: uint40(_sn >> 184),
nominator: uint40(_sn >> 144),
startDate: uint48(_sn >> 96),
endDate: uint48(_sn >> 48),
seqOfVR: uint16(_sn >> 32),
titleOfNominator: uint16(_sn >> 16),
argu: uint16(_sn)
});
}
function codifyPosition(Position memory position) public pure returns (bytes32 sn) {
bytes memory _sn = abi.encodePacked(
position.title,
position.seqOfPos,
position.acct,
position.nominator,
position.startDate,
position.endDate,
position.seqOfVR,
position.titleOfNominator,
position.argu);
assembly {
sn := mload(add(_sn, 0x20))
}
}
// ======== Setting ========
function createPosition (Repo storage repo, bytes32 snOfPos)
public
{
Position memory pos = snParser(snOfPos);
addPosition(repo, pos);
}
function addPosition(
Repo storage repo,
Position memory pos
) public {
require (pos.title > uint8(TitleOfOfficers.Shareholder), "OR.addPosition: title overflow");
require (pos.seqOfPos > 0, "OR.addPosition: zero seqOfPos");
require (pos.titleOfNominator > 0, "OR.addPosition: zero titleOfNominator");
require (pos.endDate > pos.startDate, "OR.addPosition: endDate <= startDate");
require (pos.endDate > block.timestamp, "OR.addPosition: endDate not future");
Position storage p = repo.positions[pos.seqOfPos];
if (p.seqOfPos == 0) {
if (pos.title <= uint8(TitleOfOfficers.Director))
repo.directors.posList.add(pos.seqOfPos);
else repo.managers.posList.add(pos.seqOfPos);
} else require (p.seqOfPos == pos.seqOfPos,
"OR.addPosition: remove pos first");
repo.positions[pos.seqOfPos] = pos;
}
function removePosition(Repo storage repo, uint256 seqOfPos)
public isVacant(repo, seqOfPos) returns (bool flag)
{
if (repo.directors.posList.remove(seqOfPos) ||
repo.managers.posList.remove(seqOfPos))
{
delete repo.positions[seqOfPos];
flag = true;
}
}
function takePosition (
Repo storage repo,
uint256 seqOfPos,
uint acct
) public returns (bool flag) {
require (seqOfPos > 0, "OR.takePosition: zero seqOfPos");
require (acct > 0, "OR.takePosition: zero acct");
Position storage pos = repo.positions[seqOfPos];
if (repo.directors.posList.contains(seqOfPos))
repo.directors.acctList.add(acct);
else if (repo.managers.posList.contains(seqOfPos))
repo.managers.acctList.add(acct);
else revert("OR.takePosition: pos not exist");
pos.acct = uint40(acct);
pos.startDate = uint48(block.timestamp);
repo.posInHand[acct].add(seqOfPos);
flag = true;
}
function quitPosition(
Repo storage repo,
uint256 seqOfPos,
uint acct
) public returns (bool flag)
{
Position memory pos = repo.positions[seqOfPos];
require(acct == pos.acct,
"OR.quitPosition: not the officer");
flag = vacatePosition(repo, seqOfPos);
}
function vacatePosition (
Repo storage repo,
uint seqOfPos
) public returns (bool flag)
{
Position storage pos = repo.positions[seqOfPos];
uint acct = pos.acct;
require (acct > 0, "OR.vacatePosition: empty pos");
if (repo.posInHand[acct].remove(seqOfPos)) {
pos.acct = 0;
if (pos.title <= uint8(TitleOfOfficers.Director))
repo.directors.acctList.remove(acct);
else if (repo.posInHand[acct].length() == 0) {
repo.managers.acctList.remove(acct);
}
flag = true;
}
}
//################
//## Read ##
//################
// ==== Positions ====
function posExist(Repo storage repo, uint256 seqOfPos) public view returns (bool flag) {
flag = repo.positions[seqOfPos].endDate > block.timestamp;
}
function isOccupied(Repo storage repo, uint256 seqOfPos) public view returns (bool flag) {
flag = repo.positions[seqOfPos].acct > 0;
}
function getPosition(Repo storage repo, uint256 seqOfPos) public view returns (Position memory pos) {
pos = repo.positions[seqOfPos];
}
function getFullPosInfo(Repo storage repo, uint[] memory pl)
public view returns(Position[] memory)
{
uint256 len = pl.length;
Position[] memory ls = new Position[](len);
while (len > 0) {
ls[len-1] = repo.positions[pl[len-1]];
len--;
}
return ls;
}
// ==== Managers ====
function isManager(Repo storage repo, uint256 acct) public view returns (bool flag) {
flag = repo.managers.acctList.contains(acct);
}
function getNumOfManagers(Repo storage repo) public view returns (uint256 num) {
num = repo.managers.acctList.length();
}
function getManagersList(Repo storage repo) public view returns (uint256[] memory ls) {
ls = repo.managers.acctList.values();
}
function getManagersPosList(Repo storage repo) public view returns(uint[] memory list) {
list = repo.managers.posList.values();
}
function getManagersFullPosInfo(Repo storage repo) public view
returns(Position[] memory output)
{
uint[] memory pl = repo.managers.posList.values();
output = getFullPosInfo(repo, pl);
}
// ==== Directors ====
function isDirector(Repo storage repo, uint256 acct)
public view returns (bool flag)
{
flag = repo.directors.acctList.contains(acct);
}
function getNumOfDirectors(Repo storage repo) public view
returns (uint256 num)
{
num = repo.directors.acctList.length();
}
function getDirectorsList(Repo storage repo) public view
returns (uint256[] memory ls)
{
ls = repo.directors.acctList.values();
}
function getDirectorsPosList(Repo storage repo) public view
returns (uint256[] memory ls)
{
ls = repo.directors.posList.values();
}
function getDirectorsFullPosInfo(Repo storage repo) public view
returns(Position[] memory output)
{
uint[] memory pl = repo.directors.posList.values();
output = getFullPosInfo(repo, pl);
}
// ==== Executives ====
function hasPosition(Repo storage repo, uint256 acct, uint256 seqOfPos)
public view returns (bool flag)
{
flag = repo.posInHand[acct].contains(seqOfPos);
}
function getPosInHand(Repo storage repo, uint256 acct)
public view returns (uint256[] memory ls)
{
ls = repo.posInHand[acct].values();
}
function getFullPosInfoInHand(Repo storage repo, uint acct)
public view returns (Position[] memory output)
{
uint256[] memory pl = repo.posInHand[acct].values();
output = getFullPosInfo(repo, pl);
}
function hasTitle(Repo storage repo, uint acct, uint title, IRegisterOfMembers _rom)
public view returns (bool)
{
if (title == uint8(TitleOfOfficers.Shareholder))
return _rom.isMember(acct);
if (title == uint8(TitleOfOfficers.Director))
return isDirector(repo, acct);
Position[] memory list = getFullPosInfoInHand(repo, acct);
uint len = list.length;
while (len > 0) {
if (list[len-1].title == uint16(title))
return true;
len --;
}
return false;
}
function hasNominationRight(Repo storage repo, uint seqOfPos, uint acct, IRegisterOfMembers _rom)
public view returns (bool)
{
Position memory pos = repo.positions[seqOfPos];
if (pos.endDate <= block.timestamp) return false;
else if (pos.nominator == 0)
return hasTitle(repo, acct, pos.titleOfNominator, _rom);
else return (pos.nominator == acct);
}
// ==== seatsCalculator ====
function getBoardSeatsQuota(Repo storage repo, uint256 acct) public view
returns (uint256 quota)
{
uint[] memory pl = repo.directors.posList.values();
uint256 len = pl.length;
while (len > 0) {
Position memory pos = repo.positions[pl[len-1]];
if (pos.nominator == acct) quota++;
len--;
}
}
function getBoardSeatsOccupied(Repo storage repo, uint acct) public view
returns (uint256 num)
{
uint256[] memory dl = repo.directors.acctList.values();
uint256 lenOfDL = dl.length;
while (lenOfDL > 0) {
uint256[] memory pl = repo.posInHand[dl[lenOfDL-1]].values();
uint256 lenOfPL = pl.length;
while(lenOfPL > 0) {
Position memory pos = repo.positions[pl[lenOfPL-1]];
if ( pos.title <= uint8(TitleOfOfficers.Director)) {
if (pos.nominator == acct) num++;
break;
}
lenOfPL--;
}
lenOfDL--;
}
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
import "./Checkpoints.sol";
import "./CondsRepo.sol";
import "./SharesRepo.sol";
import "./SwapsRepo.sol";
import "../comps/books/ros/IRegisterOfShares.sol";
library OptionsRepo {
using EnumerableSet for EnumerableSet.UintSet;
using Checkpoints for Checkpoints.History;
using CondsRepo for CondsRepo.Cond;
using CondsRepo for bytes32;
using SwapsRepo for SwapsRepo.Repo;
enum TypeOfOpt {
CallPrice,
PutPrice,
CallRoe,
PutRoe,
CallPriceWithCnds,
PutPriceWithCnds,
CallRoeWithCnds,
PutRoeWithCnds
}
enum StateOfOpt {
Pending,
Issued,
Executed,
Closed
}
struct Head {
uint32 seqOfOpt;
uint8 typeOfOpt;
uint16 classOfShare;
uint32 rate;
uint48 issueDate;
uint48 triggerDate;
uint16 execDays;
uint16 closingDays;
uint40 obligor;
}
struct Body {
uint48 closingDeadline;
uint40 rightholder;
uint64 paid;
uint64 par;
uint8 state;
uint16 para;
uint16 argu;
}
struct Option {
Head head;
CondsRepo.Cond cond;
Body body;
}
struct Record {
EnumerableSet.UintSet obligors;
SwapsRepo.Repo swaps;
Checkpoints.History oracles;
}
struct Repo {
mapping(uint256 => Option) options;
mapping(uint256 => Record) records;
EnumerableSet.UintSet seqList;
}
// ###############
// ## Modifier ##
// ###############
modifier optExist(Repo storage repo, uint seqOfOpt) {
require (isOption(repo, seqOfOpt), "OR.optExist: not");
_;
}
modifier onlyRightholder(Repo storage repo, uint seqOfOpt, uint caller) {
require (isRightholder(repo, seqOfOpt, caller),
"OR.mf.onlyRightholder: not");
_;
}
// ###############
// ## Write I/O ##
// ###############
// ==== cofify / parser ====
function snParser(bytes32 sn) public pure returns (Head memory head) {
uint _sn = uint(sn);
head = Head({
seqOfOpt: uint32(_sn >> 224),
typeOfOpt: uint8(_sn >> 216),
classOfShare: uint16(_sn >> 200),
rate: uint32(_sn >> 168),
issueDate: uint48(_sn >> 120),
triggerDate: uint48(_sn >> 72),
execDays: uint16(_sn >> 56),
closingDays: uint16(_sn >> 40),
obligor: uint40(_sn)
});
}
function codifyHead(Head memory head) public pure returns (bytes32 sn) {
bytes memory _sn = abi.encodePacked(
head.seqOfOpt,
head.typeOfOpt,
head.classOfShare,
head.rate,
head.issueDate,
head.triggerDate,
head.execDays,
head.closingDays,
head.obligor);
assembly {
sn := mload(add(_sn, 0x20))
}
}
// ==== Option ====
function createOption(
Repo storage repo,
bytes32 snOfOpt,
bytes32 snOfCond,
uint rightholder,
uint paid,
uint par
) public returns (Head memory head)
{
Option memory opt;
opt.head = snParser(snOfOpt);
opt.cond = snOfCond.snParser();
opt.body.closingDeadline = opt.head.triggerDate + (uint48(opt.head.execDays) + uint48(opt.head.closingDays)) * 86400;
opt.body.rightholder = uint40(rightholder);
opt.body.paid = uint64(paid);
opt.body.par = uint64(par);
head = regOption(repo, opt);
}
function issueOption(
Repo storage repo,
Option memory opt
) public returns(uint) {
Option storage o = repo.options[opt.head.seqOfOpt];
o.head.issueDate = uint48(block.timestamp);
o.body.state = uint8(StateOfOpt.Issued);
return o.head.issueDate;
}
function regOption(
Repo storage repo,
Option memory opt
) public returns(Head memory) {
require(opt.head.rate > 0, "OR.IO: ZERO rate");
require(opt.head.triggerDate > block.timestamp, "OR.IO: triggerDate not future");
require(opt.head.execDays > 0, "OR.IO: ZERO execDays");
require(opt.head.closingDays > 0, "OR.IO: ZERO closingDays");
require(opt.head.obligor > 0, "OR.IO: ZERO obligor");
require(opt.body.rightholder > 0, "OR.IO: ZERO rightholder");
require(opt.body.paid > 0, "OR.IO: ZERO paid");
require(opt.body.par >= opt.body.paid, "OR.IO: INSUFFICIENT par");
opt.head.seqOfOpt = _increaseCounter(repo);
repo.seqList.add(opt.head.seqOfOpt);
repo.options[opt.head.seqOfOpt] = opt;
repo.records[opt.head.seqOfOpt].obligors.add(opt.head.obligor);
return opt.head;
}
function removeOption(
Repo storage repo,
uint seqOfOpt
) public returns (bool flag) {
require (
repo.options[seqOfOpt].body.state == uint8(StateOfOpt.Pending),
"OR.removeOption: wrong state"
);
if (repo.seqList.remove(seqOfOpt)) {
delete repo.options[seqOfOpt];
flag = true;
}
}
// ==== Record ====
function addObligorIntoOption(Repo storage repo, uint seqOfOpt, uint256 obligor)
public returns(bool)
{
require (obligor > 0, "OR.AOIO: zero obligor");
return repo.records[seqOfOpt].obligors.add(uint40(obligor));
}
function removeObligorFromOption(Repo storage repo, uint seqOfOpt, uint256 obligor)
public returns(bool)
{
require (obligor > 0, "OR.ROFO: zero obligor");
return repo.records[seqOfOpt].obligors.remove(obligor);
}
function addObligorsIntoOption(Repo storage repo, uint seqOfOpt, uint256[] memory obligors)
public
{
Record storage rcd = repo.records[seqOfOpt];
uint256 len = obligors.length;
while (len > 0) {
rcd.obligors.add(uint40(obligors[len-1]));
len--;
}
}
// ==== ExecOption ====
function updateOracle(
Repo storage repo,
uint256 seqOfOpt,
uint d1,
uint d2,
uint d3
) public optExist(repo, seqOfOpt) {
repo.records[seqOfOpt].oracles.push(100, d1, d2, d3);
}
function execOption(
Repo storage repo,
uint256 seqOfOpt,
uint caller
) public onlyRightholder(repo, seqOfOpt, caller) {
Option storage opt = repo.options[seqOfOpt];
Record storage rcd = repo.records[seqOfOpt];
require(
opt.body.state == uint8(StateOfOpt.Issued),
"OR.EO: wrong state of Opt"
);
require(
block.timestamp >= opt.head.triggerDate,
"OR.EO: NOT reached TriggerDate"
);
require(
block.timestamp < opt.head.triggerDate + uint48(opt.head.execDays) * 86400,
"OR.EO: NOT in exercise period"
);
if (opt.head.typeOfOpt > uint8(TypeOfOpt.PutRoe)) {
Checkpoints.Checkpoint memory cp = rcd.oracles.latest();
if (opt.cond.logicOpr == uint8(CondsRepo.LogOps.ZeroPoint)) {
require(opt.cond.checkSoleCond(cp.paid),
"OR.EO: conds not satisfied");
} else if (opt.cond.logicOpr <= uint8(CondsRepo.LogOps.NotEqual)) {
require(opt.cond.checkCondsOfTwo(cp.paid, cp.par),
"OR.EO: conds not satisfied");
} else if (opt.cond.logicOpr <= uint8(CondsRepo.LogOps.NeOr)) {
require(opt.cond.checkCondsOfThree(cp.paid, cp.par, cp.cleanPaid),
"OR.EO: conds not satisfied");
} else revert("OR.EO: logical operator overflow");
}
opt.body.closingDeadline = uint48(block.timestamp) + uint48(opt.head.closingDays) * 86400;
opt.body.state = uint8(StateOfOpt.Executed);
}
// ==== Brief ====
function createSwap(
Repo storage repo,
uint256 seqOfOpt,
uint seqOfTarget,
uint paidOfTarget,
uint seqOfPledge,
uint caller,
IRegisterOfShares _ros
) public onlyRightholder(repo, seqOfOpt, caller) returns(SwapsRepo.Swap memory swap) {
Option storage opt = repo.options[seqOfOpt];
require(opt.body.state == uint8(StateOfOpt.Executed), "OR.createSwap: wrong state");
require(block.timestamp < opt.body.closingDeadline, "OR.createSwap: option expired");
swap.seqOfTarget = uint32(seqOfTarget);
swap.paidOfTarget = uint64(paidOfTarget);
swap.seqOfPledge = uint32(seqOfPledge);
swap.state = uint8(SwapsRepo.StateOfSwap.Issued);
Record storage rcd = repo.records[opt.head.seqOfOpt];
SharesRepo.Head memory headOfTarget = _ros.getShare(swap.seqOfTarget).head;
SharesRepo.Head memory headOfPledge = _ros.getShare(swap.seqOfPledge).head;
require(opt.head.classOfShare == headOfTarget.class,
"OR.createSwap: wrong target class");
require (opt.body.paid >= rcd.swaps.sumPaidOfTarget() + swap.paidOfTarget,
"OR.PS: paidOfTarget overflow");
if (opt.head.typeOfOpt % 2 == 1) { // Put Option
require(opt.body.rightholder == headOfTarget.shareholder,
"OR.createSwap: rightholder not targetholder");
require(rcd.obligors.contains(headOfPledge.shareholder),
"OR.createSwap: pledge shareholder not obligor");
swap.isPutOpt = true;
} else { // Call Opt
require(opt.body.rightholder == headOfPledge.shareholder,
"OR.createSwap: pledge shareholder not rightholder");
require(rcd.obligors.contains(headOfTarget.shareholder),
"OR.createSwap: target shareholder not obligor");
}
if (opt.head.typeOfOpt % 4 < 2)
swap.priceOfDeal = opt.head.rate;
else {
uint32 ds = uint32(((block.timestamp - headOfTarget.issueDate) + 43200) / 86400);
swap.priceOfDeal = headOfTarget.priceOfPaid * (opt.head.rate * ds + 3650000) / 3650000;
}
if (opt.head.typeOfOpt % 2 == 1) {
swap.paidOfPledge = (swap.priceOfDeal - headOfTarget.priceOfPaid) *
swap.paidOfTarget / headOfPledge.priceOfPaid;
}
return rcd.swaps.regSwap(swap);
}
function payOffSwap(
Repo storage repo,
uint seqOfOpt,
uint seqOfSwap,
uint msgValue,
uint centPrice
) public returns (SwapsRepo.Swap memory ) {
Option storage opt = repo.options[seqOfOpt];
require(opt.body.state == uint8(StateOfOpt.Executed),
"OR.payOffSwap: wrong state of Opt");
require(block.timestamp < opt.body.closingDeadline,
"OR.payOffSwap: option expired");
return repo.records[seqOfOpt].swaps.payOffSwap(seqOfSwap, msgValue, centPrice);
}
function terminateSwap(
Repo storage repo,
uint seqOfOpt,
uint seqOfSwap
) public returns (SwapsRepo.Swap memory){
Option storage opt = repo.options[seqOfOpt];
require(opt.body.state == uint8(StateOfOpt.Executed),
"OR.terminateSwap: wrong state of Opt");
require(block.timestamp >= opt.body.closingDeadline,
"OR.terminateSwap: still in closing period");
return repo.records[seqOfOpt].swaps.terminateSwap(seqOfSwap);
}
// ==== Counter ====
function _increaseCounter(Repo storage repo) private returns(uint32 seqOfOpt) {
repo.options[0].head.seqOfOpt++;
seqOfOpt = repo.options[0].head.seqOfOpt;
}
// ################
// ## Read I/O ##
// ################
// ==== Repo ====
function counterOfOptions(Repo storage repo)
public view returns (uint32)
{
return repo.options[0].head.seqOfOpt;
}
function qtyOfOptions(Repo storage repo)
public view returns (uint)
{
return repo.seqList.length();
}
function isOption(Repo storage repo, uint256 seqOfOpt)
public view returns (bool)
{
return repo.seqList.contains(seqOfOpt);
}
function getOption(Repo storage repo, uint256 seqOfOpt) public view
optExist(repo, seqOfOpt) returns (OptionsRepo.Option memory option)
{
option = repo.options[seqOfOpt];
}
function getAllOptions(Repo storage repo)
public view returns (Option[] memory)
{
uint[] memory ls = repo.seqList.values();
uint256 len = ls.length;
Option[] memory output = new Option[](len);
while (len > 0) {
output[len-1] = repo.options[ls[len-1]];
len--;
}
return output;
}
function isRightholder(Repo storage repo, uint256 seqOfOpt, uint256 acct)
public view optExist(repo, seqOfOpt) returns (bool)
{
return repo.options[seqOfOpt].body.rightholder == acct;
}
function isObligor(Repo storage repo, uint256 seqOfOpt, uint256 acct) public
view optExist(repo, seqOfOpt) returns (bool)
{
return repo.records[seqOfOpt].obligors.contains(acct);
}
function getObligorsOfOption(Repo storage repo, uint256 seqOfOpt) public
view optExist(repo, seqOfOpt) returns (uint256[] memory)
{
return repo.records[seqOfOpt].obligors.values();
}
function getSeqList(Repo storage repo) public view returns(uint[] memory) {
return repo.seqList.values();
}
// ==== Order ====
function counterOfSwaps(Repo storage repo, uint256 seqOfOpt)
public view returns (uint16)
{
return repo.records[seqOfOpt].swaps.counterOfSwaps();
}
function sumPaidOfTarget(Repo storage repo, uint256 seqOfOpt)
public view returns (uint64)
{
return repo.records[seqOfOpt].swaps.sumPaidOfTarget();
}
function isSwap(Repo storage repo, uint256 seqOfOpt, uint256 seqOfOrder)
public view returns (bool)
{
return repo.records[seqOfOpt].swaps.isSwap(seqOfOrder);
}
function getSwap(Repo storage repo, uint256 seqOfOpt, uint256 seqOfSwap)
public view returns (SwapsRepo.Swap memory)
{
return repo.records[seqOfOpt].swaps.getSwap(seqOfSwap);
}
function getAllSwapsOfOption(Repo storage repo, uint256 seqOfOpt)
public view returns (SwapsRepo.Swap[] memory )
{
return repo.records[seqOfOpt].swaps.getAllSwaps();
}
function allSwapsClosed(Repo storage repo, uint256 seqOfOpt)
public view returns (bool)
{
return repo.records[seqOfOpt].swaps.allSwapsClosed();
}
// ==== Oracles ====
function getOracleAtDate(
Repo storage repo,
uint256 seqOfOpt,
uint date
) public view optExist(repo, seqOfOpt)
returns (Checkpoints.Checkpoint memory)
{
return repo.records[seqOfOpt].oracles.getAtDate(date);
}
function getLatestOracle(Repo storage repo, uint256 seqOfOpt)
public view optExist(repo, seqOfOpt)
returns(Checkpoints.Checkpoint memory)
{
return repo.records[seqOfOpt].oracles.latest();
}
function getAllOraclesOfOption(Repo storage repo, uint256 seqOfOpt)
public view optExist(repo, seqOfOpt)
returns (Checkpoints.Checkpoint[] memory)
{
return repo.records[seqOfOpt].oracles.pointsOfHistory();
}
function checkValueOfSwap(
Repo storage repo,
uint seqOfOpt,
uint seqOfSwap,
uint centPrice
) public view optExist(repo, seqOfOpt) returns (uint) {
return repo.records[seqOfOpt].swaps.checkValueOfSwap(seqOfSwap, centPrice);
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./GoldChain.sol";
import "./EnumerableSet.sol";
library OrdersRepo {
using EnumerableSet for EnumerableSet.UintSet;
using GoldChain for GoldChain.Chain;
using GoldChain for GoldChain.Node;
enum StateOfInvestor {
Pending,
Approved,
Revoked
}
struct Investor {
uint40 userNo;
uint40 groupRep;
uint48 regDate;
uint40 verifier;
uint48 approveDate;
uint32 data;
uint8 state;
bytes32 idHash;
}
struct Deal {
uint16 classOfShare;
uint32 seqOfShare;
uint40 buyer;
uint40 groupRep;
uint64 paid;
uint32 price;
uint16 votingWeight;
}
struct Repo {
// class => Chain
mapping(uint256 => GoldChain.Chain) ordersOfClass;
EnumerableSet.UintSet classesList;
mapping(uint256 => Investor) investors;
uint[] investorsList;
// ---- tempArry ----
GoldChain.Node[] expired;
Deal[] deals;
}
//################
//## Modifier ##
//################
modifier investorExist(
Repo storage repo,
uint acct
) {
require(isInvestor(repo, acct),
"OR.investorExist: not");
_;
}
modifier classExist(
Repo storage repo,
uint classOfShare
) {
require (isClass(repo, classOfShare),
"OR.classExist: not");
_;
}
//#################
//## Write I/O ##
//#################
// ==== Codify & Parse ====
function parseSn(bytes32 sn) public pure returns(
Deal memory deal
) {
uint _sn = uint(sn);
deal.classOfShare = uint16(_sn >> 240);
deal.seqOfShare = uint32(_sn >> 208);
deal.buyer = uint40(_sn >> 168);
deal.groupRep = uint40(_sn >> 128);
deal.paid = uint64(_sn >> 64);
deal.price = uint32(_sn >> 32);
deal.votingWeight = uint16(_sn >> 16);
}
function codifyDeal(
Deal memory deal
) public pure returns(bytes32 sn) {
bytes memory _sn =
abi.encodePacked(
deal.classOfShare,
deal.seqOfShare,
deal.buyer,
deal.groupRep,
deal.paid,
deal.price,
deal.votingWeight
);
assembly {
sn := mload(add(_sn, 0x20))
}
}
// ==== Investor ====
function regInvestor(
Repo storage repo,
uint userNo,
uint groupRep,
bytes32 idHash
) public {
require(idHash != bytes32(0),
"OR.regInvestor: zero idHash");
uint40 user = uint40(userNo);
require(user > 0,
"OR.regInvestor: zero userNo");
Investor storage investor = repo.investors[user];
investor.userNo = user;
investor.groupRep = uint40(groupRep);
investor.idHash = idHash;
if (!isInvestor(repo, userNo)) {
repo.investorsList.push(user);
investor.regDate = uint48(block.timestamp);
} else {
if (investor.state == uint8(StateOfInvestor.Approved))
_decreaseQtyOfInvestors(repo);
investor.state = uint8(StateOfInvestor.Pending);
}
}
function approveInvestor(
Repo storage repo,
uint acct,
uint verifier
) public investorExist(repo, acct) {
Investor storage investor = repo.investors[acct];
require(investor.state != uint8(StateOfInvestor.Approved),
"OR,apprInv: wrong state");
investor.verifier = uint40(verifier);
investor.approveDate = uint48(block.timestamp);
investor.state = uint8(StateOfInvestor.Approved);
_increaseQtyOfInvestors(repo);
}
function revokeInvestor(
Repo storage repo,
uint acct,
uint verifier
) public {
Investor storage investor = repo.investors[acct];
require(investor.state == uint8(StateOfInvestor.Approved),
"OR,revokeInvestor: wrong state");
investor.verifier = uint40(verifier);
investor.approveDate = uint48(block.timestamp);
investor.state = uint8(StateOfInvestor.Revoked);
_decreaseQtyOfInvestors(repo);
}
// ==== Order ====
function placeSellOrder(
Repo storage repo,
uint classOfShare,
uint seqOfShare,
uint votingWeight,
uint paid,
uint price,
uint execHours,
bool sortFromHead
) public returns (bytes32 sn) {
repo.classesList.add(classOfShare);
GoldChain.Chain storage chain =
repo.ordersOfClass[classOfShare];
sn = chain.createNode(
seqOfShare,
votingWeight,
paid,
price,
execHours,
sortFromHead
);
}
function withdrawSellOrder(
Repo storage repo,
uint classOfShare,
uint seqOfOrder
) public classExist(repo, classOfShare)
returns (GoldChain.Node memory)
{
return repo.ordersOfClass[classOfShare].offChain(seqOfOrder);
}
function placeBuyOrder(
Repo storage repo,
uint acct,
uint classOfShare,
uint paid,
uint price
) public classExist(repo, classOfShare) returns (
Deal[] memory deals,
Deal memory call,
GoldChain.Node[] memory expired
) {
Investor memory investor =
getInvestor(repo, acct);
require (investor.state == uint8(StateOfInvestor.Approved),
"OR.placeBuyOrder: wrong stateOfInvestor");
call.classOfShare = uint16(classOfShare);
call.paid = uint64(paid);
call.price = uint32(price);
call.buyer = investor.userNo;
call.groupRep = investor.groupRep;
_checkOffers(repo, call);
deals = repo.deals;
delete repo.deals;
expired = repo.expired;
delete repo.expired;
}
function _checkOffers(
Repo storage repo,
Deal memory call
) private {
GoldChain.Chain storage chain =
repo.ordersOfClass[call.classOfShare];
uint32 seqOfOffer = chain.head();
while(seqOfOffer > 0 && call.paid > 0) {
GoldChain.Node memory offer = chain.nodes[seqOfOffer];
if (offer.expireDate <= block.timestamp) {
repo.expired.push(
chain.offChain(seqOfOffer)
);
seqOfOffer = offer.next;
continue;
}
if (offer.price <= call.price) {
bool paidAsPut = offer.paid <= call.paid;
Deal memory deal = Deal({
classOfShare: call.classOfShare,
seqOfShare: offer.seqOfShare,
buyer: call.buyer,
groupRep: call.groupRep,
paid: paidAsPut ? offer.paid : call.paid,
price: offer.price,
votingWeight: offer.votingWeight
});
repo.deals.push(deal);
if (paidAsPut) {
chain.offChain(seqOfOffer);
seqOfOffer = offer.next;
} else {
chain.nodes[seqOfOffer].paid -= deal.paid;
}
call.paid -= deal.paid;
} else break;
}
}
function _increaseQtyOfInvestors(
Repo storage repo
) private {
repo.investors[0].verifier++;
}
function _decreaseQtyOfInvestors(
Repo storage repo
) private {
repo.investors[0].verifier--;
}
//################
//## Read I/O ##
//################
// ==== Investor ====
function isInvestor(
Repo storage repo,
uint acct
) public view returns(bool) {
return repo.investors[acct].regDate > 0;
}
function getInvestor(
Repo storage repo,
uint acct
) public view investorExist(repo, acct) returns(Investor memory) {
return repo.investors[acct];
}
function getQtyOfInvestors(
Repo storage repo
) public view returns(uint) {
return repo.investors[0].verifier;
}
function investorList(
Repo storage repo
) public view returns(uint[] memory) {
return repo.investorsList;
}
function investorInfoList(
Repo storage repo
) public view returns(Investor[] memory list) {
uint[] memory seqList = repo.investorsList;
uint len = seqList.length;
list = new Investor[](len);
while (len > 0) {
list[len - 1] = repo.investors[seqList[len - 1]];
len--;
}
return list;
}
// ==== Class ====
function isClass(
Repo storage repo,
uint classOfShare
) public view returns (bool) {
return repo.classesList.contains(classOfShare);
}
function getClassesList(
Repo storage repo
) public view returns (uint[] memory) {
return repo.classesList.values();
}
// ==== TempArrays ====
function getExpired(
Repo storage repo
) public view returns (GoldChain.Node[] memory) {
return repo.expired;
}
function getDeals(
Repo storage repo
) public view returns(Deal[] memory) {
return repo.deals;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
library PledgesRepo {
using EnumerableSet for EnumerableSet.Bytes32Set;
enum StateOfPld {
Pending,
Issued,
Locked,
Released,
Executed,
Revoked
}
struct Head {
uint32 seqOfShare;
uint16 seqOfPld;
uint48 createDate;
uint16 daysToMaturity;
uint16 guaranteeDays;
uint40 creditor;
uint40 debtor;
uint40 pledgor;
uint8 state;
}
struct Body {
uint64 paid;
uint64 par;
uint64 guaranteedAmt;
uint16 preSeq;
uint16 execDays;
uint16 para;
uint16 argu;
}
struct Pledge {
Head head;
Body body;
bytes32 hashLock;
}
struct Repo{
// seqOfShare => seqOfPld => Pledge
mapping(uint256 => mapping(uint256 => Pledge)) pledges;
EnumerableSet.Bytes32Set snList;
}
//##################
//## Write I/O ##
//##################
function snParser(bytes32 sn) public pure returns (Head memory head) {
uint _sn = uint(sn);
head = Head({
seqOfShare: uint32(_sn >> 224),
seqOfPld: uint16(_sn >> 208),
createDate: uint48(_sn >> 160),
daysToMaturity: uint16(_sn >> 144),
guaranteeDays: uint16(_sn >> 128),
creditor: uint40(_sn >> 88),
debtor: uint40(_sn >> 48),
pledgor: uint40(_sn >> 8),
state: uint8(_sn)
});
}
function codifyHead(Head memory head) public pure returns (bytes32 sn) {
bytes memory _sn = abi.encodePacked(
head.seqOfShare,
head.seqOfPld,
head.createDate,
head.daysToMaturity,
head.guaranteeDays,
head.creditor,
head.pledgor,
head.debtor,
head.state);
assembly {
sn := mload(add(_sn, 0x20))
}
}
function createPledge(
Repo storage repo,
bytes32 snOfPld,
uint paid,
uint par,
uint guaranteedAmt,
uint execDays
) public returns (Head memory head)
{
head = snParser(snOfPld);
head = issuePledge(repo, head, paid, par, guaranteedAmt, execDays);
}
function issuePledge(
Repo storage repo,
Head memory head,
uint paid,
uint par,
uint guaranteedAmt,
uint execDays
) public returns(Head memory regHead) {
require (guaranteedAmt > 0, "PR.issuePld: zero guaranteedAmt");
require (par > 0, "PR.issuePld: zero par");
require (par >= paid, "PR.issuePld: paid overflow");
Pledge memory pld;
pld.head = head;
pld.head.createDate = uint48(block.timestamp);
pld.head.state = uint8(StateOfPld.Issued);
pld.body = Body({
paid: uint64(paid),
par: uint64(par),
guaranteedAmt: uint64(guaranteedAmt),
preSeq:0,
execDays: uint16(execDays),
para:0,
argu:0
});
regHead = regPledge(repo, pld);
}
function regPledge(
Repo storage repo,
Pledge memory pld
) public returns(Head memory){
require(pld.head.seqOfShare > 0,"PR.regPledge: zero seqOfShare");
pld.head.seqOfPld = _increaseCounterOfPld(repo, pld.head.seqOfShare);
repo.pledges[pld.head.seqOfShare][pld.head.seqOfPld] = pld;
repo.snList.add(codifyHead(pld.head));
return pld.head;
}
// ==== Update Pledge ====
function splitPledge(
Repo storage repo,
uint256 seqOfShare,
uint256 seqOfPld,
uint buyer,
uint amt,
uint caller
) public returns(Pledge memory newPld) {
Pledge storage pld = repo.pledges[seqOfShare][seqOfPld];
require(caller == pld.head.creditor, "PR.splitPld: not creditor");
require(!isExpired(pld), "PR.splitPld: pledge expired");
require(pld.head.state == uint8(StateOfPld.Issued) ||
pld.head.state == uint8(StateOfPld.Locked), "PR.splitPld: wrong state");
require(amt > 0, "PR.splitPld: zero amt");
newPld = pld;
if (amt < pld.body.guaranteedAmt) {
uint64 ratio = uint64(amt) * 10000 / newPld.body.guaranteedAmt;
newPld.body.paid = pld.body.paid * ratio / 10000;
newPld.body.par = pld.body.par * ratio / 10000;
newPld.body.guaranteedAmt = uint64(amt);
pld.body.paid -= newPld.body.paid;
pld.body.par -= newPld.body.par;
pld.body.guaranteedAmt -= newPld.body.guaranteedAmt;
} else if (amt == pld.body.guaranteedAmt) {
pld.head.state = uint8(StateOfPld.Released);
} else revert("PR.splitPld: amt overflow");
if (buyer > 0) {
newPld.body.preSeq = pld.head.seqOfPld;
newPld.head.creditor = uint40(buyer);
newPld.head = regPledge(repo, newPld);
}
}
function extendPledge(
Pledge storage pld,
uint extDays,
uint caller
) public {
require(caller == pld.head.pledgor, "PR.extendPld: not pledgor");
require(pld.head.state == uint8(StateOfPld.Issued) ||
pld.head.state == uint8(StateOfPld.Locked), "PR.EP: wrong state");
require(!isExpired(pld), "PR.UP: pledge expired");
pld.head.guaranteeDays += uint16(extDays);
}
// ==== Lock & Release ====
function lockPledge(
Pledge storage pld,
bytes32 hashLock,
uint caller
) public {
require(caller == pld.head.creditor, "PR.lockPld: not creditor");
require (!isExpired(pld), "PR.lockPld: pledge expired");
require (hashLock != bytes32(0), "PR.lockPld: zero hashLock");
if (pld.head.state == uint8(StateOfPld.Issued)){
pld.head.state = uint8(StateOfPld.Locked);
pld.hashLock = hashLock;
} else revert ("PR.lockPld: wrong state");
}
function releasePledge(
Pledge storage pld,
string memory hashKey
) public {
require (pld.head.state == uint8(StateOfPld.Locked), "PR.RP: wrong state");
if (pld.hashLock == keccak256(bytes(hashKey))) {
pld.head.state = uint8(StateOfPld.Released);
} else revert("PR.releasePld: wrong Key");
}
function execPledge(Pledge storage pld, uint caller) public {
require(caller == pld.head.creditor, "PR.execPld: not creditor");
require(isTriggerd(pld), "PR.execPld: pledge not triggered");
require(!isExpired(pld), "PR.execPld: pledge expired");
if (pld.head.state == uint8(StateOfPld.Issued) ||
pld.head.state == uint8(StateOfPld.Locked))
{
pld.head.state = uint8(StateOfPld.Executed);
} else revert ("PR.execPld: wrong state");
}
function revokePledge(Pledge storage pld, uint caller) public {
require(caller == pld.head.pledgor, "PR.revokePld: not pledgor");
require(isExpired(pld), "PR.revokePld: pledge not expired");
if (pld.head.state == uint8(StateOfPld.Issued) ||
pld.head.state == uint8(StateOfPld.Locked))
{
pld.head.state = uint8(StateOfPld.Revoked);
} else revert ("PR.revokePld: wrong state");
}
// ==== Counter ====
function _increaseCounterOfPld(Repo storage repo, uint256 seqOfShare)
private returns (uint16 seqOfPld)
{
repo.pledges[seqOfShare][0].head.seqOfPld++;
seqOfPld = repo.pledges[seqOfShare][0].head.seqOfPld;
}
//#################
//## Read ##
//#################
function isTriggerd(Pledge storage pld) public view returns(bool) {
uint64 triggerDate = pld.head.createDate + uint48(pld.head.daysToMaturity) * 86400;
return block.timestamp >= triggerDate;
}
function isExpired(Pledge storage pld) public view returns(bool) {
uint64 expireDate = pld.head.createDate + uint48(pld.head.daysToMaturity + pld.head.guaranteeDays) * 86400;
return block.timestamp >= expireDate;
}
function counterOfPld(Repo storage repo, uint256 seqOfShare)
public view returns (uint16)
{
return repo.pledges[seqOfShare][0].head.seqOfPld;
}
function isPledge(Repo storage repo, uint seqOfShare, uint seqOfPledge)
public view returns (bool)
{
return repo.pledges[seqOfShare][seqOfPledge].head.createDate > 0;
}
function getSNList(Repo storage repo) public view returns (bytes32[] memory list)
{
list = repo.snList.values();
}
function getPledge(Repo storage repo, uint256 seqOfShare, uint seqOfPld)
public view returns (Pledge memory)
{
return repo.pledges[seqOfShare][seqOfPld];
}
function getPledgesOfShare(Repo storage repo, uint256 seqOfShare)
public view returns (Pledge[] memory)
{
uint256 len = counterOfPld(repo, seqOfShare);
Pledge[] memory output = new Pledge[](len);
while (len > 0) {
output[len - 1] = repo.pledges[seqOfShare][len];
len--;
}
return output;
}
function getAllPledges(Repo storage repo)
public view returns (Pledge[] memory)
{
bytes32[] memory snList = getSNList(repo);
uint len = snList.length;
Pledge[] memory ls = new Pledge[](len);
while( len > 0 ) {
Head memory head = snParser(snList[len - 1]);
ls[len - 1] = repo.pledges[head.seqOfShare][head.seqOfPld];
len--;
}
return ls;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library RolesRepo {
struct Role {
address admin;
mapping(address => bool) isMember;
}
struct Repo {
address owner;
uint8 state; // 0-pending; 1-initiated; 2-finalized
mapping(bytes32 => Role) roles;
}
// ##################
// ## Modifier ##
// ##################
modifier isOwner(Repo storage repo, address caller) {
require(
repo.owner == caller,
"RR.isOwner: not owner"
);
_;
}
modifier isRoleAdmin(Repo storage repo, bytes32 role,
address caller)
{
require(
repo.roles[role].admin == caller,
"RR.isRoleAdmin: not admin"
);
_;
}
// #################
// ## Write ##
// #################
function initDoc(Repo storage repo, address owner) public
{
require(repo.state == 0, "already initiated");
repo.state = 1;
repo.owner = owner;
}
function setOwner(
Repo storage repo,
address acct,
address caller
) public isOwner(repo, caller){
repo.owner = acct;
}
// ==== role ====
function setRoleAdmin(
Repo storage repo,
bytes32 role,
address acct,
address caller
) public isOwner(repo, caller) {
repo.roles[role].admin = acct;
repo.roles[role].isMember[acct] = true;
}
function quitRoleAdmin(
Repo storage repo,
bytes32 role,
address caller
) public isRoleAdmin(repo, role, caller) {
delete repo.roles[role].admin;
delete repo.roles[role].isMember[caller];
}
function grantRole(
Repo storage repo,
bytes32 role,
address acct,
address caller
) public isRoleAdmin(repo, role, caller) {
repo.roles[role].isMember[acct] = true;
}
function revokeRole(
Repo storage repo,
bytes32 role,
address acct,
address caller
) public isRoleAdmin(repo, role, caller) {
delete repo.roles[role].isMember[acct];
}
function renounceRole(
Repo storage repo,
bytes32 role,
address caller
) public {
delete repo.roles[role].isMember[caller];
}
function abandonRole(
Repo storage repo,
bytes32 role,
address caller
) public isOwner(repo, caller) {
delete repo.roles[role];
}
// ###############
// ## Read ##
// ###############
function getOwner(
Repo storage repo
) public view returns (address) {
return repo.owner;
}
function getRoleAdmin(Repo storage repo, bytes32 role)
public view returns (address)
{
return repo.roles[role].admin;
}
function hasRole(
Repo storage repo,
bytes32 role,
address acct
) public view returns (bool) {
return repo.roles[role].isMember[acct];
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library RulesParser {
// ======== GovernanceRule ========
// bytes32 public constant SHA_INIT_GR =
// bytes32(uint(0x0000000000000000010000000000000000000000000000000000000000000000));
struct GovernanceRule {
uint32 fundApprovalThreshold;
bool basedOnPar;
uint16 proposeWeightRatioOfGM;
uint16 proposeHeadRatioOfMembers;
uint16 proposeHeadRatioOfDirectorsInGM;
uint16 proposeHeadRatioOfDirectorsInBoard;
uint16 maxQtyOfMembers;
uint16 quorumOfGM;
uint8 maxNumOfDirectors;
uint16 tenureMonOfBoard;
uint16 quorumOfBoardMeeting;
uint48 establishedDate;
uint8 businessTermInYears;
uint8 typeOfComp;
uint16 minVoteRatioOnChain;
}
function governanceRuleParser(bytes32 sn) public pure returns (GovernanceRule memory rule) {
uint _sn = uint(sn);
rule = GovernanceRule({
fundApprovalThreshold: uint32(_sn >> 224),
basedOnPar: uint8(_sn >> 216) == 1,
proposeWeightRatioOfGM: uint16(_sn >> 200),
proposeHeadRatioOfMembers: uint16(_sn >> 184),
proposeHeadRatioOfDirectorsInGM: uint16(_sn >> 168),
proposeHeadRatioOfDirectorsInBoard: uint16(_sn >> 152),
maxQtyOfMembers: uint16(_sn >> 136),
quorumOfGM: uint16(_sn >> 120),
maxNumOfDirectors: uint8(_sn >> 112),
tenureMonOfBoard: uint16(_sn >> 96),
quorumOfBoardMeeting: uint16(_sn >> 80),
establishedDate: uint48(_sn >> 32),
businessTermInYears: uint8(_sn >> 24),
typeOfComp: uint8(_sn >> 16),
minVoteRatioOnChain: uint16(_sn)
});
}
// ---- VotingRule ----
// bytes32 public constant SHA_INIT_VR =
// bytes32(uint(0x00080c080100001a0b0000010000000000000100000000000000000000000000));
struct VotingRule{
uint16 seqOfRule;
uint8 qtyOfSubRule;
uint8 seqOfSubRule;
uint8 authority;
uint16 headRatio;
uint16 amountRatio;
bool onlyAttendance;
bool impliedConsent;
bool partyAsConsent;
bool againstShallBuy;
uint8 frExecDays;
uint8 dtExecDays;
uint8 dtConfirmDays;
uint8 invExitDays;
uint8 votePrepareDays;
uint8 votingDays;
uint8 execDaysForPutOpt;
uint40[2] vetoers;
uint16 para;
}
function votingRuleParser(bytes32 sn) public pure returns (VotingRule memory rule) {
uint _sn = uint(sn);
rule = VotingRule({
seqOfRule: uint16(_sn >> 240),
qtyOfSubRule: uint8(_sn >> 232),
seqOfSubRule: uint8(_sn >> 224),
authority: uint8(_sn >> 216),
headRatio: uint16(_sn >> 200),
amountRatio: uint16(_sn >> 184),
onlyAttendance: uint8(_sn >> 176) == 1,
impliedConsent: uint8(_sn >> 168) == 1,
partyAsConsent: uint8(_sn >> 160) == 1,
againstShallBuy: uint8(_sn >> 152) == 1,
frExecDays: uint8(_sn >> 144),
dtExecDays: uint8(_sn >> 136),
dtConfirmDays: uint8(_sn >> 128),
invExitDays: uint8(_sn >> 120),
votePrepareDays: uint8(_sn >> 112),
votingDays: uint8(_sn >> 104),
execDaysForPutOpt: uint8(_sn >> 96),
vetoers: [uint40(_sn >> 56), uint40(_sn >> 16)],
para: uint16(_sn)
});
}
// ---- BoardSeatsRule ----
/*
1: Chairman;
2: ViceChairman;
3: Director;
...
*/
struct PositionAllocateRule {
uint16 seqOfRule;
uint8 qtyOfSubRule;
uint8 seqOfSubRule;
bool removePos;
uint16 seqOfPos;
uint16 titleOfPos;
uint40 nominator;
uint16 titleOfNominator;
uint16 seqOfVR;
uint48 endDate;
uint16 para;
uint16 argu;
uint32 data;
}
function positionAllocateRuleParser(bytes32 sn) public pure returns(PositionAllocateRule memory rule) {
uint _sn = uint(sn);
rule = PositionAllocateRule({
seqOfRule: uint16(_sn >> 240),
qtyOfSubRule: uint8(_sn >> 232),
seqOfSubRule: uint8(_sn >> 224),
removePos: uint8(_sn >> 216) == 1,
seqOfPos: uint16(_sn >> 200),
titleOfPos: uint16(_sn >> 184),
nominator: uint40(_sn >> 144),
titleOfNominator: uint16(_sn >> 128),
seqOfVR: uint16(_sn >> 112),
endDate: uint48(_sn >> 64),
para: uint16(_sn >> 48),
argu: uint16(_sn >> 32),
data: uint32(_sn)
});
}
// ---- FirstRefusal Rule ----
struct FirstRefusalRule {
uint16 seqOfRule;
uint8 qtyOfSubRule;
uint8 seqOfSubRule;
uint8 typeOfDeal;
bool membersEqual;
bool proRata;
bool basedOnPar;
uint40[4] rightholders;
uint16 para;
uint16 argu;
}
function firstRefusalRuleParser(bytes32 sn) public pure returns(FirstRefusalRule memory rule) {
uint _sn = uint(sn);
rule = FirstRefusalRule({
seqOfRule: uint16(_sn >> 240),
qtyOfSubRule: uint8(_sn >> 232),
seqOfSubRule: uint8(_sn >> 224),
typeOfDeal: uint8(_sn >> 216),
membersEqual: uint8(_sn >> 208) == 1,
proRata: uint8(_sn >> 200) == 1,
basedOnPar: uint8(_sn >> 192) == 1,
rightholders: [uint40(_sn >> 152), uint40(_sn >> 112), uint40(_sn >> 72), uint40(_sn >> 32)],
para: uint16(_sn >> 16),
argu: uint16(_sn)
});
}
// ---- GroupUpdateOrder ----
struct GroupUpdateOrder {
uint16 seqOfRule;
uint8 qtyOfSubRule;
uint8 seqOfSubRule;
bool addMember;
uint40 groupRep;
uint40[4] members;
uint16 para;
}
function groupUpdateOrderParser(bytes32 sn) public pure returns(GroupUpdateOrder memory order) {
uint _sn = uint(sn);
order = GroupUpdateOrder({
seqOfRule: uint16(_sn >> 240),
qtyOfSubRule: uint8(_sn >> 232),
seqOfSubRule: uint8(_sn >> 224),
addMember: uint8(_sn >> 216) == 1,
groupRep: uint40(_sn >> 176),
members: [
uint40(_sn >> 136),
uint40(_sn >> 96),
uint40(_sn >> 56),
uint40(_sn >> 16)
],
para: uint16(_sn)
});
}
// ---- ListingRule ----
struct ListingRule {
uint16 seqOfRule;
uint16 titleOfIssuer;
uint16 classOfShare;
uint64 maxTotalPar;
uint16 titleOfVerifier;
uint16 maxQtyOfInvestors;
uint32 ceilingPrice;
uint32 floorPrice;
uint16 lockupDays;
uint16 offPrice;
uint16 votingWeight;
}
function listingRuleParser(bytes32 sn) public pure returns(ListingRule memory rule) {
uint _sn = uint(sn);
rule = ListingRule({
seqOfRule: uint16(_sn >> 240),
titleOfIssuer: uint16(_sn >> 224),
classOfShare: uint16(_sn >> 208),
maxTotalPar: uint64(_sn >> 144),
titleOfVerifier: uint16(_sn >> 128),
maxQtyOfInvestors: uint16(_sn >> 112),
ceilingPrice: uint32(_sn >> 80),
floorPrice: uint32(_sn >> 48),
lockupDays: uint16(_sn >> 32),
offPrice: uint16(_sn >> 16),
votingWeight: uint16(_sn)
});
}
// ======== LinkRule ========
struct LinkRule {
uint48 triggerDate;
uint16 effectiveDays;
uint8 triggerType;
uint16 shareRatioThreshold;
uint32 rate;
bool proRata;
uint16 seq;
uint16 para;
uint16 argu;
uint16 ref;
uint64 data;
}
function linkRuleParser(bytes32 sn) public pure returns (LinkRule memory rule) {
uint _sn = uint(sn);
rule = LinkRule({
triggerDate: uint48(_sn >> 208),
effectiveDays: uint16(_sn >> 192),
triggerType: uint8(_sn >> 184),
shareRatioThreshold: uint16(_sn >> 168),
rate: uint32(_sn >> 136),
proRata: uint8(_sn >> 128) == 1,
seq: uint16(_sn >> 112),
para: uint16(_sn >> 96),
argu: uint16(_sn >> 80),
ref: uint16(_sn >> 64),
data: uint64(_sn)
});
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
library SharesRepo {
using EnumerableSet for EnumerableSet.UintSet;
struct Head {
uint16 class;
uint32 seqOfShare;
uint32 preSeq;
uint48 issueDate;
uint40 shareholder;
uint32 priceOfPaid;
uint32 priceOfPar;
uint16 votingWeight;
uint8 argu;
}
struct Body {
uint48 payInDeadline;
uint64 paid;
uint64 par;
uint64 cleanPaid;
uint8 state;
uint8 para;
}
struct Share {
Head head;
Body body;
}
struct Class{
Share info;
EnumerableSet.UintSet seqList;
}
struct Repo {
// seqOfClass => Class
mapping(uint256 => Class) classes;
// seqOfShare => Share
mapping(uint => Share) shares;
}
//####################
//## Modifier ##
//####################
modifier shareExist(
Repo storage repo,
uint seqOfShare
) {
require(isShare(repo, seqOfShare),
"SR.shareExist: not");
_;
}
//#################
//## Write ##
//#################
function snParser(bytes32 sn) public pure returns(Head memory head)
{
uint _sn = uint(sn);
head = Head({
class: uint16(_sn >> 240),
seqOfShare: uint32(_sn >> 208),
preSeq: uint32(_sn >> 176),
issueDate: uint48(_sn >> 128),
shareholder: uint40(_sn >> 88),
priceOfPaid: uint32(_sn >> 56),
priceOfPar: uint32(_sn >> 24),
votingWeight: uint16(_sn >> 8),
argu: uint8(_sn)
});
}
function codifyHead(Head memory head) public pure returns (bytes32 sn)
{
bytes memory _sn =
abi.encodePacked(
head.class,
head.seqOfShare,
head.preSeq,
head.issueDate,
head.shareholder,
head.priceOfPaid,
head.priceOfPar,
head.votingWeight,
head.argu
);
assembly {
sn := mload(add(_sn, 0x20))
}
}
// ==== issue/regist share ====
function createShare(
bytes32 sharenumber,
uint payInDeadline,
uint paid,
uint par
) public pure returns (Share memory share) {
share.head = snParser(sharenumber);
share.body = Body({
payInDeadline: uint48(payInDeadline),
paid: uint64(paid),
par: uint64(par),
cleanPaid: uint64(paid),
state: 0,
para: 0
});
}
function addShare(Repo storage repo, Share memory share)
public returns(Share memory newShare)
{
newShare = regShare(repo, share);
Share storage info = repo.classes[newShare.head.class].info;
if (info.head.issueDate == 0)
repo.classes[newShare.head.class].info.head =
newShare.head;
}
function regShare(Repo storage repo, Share memory share)
public returns(Share memory)
{
require(share.head.class > 0, "SR.regShare: zero class");
require(share.body.par > 0, "SR.regShare: zero par");
require(share.body.par >= share.body.paid, "SR.regShare: paid overflow");
require(share.head.issueDate <= block.timestamp, "SR.regShare: future issueDate");
require(share.head.issueDate <= share.body.payInDeadline, "SR.regShare: issueDate later than payInDeadline");
require(share.head.shareholder > 0, "SR.regShare: zero shareholder");
require(share.head.votingWeight > 0, "SR.regShare: zero votingWeight");
if (share.head.class > counterOfClasses(repo))
share.head.class = _increaseCounterOfClasses(repo);
Class storage class = repo.classes[share.head.class];
if (!class.seqList.contains(share.head.seqOfShare)) {
share.head.seqOfShare = _increaseCounterOfShares(repo);
if (share.head.issueDate == 0)
share.head.issueDate = uint48(block.timestamp);
class.seqList.add(share.head.seqOfShare);
repo.classes[0].seqList.add(share.head.seqOfShare);
}
repo.shares[share.head.seqOfShare] = share;
return share;
}
// ==== counters ====
function _increaseCounterOfShares(
Repo storage repo
) private returns(uint32) {
Head storage h = repo.shares[0].head;
do {
unchecked {
h.seqOfShare++;
}
} while (isShare(repo, h.seqOfShare) ||
h.seqOfShare == 0);
return h.seqOfShare;
}
function _increaseCounterOfClasses(Repo storage repo)
private returns(uint16)
{
repo.shares[0].head.class++;
return repo.shares[0].head.class;
}
// ==== amountChange ====
function payInCapital(
Repo storage repo,
uint seqOfShare,
uint amt
) public shareExist(repo, seqOfShare) {
Share storage share = repo.shares[seqOfShare];
uint64 deltaPaid = uint64(amt);
require(deltaPaid > 0, "SR.payInCap: zero amt");
require(block.timestamp <= share.body.payInDeadline,
"SR.payInCap: missed deadline");
require(share.body.paid + deltaPaid <= share.body.par,
"SR.payInCap: amt overflow");
share.body.paid += deltaPaid;
share.body.cleanPaid += deltaPaid;
}
function subAmtFromShare(
Repo storage repo,
uint seqOfShare,
uint paid,
uint par
) public shareExist(repo, seqOfShare) {
Share storage share = repo.shares[seqOfShare];
Class storage class = repo.classes[share.head.class];
uint64 deltaPaid = uint64(paid);
uint64 deltaPar = uint64(par);
require(deltaPar > 0, "SR.subAmt: zero par");
require(share.body.cleanPaid >= deltaPaid, "SR.subAmt: insufficient cleanPaid");
if (deltaPar == share.body.par) {
class.seqList.remove(seqOfShare);
repo.classes[0].seqList.remove(seqOfShare);
delete repo.shares[seqOfShare];
} else {
share.body.paid -= deltaPaid;
share.body.par -= deltaPar;
share.body.cleanPaid -= deltaPaid;
require(share.body.par >= share.body.paid,
"SR.subAmt: result paid overflow");
}
}
function increaseCleanPaid(
Repo storage repo,
bool isIncrease,
uint seqOfShare,
uint paid
) public shareExist(repo, seqOfShare) {
Share storage share = repo.shares[seqOfShare];
uint64 deltaClean = uint64(paid);
require(deltaClean > 0, "SR.incrClean: zero amt");
if (isIncrease && share.body.cleanPaid + deltaClean <= share.body.paid)
share.body.cleanPaid += deltaClean;
else if(!isIncrease && share.body.cleanPaid >= deltaClean)
share.body.cleanPaid -= deltaClean;
else revert("SR.incrClean: clean overflow");
}
// ---- EquityOfClass ----
function increaseEquityOfClass(
Repo storage repo,
bool isIncrease,
uint classOfShare,
uint deltaPaid,
uint deltaPar,
uint deltaCleanPaid
) public {
Body storage equity = repo.classes[classOfShare].info.body;
if (isIncrease) {
equity.paid += uint64(deltaPaid);
equity.par += uint64(deltaPar);
equity.cleanPaid += uint64(deltaCleanPaid);
} else {
equity.paid -= uint64(deltaPaid);
equity.par -= uint64(deltaPar);
equity.cleanPaid -= uint64(deltaCleanPaid);
}
}
function updatePriceOfPaid(
Repo storage repo,
uint seqOfShare,
uint newPrice
) public shareExist(repo, seqOfShare) {
Share storage share = repo.shares[seqOfShare];
share.head.priceOfPaid = uint32(newPrice);
}
function updatePayInDeadline(
Repo storage repo,
uint seqOfShare,
uint deadline
) public shareExist(repo, seqOfShare) {
Share storage share = repo.shares[seqOfShare];
uint48 newLine = uint48(deadline);
require (block.timestamp < newLine,
"SR.updatePayInDeadline: not future");
share.body.payInDeadline = newLine;
}
//####################
//## Read I/O ##
//####################
// ---- Counter ----
function counterOfShares(
Repo storage repo
) public view returns(uint32) {
return repo.shares[0].head.seqOfShare;
}
function counterOfClasses(
Repo storage repo
) public view returns(uint16) {
return repo.shares[0].head.class;
}
// ---- Share ----
function isShare(
Repo storage repo,
uint seqOfShare
) public view returns(bool) {
return repo.shares[seqOfShare].head.issueDate > 0;
}
function getShare(
Repo storage repo,
uint seqOfShare
) public view shareExist(repo, seqOfShare) returns (
Share memory
) {
return repo.shares[seqOfShare];
}
function getQtyOfShares(
Repo storage repo
) public view returns(uint) {
return repo.classes[0].seqList.length();
}
function getSeqListOfShares(
Repo storage repo
) public view returns(uint[] memory) {
return repo.classes[0].seqList.values();
}
function getSharesList(
Repo storage repo
) public view returns(Share[] memory) {
uint[] memory seqList = repo.classes[0].seqList.values();
return _getShares(repo, seqList);
}
// ---- Class ----
function getQtyOfSharesInClass(
Repo storage repo,
uint classOfShare
) public view returns (uint) {
return repo.classes[classOfShare].seqList.length();
}
function getSeqListOfClass(
Repo storage repo,
uint classOfShare
) public view returns (uint[] memory) {
return repo.classes[classOfShare].seqList.values();
}
function getInfoOfClass(
Repo storage repo,
uint classOfShare
) public view returns (Share memory) {
return repo.classes[classOfShare].info;
}
function getSharesOfClass(
Repo storage repo,
uint classOfShare
) public view returns (Share[] memory) {
uint[] memory seqList =
repo.classes[classOfShare].seqList.values();
return _getShares(repo, seqList);
}
function _getShares(
Repo storage repo,
uint[] memory seqList
) private view returns(Share[] memory list) {
uint len = seqList.length;
list = new Share[](len);
while(len > 0) {
list[len - 1] = repo.shares[seqList[len - 1]];
len--;
}
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./EnumerableSet.sol";
library SigsRepo {
using EnumerableSet for EnumerableSet.UintSet;
struct Signature {
uint40 signer;
uint48 sigDate;
uint64 blocknumber;
bool flag;
uint16 para;
uint16 arg;
uint16 seq;
uint16 attr;
uint32 data;
}
struct Blank{
EnumerableSet.UintSet seqOfDeals;
Signature sig;
bytes32 sigHash;
}
// blanks[0].sig {
// sigDate: circulateDate;
// flag: established;
// para: counterOfBlanks;
// arg: counterOfSigs;
// seq: signingDays;
// attr: closingDays;
// }
struct Page {
// party => Blank
mapping(uint256 => Blank) blanks;
EnumerableSet.UintSet buyers;
EnumerableSet.UintSet sellers;
}
//###################
//## 设置接口 ##
//###################
function circulateDoc(
Page storage p
) public {
p.blanks[0].sig.sigDate = uint48(block.timestamp);
}
function setTiming(
Page storage p,
uint signingDays,
uint closingDays
) public {
p.blanks[0].sig.seq = uint16(signingDays);
p.blanks[0].sig.attr = uint16(closingDays);
}
function addBlank(
Page storage p,
bool beBuyer,
uint256 seq,
uint256 acct
) public {
require (seq > 0, "SR.AB: zero seq");
require (acct > 0, "SR.AB: zero acct");
if (beBuyer) {
require(!p.sellers.contains(acct), "SR.AB: seller intends to buy");
p.buyers.add(acct);
} else {
require(!p.buyers.contains(acct), "SR.AB: buyer intends to sell");
p.sellers.add(acct);
}
if (p.blanks[uint40(acct)].seqOfDeals.add(uint16(seq)))
_increaseCounterOfBlanks(p);
}
function removeBlank(
Page storage p,
uint256 seq,
uint256 acct
) public {
if (p.buyers.contains(acct) || p.sellers.contains(acct)) {
if (p.blanks[acct].seqOfDeals.remove(seq))
_decreaseCounterOfBlanks(p);
if (p.blanks[acct].seqOfDeals.length() == 0) {
delete p.blanks[acct];
p.buyers.remove(acct) || p.sellers.remove(acct);
}
}
}
function signDoc(Page storage p, uint256 acct, bytes32 sigHash)
public
{
require(block.timestamp < getSigDeadline(p) ||
getSigningDays(p) == 0,
"SR.SD: missed sigDeadline");
require(!established(p),
"SR.SD: Doc already established");
if ((p.buyers.contains(acct) || p.sellers.contains(acct)) &&
p.blanks[acct].sig.sigDate == 0) {
Signature storage sig = p.blanks[acct].sig;
sig.signer = uint40(acct);
sig.sigDate = uint48(block.timestamp);
sig.blocknumber = uint64(block.number);
p.blanks[acct].sigHash = sigHash;
_increaseCounterOfSigs(p, p.blanks[acct].seqOfDeals.length());
}
}
function regSig(Page storage p, uint256 acct, uint sigDate, bytes32 sigHash)
public returns (bool flag)
{
require(block.timestamp < getSigDeadline(p),
"SR.RS: missed sigDeadline");
require(!established(p),
"SR.regSig: Doc already established");
if (p.buyers.contains(acct) || p.sellers.contains(acct)) {
Signature storage sig = p.blanks[acct].sig;
sig.signer = uint40(acct);
sig.sigDate = uint48(sigDate);
sig.blocknumber = uint64(block.number);
p.blanks[acct].sigHash = sigHash;
_increaseCounterOfSigs(p, 1);
flag = true;
}
}
function _increaseCounterOfBlanks(Page storage p) private {
p.blanks[0].sig.para++;
}
function _decreaseCounterOfBlanks(Page storage p) private {
p.blanks[0].sig.para--;
}
function _increaseCounterOfSigs(Page storage p, uint qtyOfDeals) private {
p.blanks[0].sig.arg += uint16(qtyOfDeals);
}
//####################
//## Read I/O ##
//####################
function circulated(Page storage p) public view returns (bool)
{
return p.blanks[0].sig.sigDate > 0;
}
function established(Page storage p) public view returns (bool)
{
return counterOfBlanks(p) > 0
&& counterOfBlanks(p) == counterOfSigs(p);
}
function counterOfBlanks(Page storage p) public view returns(uint16) {
return p.blanks[0].sig.para;
}
function counterOfSigs(Page storage p) public view returns(uint16) {
return p.blanks[0].sig.arg;
}
function getCirculateDate(Page storage p) public view returns(uint48) {
return p.blanks[0].sig.sigDate;
}
function getSigningDays(Page storage p) public view returns(uint16) {
return p.blanks[0].sig.seq;
}
function getClosingDays(Page storage p) public view returns(uint16) {
return p.blanks[0].sig.attr;
}
function getSigDeadline(Page storage p) public view returns(uint48) {
return p.blanks[0].sig.sigDate + uint48(p.blanks[0].sig.seq) * 86400;
}
function getClosingDeadline(Page storage p) public view returns(uint48) {
return p.blanks[0].sig.sigDate + uint48(p.blanks[0].sig.attr) * 86400;
}
function isSigner(Page storage p, uint256 acct)
public view returns (bool)
{
return p.blanks[acct].sig.signer > 0;
}
function sigOfParty(Page storage p, uint256 acct) public view
returns (
uint256[] memory seqOfDeals,
Signature memory sig,
bytes32 sigHash
)
{
seqOfDeals = p.blanks[acct].seqOfDeals.values();
sig = p.blanks[acct].sig;
sigHash = p.blanks[acct].sigHash;
}
function sigsOfPage(Page storage p) public view
returns (
Signature[] memory sigsOfBuyer,
Signature[]memory sigsOfSeller
)
{
sigsOfBuyer = sigsOfSide(p, p.buyers);
sigsOfSeller = sigsOfSide(p, p.sellers);
}
function sigsOfSide(Page storage p, EnumerableSet.UintSet storage partiesOfSide)
public view
returns (Signature[] memory)
{
uint256[] memory parties = partiesOfSide.values();
uint256 len = parties.length;
Signature[] memory sigs = new Signature[](len);
while (len > 0) {
sigs[len-1] = p.blanks[parties[len-1]].sig;
len--;
}
return sigs;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
library SwapsRepo {
enum StateOfSwap {
Pending,
Issued,
Closed,
Terminated
}
struct Swap {
uint16 seqOfSwap;
uint32 seqOfPledge;
uint64 paidOfPledge;
uint32 seqOfTarget;
uint64 paidOfTarget;
uint32 priceOfDeal;
bool isPutOpt;
uint8 state;
}
struct Repo {
// seqOfSwap => Swap
mapping(uint256 => Swap) swaps;
}
// ###############
// ## Modifier ##
// ###############
modifier swapExist(Repo storage repo, uint seqOfSwap) {
require (isSwap(repo, seqOfSwap), "SR.swapExist: not");
_;
}
// ###############
// ## Write I/O ##
// ###############
// ==== cofify / parser ====
function codifySwap(Swap memory swap) public pure returns (bytes32 sn) {
bytes memory _sn = abi.encodePacked(
swap.seqOfSwap,
swap.seqOfPledge,
swap.paidOfPledge,
swap.seqOfTarget,
swap.paidOfTarget,
swap.priceOfDeal,
swap.isPutOpt,
swap.state);
assembly {
sn := mload(add(_sn, 0x20))
}
}
function regSwap(
Repo storage repo,
Swap memory swap
) public returns(Swap memory) {
require(swap.seqOfTarget * swap.paidOfTarget * swap.seqOfPledge > 0,
"SWR.regSwap: zero para");
swap.seqOfSwap = _increaseCounter(repo);
repo.swaps[swap.seqOfSwap] = swap;
repo.swaps[0].paidOfTarget += swap.paidOfTarget;
return swap;
}
function payOffSwap(
Repo storage repo,
uint seqOfSwap,
uint msgValue,
uint centPrice
) public returns (Swap memory ) {
Swap storage swap = repo.swaps[seqOfSwap];
require(swap.state == uint8(StateOfSwap.Issued),
"SWR.payOffSwap: wrong state");
require (uint(swap.paidOfTarget) * uint(swap.priceOfDeal) * centPrice / 100 <= msgValue, "SWR.payOffSwap: insufficient amt");
swap.state = uint8(StateOfSwap.Closed);
return swap;
}
function terminateSwap(
Repo storage repo,
uint seqOfSwap
) public returns (Swap memory){
Swap storage swap = repo.swaps[seqOfSwap];
require(swap.state == uint8(StateOfSwap.Issued),
"SWR.terminateSwap: wrong state");
swap.state = uint8(StateOfSwap.Terminated);
return swap;
}
// ==== Counter ====
function _increaseCounter(Repo storage repo) private returns(uint16) {
repo.swaps[0].seqOfSwap++;
return repo.swaps[0].seqOfSwap;
}
// ################
// ## Read I/O ##
// ################
function counterOfSwaps(Repo storage repo)
public view returns (uint16)
{
return repo.swaps[0].seqOfSwap;
}
function sumPaidOfTarget(Repo storage repo)
public view returns (uint64)
{
return repo.swaps[0].paidOfTarget;
}
function isSwap(Repo storage repo, uint256 seqOfSwap)
public view returns (bool)
{
return seqOfSwap <= counterOfSwaps(repo);
}
function getSwap(Repo storage repo, uint256 seqOfSwap)
public view swapExist(repo, seqOfSwap) returns (Swap memory)
{
return repo.swaps[seqOfSwap];
}
function checkValueOfSwap(
Repo storage repo,
uint seqOfSwap,
uint centPrice
) public view returns (uint) {
Swap memory swap = getSwap(repo, seqOfSwap);
return uint(swap.paidOfTarget) * uint(swap.priceOfDeal) * centPrice / 100;
}
function getAllSwaps(Repo storage repo)
public view returns (Swap[] memory )
{
uint256 len = counterOfSwaps(repo);
Swap[] memory swaps = new Swap[](len);
while (len > 0) {
swaps[len-1] = repo.swaps[len];
len--;
}
return swaps;
}
function allSwapsClosed(Repo storage repo)
public view returns (bool)
{
uint256 len = counterOfSwaps(repo);
while (len > 0) {
if (repo.swaps[len].state < uint8(StateOfSwap.Closed))
return false;
len--;
}
return true;
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "../comps/books/roa/IInvestmentAgreement.sol";
import "./DealsRepo.sol";
library TopChain {
enum CatOfNode {
IndepMemberInQueue, // 0
GroupRepInQueue, // 1
GroupMember, // 2
IndepMemberOnChain, // 3
GroupRepOnChain // 4
}
struct Node {
uint40 prev;
uint40 next;
uint40 ptr;
uint64 amt;
uint64 sum;
uint8 cat;
}
struct Para {
uint40 tail;
uint40 head;
uint32 maxQtyOfMembers;
uint16 minVoteRatioOnChain;
uint32 qtyOfSticks;
uint32 qtyOfBranches;
uint32 qtyOfMembers;
uint16 para;
uint16 argu;
}
struct Chain {
// usrNo => Node
mapping(uint256 => Node) nodes;
Para para;
}
/* Node[0] {
prev: tail;
next: head;
ptr: pending;
amt: pending;
sum: totalVotes;
cat: basedOnPar;
} */
//#################
//## Modifier ##
//#################
modifier memberExist(Chain storage chain, uint256 acct) {
require(isMember(chain, acct), "TC.memberExist: acct not member");
_;
}
//#################
//## Write I/O ##
//#################
// ==== Options ====
function setMaxQtyOfMembers(Chain storage chain, uint max) public {
chain.para.maxQtyOfMembers = uint32(max);
}
function setMinVoteRatioOnChain(Chain storage chain, uint min) public {
require(min < 5000, "minVoteRatioOnChain: overflow");
chain.para.minVoteRatioOnChain = uint16(min);
}
function setVoteBase(
Chain storage chain,
bool _basedOnPar
) public {
chain.nodes[0].cat = _basedOnPar ? 1 : 0;
}
// ==== Node ====
function addNode(Chain storage chain, uint acct) public {
require(acct > 0, "TC.addNode: zero acct");
Node storage n = chain.nodes[acct];
if (n.ptr == 0) {
require( maxQtyOfMembers(chain) == 0 ||
qtyOfMembers(chain) < maxQtyOfMembers(chain),
"TC.addNode: no vacance"
);
n.ptr = uint40(acct);
_appendToQueue(chain, n, n.ptr);
_increaseQtyOfMembers(chain);
}
}
function delNode(Chain storage chain, uint acct) public {
_carveOut(chain, acct);
delete chain.nodes[acct];
_decreaseQtyOfMembers(chain);
}
// ==== ChangeAmt ====
function increaseTotalVotes(
Chain storage chain,
uint deltaAmt,
bool isIncrease
) public {
uint40 amt = uint40(deltaAmt);
if (isIncrease) _increaseTotalVotes(chain, amt);
else _decreaseTotalVotes(chain, amt);
}
function increaseAmt(
Chain storage chain,
uint256 acct,
uint deltaAmt,
bool isIncrease
) public memberExist(chain, acct) {
uint64 amt = uint64(deltaAmt);
Node storage n = chain.nodes[acct];
if (isIncrease) {
n.amt += amt;
n.sum += amt;
} else {
n.amt -= amt;
n.sum -= amt;
}
if (n.cat == uint8(CatOfNode.GroupMember)) {
Node storage r = chain.nodes[n.ptr];
if (isIncrease) {
r.sum += amt;
if (r.cat == uint8(CatOfNode.GroupRepOnChain))
_move(chain, n.ptr, isIncrease);
else if (_onChainTest(chain, r))
_upChainAndMove(chain, r, n.ptr);
} else {
r.sum -= amt;
if (r.cat == uint8(CatOfNode.GroupRepOnChain)) {
if (!_onChainTest(chain, r))
_offChain(chain, r, n.ptr);
else _move(chain, n.ptr, isIncrease);
}
}
} else if (_isOnChain(n)) {
if (isIncrease) _move(chain, n.ptr, isIncrease);
else {
if (!_onChainTest(chain, n))
_offChain(chain, n, n.ptr);
else _move(chain, n.ptr, isIncrease);
}
} else if(isIncrease && _onChainTest(chain, n))
_upChainAndMove(chain, n, n.ptr);
}
// ==== Grouping ====
function top2Sub(
Chain storage chain,
uint256 acct,
uint256 root
) public memberExist(chain, root) {
Node storage n = chain.nodes[acct];
Node storage r = chain.nodes[root];
require(acct != root, "TC.T2S: self grouping");
require(_isIndepMember(n), "TC.T2S: not indepMember");
require(_notGroupMember(r), "TC.T2S: leaf as root");
_carveOut(chain, n.ptr);
_vInsert(chain, n.ptr, uint40(root));
}
function sub2Top(Chain storage chain, uint256 acct) public {
Node storage n = chain.nodes[acct];
require(_isInGroup(n), "TC.S2T: not in a branch");
_carveOut(chain, acct);
n.sum = n.amt;
n.ptr = uint40(acct);
if (_onChainTest(chain, n)) _upChainAndMove(chain, n, n.ptr);
else _appendToQueue(chain, n, n.ptr);
}
// ==== CarveOut ====
function _branchOff(Chain storage chain, uint256 root) private {
Node storage r = chain.nodes[root];
if (_isOnChain(r)) {
chain.nodes[r.next].prev = r.prev;
chain.nodes[r.prev].next = r.next;
_decreaseQtyOfBranches(chain);
} else {
if (r.prev > 0 && r.next > 0) {
chain.nodes[r.next].prev = r.prev;
chain.nodes[r.prev].next = r.next;
}else if (r.prev == 0 && r.next == 0) {
chain.para.tail = 0;
chain.para.head = 0;
}else if (r.next == 0) {
chain.para.tail = r.prev;
chain.nodes[r.prev].next = 0;
} else if (r.prev == 0) {
chain.nodes[r.next].prev = 0;
chain.para.head = r.next;
}
_decreaseQtyOfSticks(chain);
}
}
function _carveOut(Chain storage chain, uint acct)
private
memberExist(chain, acct)
{
Node storage n = chain.nodes[acct];
if (_isIndepMember(n)) {
_branchOff(chain, acct);
} else if (_isGroupRep(n)) {
if (n.cat == uint8(CatOfNode.GroupRepOnChain) || (n.prev > 0 && n.next > 0)) {
chain.nodes[n.prev].next = n.ptr;
chain.nodes[n.next].prev = n.ptr;
} else {
if (n.prev == 0 && n.next == 0) {
chain.para.tail = n.ptr;
chain.para.head = n.ptr;
} else if (n.next == 0) {
chain.para.tail = n.ptr;
chain.nodes[n.prev].next = n.ptr;
} else if (n.prev == 0) {
chain.nodes[n.next].prev = n.ptr;
chain.para.head = n.ptr;
}
}
Node storage d = chain.nodes[n.ptr];
d.ptr = d.next;
d.prev = n.prev;
d.next = n.next;
if (d.ptr > 0) {
uint40 cur = d.ptr;
while (cur > 0) {
chain.nodes[cur].ptr = n.ptr;
cur = chain.nodes[cur].next;
}
d.cat = n.cat;
} else {
d.ptr = n.ptr;
d.cat = n.cat == uint8(CatOfNode.GroupRepInQueue)
? uint8(CatOfNode.IndepMemberInQueue)
: uint8(CatOfNode.IndepMemberOnChain);
}
d.sum = n.sum - n.amt;
_offChainCheck(chain, d, n.ptr);
} else if (_isGroupMember(n)) {
Node storage u = chain.nodes[n.prev];
if (n.next > 0) chain.nodes[n.next].prev = n.prev;
if (u.cat == uint8(CatOfNode.GroupMember)) u.next = n.next;
else if (n.next > 0) {
u.ptr = n.next;
} else {
u.ptr = n.ptr;
u.cat = u.cat == uint8(CatOfNode.GroupRepInQueue)
? uint8(CatOfNode.IndepMemberInQueue)
: uint8(CatOfNode.IndepMemberOnChain);
}
Node storage r = chain.nodes[n.ptr];
r.sum -= n.amt;
_offChainCheck(chain, r, n.ptr);
}
}
function _offChainCheck(
Chain storage chain,
Node storage r,
uint40 acct
) private {
if (_isOnChain(r)) {
if (_onChainTest(chain, r)) _move(chain, acct, false);
else _offChain(chain, r, acct);
}
}
// ==== Insert ====
function _hInsert(
Chain storage chain,
uint acct,
uint prev,
uint next
) private {
Node storage n = chain.nodes[acct];
chain.nodes[prev].next = uint40(acct);
n.prev = uint40(prev);
chain.nodes[next].prev = uint40(acct);
n.next = uint40(next);
}
function _vInsert(
Chain storage chain,
uint40 acct,
uint40 root
) private {
Node storage n = chain.nodes[acct];
Node storage r = chain.nodes[root];
if (_isIndepMember(r)) {
r.cat = r.cat == uint8(CatOfNode.IndepMemberInQueue)
? uint8(CatOfNode.GroupRepInQueue)
: uint8(CatOfNode.GroupRepOnChain);
n.next = 0;
} else if (_isGroupRep(r)) {
n.next = r.ptr;
chain.nodes[n.next].prev = acct;
}
n.prev = root;
n.ptr = root;
n.cat = uint8(CatOfNode.GroupMember);
r.ptr = acct;
r.sum += n.amt;
if (_isOnChain(r)) _move(chain, root, true);
else if (_onChainTest(chain, r)) {
_upChainAndMove(chain, r, root);
}
}
// ==== Move ====
function _move(
Chain storage chain,
uint acct,
bool increase
) private {
Node storage n = chain.nodes[acct];
(uint256 prev, uint256 next) = getPos(
chain,
n.sum,
n.prev,
n.next,
increase
);
if (next != n.next || prev != n.prev) {
_branchOff(chain, acct);
_hInsert(chain, acct, prev, next);
}
}
// ==== Chain & Queue ====
function _appendToQueue(
Chain storage chain,
Node storage n,
uint40 acct
) private {
if (chain.para.qtyOfSticks > 0) {
chain.nodes[chain.para.tail].next = acct;
} else {
chain.para.head = acct;
}
n.prev = chain.para.tail;
n.next = 0;
chain.para.tail = acct;
if (_isOnChain(n)) n.cat -= 3;
_increaseQtyOfSticks(chain);
}
function _appendToChain(
Chain storage chain,
Node storage n,
uint40 acct
) private {
n.prev = chain.nodes[0].prev;
chain.nodes[n.prev].next = acct;
chain.nodes[0].prev = acct;
n.next = 0;
if (_isInQueue(n)) n.cat += 3;
_increaseQtyOfBranches(chain);
}
function _onChainTest(
Chain storage chain,
Node storage r
) private view returns(bool) {
return uint(r.sum) * 10000 >= uint(totalVotes(chain)) * minVoteRatioOnChain(chain);
}
function _upChainAndMove(
Chain storage chain,
Node storage n,
uint40 acct
) private {
_trimChain(chain);
_branchOff(chain, acct);
_appendToChain(chain, n, acct);
_move(chain, acct, true);
}
function _trimChain(
Chain storage chain
) private {
uint40 cur = chain.nodes[0].prev;
while (cur > 0) {
Node storage t = chain.nodes[cur];
uint40 prev = t.prev;
if (!_onChainTest(chain, t))
_offChain(chain, t, cur);
else break;
cur = prev;
}
}
function _offChain(
Chain storage chain,
Node storage n,
uint40 acct
) private {
_branchOff(chain, acct);
_appendToQueue(chain, n, acct);
}
// ---- Categories Of Node ----
function _isOnChain(
Node storage n
) private view returns (bool) {
return n.cat > 2;
}
function _isInQueue(
Node storage n
) private view returns (bool) {
return n.cat < 2;
}
function _isIndepMember(
Node storage n
) private view returns (bool) {
return n.cat % 3 == 0;
}
function _isInGroup(
Node storage n
) private view returns (bool) {
return n.cat % 3 > 0;
}
function _isGroupRep(
Node storage n
) private view returns (bool) {
return n.cat % 3 == 1;
}
function _isGroupMember(
Node storage n
) private view returns (bool) {
return n.cat == uint8(CatOfNode.GroupMember);
}
function _notGroupMember(
Node storage n
) private view returns (bool) {
return n.cat % 3 < 2;
}
// ==== setting ====
function _increaseQtyOfBranches(Chain storage chain) private {
chain.para.qtyOfBranches++;
}
function _increaseQtyOfMembers(Chain storage chain) private {
chain.para.qtyOfMembers++;
}
function _increaseQtyOfSticks(Chain storage chain) private {
chain.para.qtyOfSticks++;
}
function _increaseTotalVotes(Chain storage chain, uint64 deltaAmt) private {
chain.nodes[0].sum += deltaAmt;
}
function _decreaseQtyOfBranches(Chain storage chain) private {
chain.para.qtyOfBranches--;
}
function _decreaseQtyOfMembers(Chain storage chain) private {
chain.para.qtyOfMembers--;
}
function _decreaseQtyOfSticks(Chain storage chain) private {
chain.para.qtyOfSticks--;
}
function _decreaseTotalVotes(Chain storage chain, uint64 deltaAmt) private {
chain.nodes[0].sum -= deltaAmt;
}
//################
//## Read ##
//################
function isMember(Chain storage chain, uint256 acct)
public
view
returns (bool)
{
return chain.nodes[acct].ptr != 0;
}
// ==== Zero Node ====
function tail(Chain storage chain) public view returns (uint40) {
return chain.nodes[0].prev;
}
function head(Chain storage chain) public view returns (uint40) {
return chain.nodes[0].next;
}
function totalVotes(Chain storage chain) public view returns (uint64) {
return chain.nodes[0].sum;
}
function basedOnPar(Chain storage chain) public view returns (bool) {
return chain.nodes[0].cat == 1;
}
// ---- Para ----
function headOfQueue(Chain storage chain)
public
view
returns (uint40)
{
return chain.para.head;
}
function tailOfQueue(Chain storage chain)
public
view
returns (uint40)
{
return chain.para.tail;
}
function maxQtyOfMembers(Chain storage chain)
public
view
returns (uint32)
{
return chain.para.maxQtyOfMembers;
}
function minVoteRatioOnChain(Chain storage chain)
public
view
returns (uint16)
{
uint16 min = chain.para.minVoteRatioOnChain;
return min > 0 ? min : 500;
}
function qtyOfBranches(Chain storage chain) public view returns (uint32) {
return chain.para.qtyOfBranches;
}
function qtyOfGroups(Chain storage chain) public view returns (uint32) {
return chain.para.qtyOfBranches + chain.para.qtyOfSticks;
}
function qtyOfTopMembers(Chain storage chain)
public view
returns(uint qty)
{
uint cur = chain.nodes[0].next;
while(cur > 0) {
qty++;
cur = nextNode(chain, cur);
}
}
function qtyOfMembers(Chain storage chain) public view returns (uint32) {
return chain.para.qtyOfMembers;
}
// ==== locate position ====
function getPos(
Chain storage chain,
uint256 amount,
uint256 prev,
uint256 next,
bool increase
) public view returns (uint256, uint256) {
if (increase)
while (prev > 0 && chain.nodes[prev].sum < amount) {
next = prev;
prev = chain.nodes[prev].prev;
}
else
while (next > 0 && chain.nodes[next].sum > amount) {
prev = next;
next = chain.nodes[next].next;
}
return (prev, next);
}
function nextNode(Chain storage chain, uint256 acct)
public view returns (uint256 next)
{
Node storage n = chain.nodes[acct];
if (_isIndepMember(n)) {
next = n.next;
} else if (_isGroupRep(n)) {
next = n.ptr;
} else if (_isGroupMember(n)) {
next = (n.next > 0) ? n.next : chain.nodes[n.ptr].next;
}
}
function getNode(Chain storage chain, uint256 acct)
public view returns (Node memory n)
{
n = chain.nodes[acct];
}
// ==== group ====
function rootOf(Chain storage chain, uint256 acct)
public
view
memberExist(chain, acct)
returns (uint40 group)
{
Node storage n = chain.nodes[acct];
group = (n.cat == uint8(CatOfNode.GroupMember)) ? n.ptr : uint40(acct) ;
}
function deepOfBranch(Chain storage chain, uint256 acct)
public
view
memberExist(chain, acct)
returns (uint256 deep)
{
Node storage n = chain.nodes[acct];
if (_isIndepMember(n)) deep = 1;
else if (_isGroupRep(n)) deep = _deepOfBranch(chain, acct);
else deep = _deepOfBranch(chain, n.ptr);
}
function _deepOfBranch(Chain storage chain, uint256 root)
private
view
returns (uint256 deep)
{
deep = 1;
uint40 next = chain.nodes[root].ptr;
while (next > 0) {
deep++;
next = chain.nodes[next].next;
}
}
function votesOfGroup(Chain storage chain, uint256 acct)
public
view
returns (uint64 votes)
{
uint256 group = rootOf(chain, acct);
votes = chain.nodes[group].sum;
}
function membersOfGroup(Chain storage chain, uint256 acct)
public
view
returns (uint256[] memory list)
{
uint256 cur = rootOf(chain, acct);
uint256 len = deepOfBranch(chain, acct);
list = new uint256[](len);
uint256 i = 0;
while (i < len) {
list[i] = cur;
cur = nextNode(chain, cur);
i++;
}
}
function affiliated(
Chain storage chain,
uint256 acct1,
uint256 acct2
)
public
view
memberExist(chain, acct1)
memberExist(chain, acct2)
returns (bool)
{
Node storage n1 = chain.nodes[acct1];
Node storage n2 = chain.nodes[acct2];
return n1.ptr == n2.ptr || n1.ptr == acct2 || n2.ptr == acct1;
}
// ==== members ====
function topMembersList(Chain storage chain)
public view
returns (uint256[] memory list)
{
uint256 len = qtyOfTopMembers(chain);
list = new uint[](len);
len = 0;
uint cur = chain.nodes[0].next;
_seqListOfQueue(chain, list, cur, len);
}
function sortedMembersList(Chain storage chain)
public
view
returns (uint256[] memory list)
{
uint256 len = qtyOfMembers(chain);
list = new uint[](len);
uint cur = chain.nodes[0].next;
len = 0;
len = _seqListOfQueue(chain, list, cur, len);
cur = chain.para.head;
_seqListOfQueue(chain, list, cur, len);
}
function _seqListOfQueue(
Chain storage chain,
uint[] memory list,
uint cur,
uint i
) private view returns (uint) {
while (cur > 0) {
list[i] = cur;
cur = nextNode(chain, cur);
i++;
}
return i;
}
// ==== Backup / Restore ====
function getSnapshot(Chain storage chain)
public view
returns (Node[] memory list, Para memory para)
{
para = chain.para;
uint256 len = qtyOfMembers(chain);
list = new Node[](len + 1);
list[0] = chain.nodes[0];
uint256 cur = chain.nodes[0].next;
len = 1;
len = _backupNodes(chain, list, cur, len);
cur = para.head;
_backupNodes(chain, list, cur, len);
}
function _backupNodes(
Chain storage chain,
Node[] memory list,
uint cur,
uint i
) private view returns (uint) {
while (cur > 0) {
list[i] = chain.nodes[cur];
cur = nextNode(chain, cur);
i++;
}
return i;
}
function restoreChain(
Chain storage chain,
Node[] memory list,
Para memory para
) public {
chain.nodes[0] = list[0];
chain.para = para;
uint256 cur = list[0].next;
uint256 i = 1;
i = _restoreNodes(chain, list, cur, i);
cur = para.head;
_restoreNodes(chain, list, cur, i);
}
function _restoreNodes(
Chain storage chain,
Node[] memory list,
uint cur,
uint i
) private returns (uint) {
while (cur > 0) {
chain.nodes[cur] = list[i];
cur = nextNode(chain, cur);
i++;
}
return i;
}
// ==== MockDeals ====
function mockDealsOfIA(
Chain storage chain,
IInvestmentAgreement _ia
) public {
uint[] memory seqList = _ia.getSeqList();
uint256 len = seqList.length;
while (len > 0) {
DealsRepo.Deal memory deal = _ia.getDeal(seqList[len-1]);
uint64 amount = basedOnPar(chain) ? deal.body.par : deal.body.paid;
if (deal.head.seller > 0) {
mockDealOfSell(chain, deal.head.seller, amount);
}
mockDealOfBuy(chain, deal.body.buyer, deal.body.groupOfBuyer, amount);
len--;
}
}
function mockDealOfSell(
Chain storage chain,
uint256 seller,
uint amount
) public {
increaseAmt(chain, seller, amount, false);
if (chain.nodes[seller].amt == 0)
delNode(chain, seller);
}
function mockDealOfBuy(
Chain storage chain,
uint256 buyer,
uint256 group,
uint amount
) public {
addNode(chain, buyer);
increaseAmt(chain, buyer, amount, true);
if (rootOf(chain, buyer) != group)
top2Sub(chain, buyer, group);
}
}// SPDX-License-Identifier: UNLICENSED
/* *
* Copyright (c) 2021-2023 LI LI @ JINGTIAN & GONGCHENG.
*
* This WORK is licensed under ComBoox SoftWare License 1.0, a copy of which
* can be obtained at:
* [https://github.com/paul-lee-attorney/comboox]
*
* THIS WORK IS PROVIDED ON AN "AS IS" BASIS, WITHOUT
* WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
* TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. IN NO
* EVENT SHALL ANY CONTRIBUTOR BE LIABLE TO YOU FOR ANY DAMAGES.
*
* YOU ARE PROHIBITED FROM DEPLOYING THE SMART CONTRACTS OF THIS WORK, IN WHOLE
* OR IN PART, FOR WHATEVER PURPOSE, ON ANY BLOCKCHAIN NETWORK THAT HAS ONE OR
* MORE NODES THAT ARE OUT OF YOUR CONTROL.
* */
pragma solidity ^0.8.8;
import "./LockersRepo.sol";
library UsersRepo {
using LockersRepo for LockersRepo.Repo;
struct Key {
address pubKey;
uint16 discount;
uint40 gift;
uint40 coupon;
}
struct User {
Key primeKey;
Key backupKey;
}
struct Rule {
uint40 eoaRewards;
uint40 coaRewards;
uint40 floor;
uint16 rate;
uint16 para;
}
struct Repo {
// userNo => User
mapping(uint256 => User) users;
// key => userNo
mapping(address => uint) userNo;
LockersRepo.Repo lockers;
}
// platformRule: Rule({
// eoaRewards: users[0].primeKey.gift,
// coaRewards: users[0].backupKey.gift,
// floor: users[0].backupKey.coupon,
// rate: users[0].primeKey.discount,
// para: users[0].backupKey.discount
// });
// counterOfUers: users[0].primeKey.coupon;
// ####################
// ## Modifier ##
// ####################
modifier onlyOwner(Repo storage repo, address msgSender) {
require(msgSender == getOwner(repo),
"UR.mf.OO: not owner");
_;
}
modifier onlyKeeper(Repo storage repo, address msgSender) {
require(msgSender == getBookeeper(repo),
"UR.mf.OK: not bookeeper");
_;
}
modifier onlyPrimeKey(Repo storage repo, address msgSender) {
require(msgSender == repo.users[getUserNo(repo, msgSender)].primeKey.pubKey,
"UR.mf.OPK: not primeKey");
_;
}
// ########################
// ## Opts Setting ##
// ########################
function ruleParser(bytes32 sn) public pure
returns(Rule memory rule)
{
uint _sn = uint(sn);
rule = Rule({
eoaRewards: uint40(_sn >> 216),
coaRewards: uint40(_sn >> 176),
floor: uint40(_sn >> 136),
rate: uint16(_sn >> 120),
para: uint16(_sn >> 96)
});
}
function setPlatformRule(Repo storage repo, bytes32 snOfRule, address msgSender)
public onlyOwner(repo, msgSender) onlyPrimeKey(repo, msgSender)
{
Rule memory rule = ruleParser(snOfRule);
User storage opt = repo.users[0];
opt.primeKey.discount = rule.rate;
opt.primeKey.gift = rule.eoaRewards;
opt.backupKey.discount = rule.para;
opt.backupKey.gift = rule.coaRewards;
opt.backupKey.coupon = rule.floor;
}
function getPlatformRule(Repo storage repo) public view
returns (Rule memory rule)
{
User storage opt = repo.users[0];
rule = Rule({
eoaRewards: opt.primeKey.gift,
coaRewards: opt.backupKey.gift,
floor: opt.backupKey.coupon,
rate: opt.primeKey.discount,
para: opt.backupKey.discount
});
}
function transferOwnership(Repo storage repo, address newOwner, address msgSender)
public onlyOwner(repo, msgSender)
{
repo.users[1].primeKey.pubKey = newOwner;
}
function handoverCenterKey(Repo storage repo, address newKeeper, address msgSender)
public onlyKeeper(repo, msgSender)
{
repo.users[1].backupKey.pubKey = newKeeper;
}
// ==== Author Setting ====
function infoParser(bytes32 info) public pure returns(Key memory)
{
uint _info = uint(info);
Key memory out = Key({
pubKey: address(0),
discount: uint16(_info >> 80),
gift: uint40(_info >> 40),
coupon: uint40(_info)
});
return out;
}
function setRoyaltyRule(
Repo storage repo,
bytes32 snOfRoyalty,
address msgSender
) public onlyPrimeKey(repo, msgSender) {
Key memory rule = infoParser(snOfRoyalty);
uint author = getUserNo(repo, msgSender);
User storage a = repo.users[author];
a.backupKey.discount = rule.discount;
a.backupKey.gift = rule.gift;
a.backupKey.coupon = rule.coupon;
}
function getRoyaltyRule(Repo storage repo, uint author)
public view returns (Key memory)
{
require (author > 0, 'zero author');
Key memory rule = repo.users[author].backupKey;
delete rule.pubKey;
return rule;
}
// ##################
// ## Points ##
// ##################
function mintAndLockPoints(Repo storage repo, uint to, uint amt, uint expireDate, bytes32 hashLock, address msgSender)
public onlyOwner(repo, msgSender) returns (LockersRepo.Head memory head)
{
head = _prepareLockerHead(repo, to, amt, expireDate, msgSender);
repo.lockers.lockPoints(head, hashLock);
}
function _prepareLockerHead(
Repo storage repo,
uint to,
uint amt,
uint expireDate,
address msgSender
) private view returns (LockersRepo.Head memory head) {
uint40 caller = getUserNo(repo, msgSender);
require((amt >> 128) == 0,
"UR.prepareLockerHead: amt overflow");
head = LockersRepo.Head({
from: caller,
to: uint40(to),
expireDate: uint48(expireDate),
value: uint128(amt)
});
}
function lockPoints(Repo storage repo, uint to, uint amt, uint expireDate, bytes32 hashLock, address msgSender)
public onlyPrimeKey(repo, msgSender) returns (LockersRepo.Head memory head)
{
head = _prepareLockerHead(repo, to, amt, expireDate, msgSender);
repo.lockers.lockPoints(head, hashLock);
}
function lockConsideration(
Repo storage repo,
uint to,
uint amt,
uint expireDate,
address counterLocker,
bytes calldata payload,
bytes32 hashLock,
address msgSender
) public onlyPrimeKey(repo, msgSender) returns (LockersRepo.Head memory head) {
head = _prepareLockerHead(repo, to, amt, expireDate, msgSender);
LockersRepo.Body memory body = LockersRepo.Body({
counterLocker: counterLocker,
payload: payload
});
repo.lockers.lockConsideration(head, body, hashLock);
}
function pickupPoints(
Repo storage repo,
bytes32 hashLock,
string memory hashKey,
address msgSender
) public returns (LockersRepo.Head memory head)
{
uint caller = getUserNo(repo, msgSender);
head = repo.lockers.pickupPoints(hashLock, hashKey, caller);
}
function withdrawDeposit(
Repo storage repo,
bytes32 hashLock,
address msgSender
) public onlyPrimeKey(repo, msgSender) returns (LockersRepo.Head memory head) {
uint caller = getUserNo(repo, msgSender);
head = repo.lockers.withdrawDeposit(hashLock, caller);
}
function getLocker(
Repo storage repo,
bytes32 hashLock
) public view returns (LockersRepo.Locker memory locker)
{
locker = repo.lockers.getLocker(hashLock);
}
function getLocksList(
Repo storage repo
) public view returns (bytes32[] memory)
{
return repo.lockers.getSnList();
}
// ##########################
// ## User & Members ##
// ##########################
// ==== reg user ====
function _increaseCounterOfUsers(Repo storage repo) private returns (uint40) {
repo.users[0].primeKey.coupon++;
return repo.users[0].primeKey.coupon;
}
function regUser(Repo storage repo, address msgSender) public
returns (User memory )
{
require(!isKey(repo, msgSender), "UserRepo.RegUser: used key");
uint seqOfUser = _increaseCounterOfUsers(repo);
repo.userNo[msgSender] = seqOfUser;
User memory user;
user.primeKey.pubKey = msgSender;
Rule memory rule = getPlatformRule(repo);
if (_isContract(msgSender)) {
user.primeKey.discount = 1;
user.primeKey.gift = rule.coaRewards;
} else user.primeKey.gift = rule.eoaRewards;
repo.users[seqOfUser] = user;
return user;
}
function _isContract(address acct) private view returns (bool) {
uint32 size;
assembly {
size := extcodesize(acct)
}
return size != 0;
}
function setBackupKey(Repo storage repo, address bKey, address msgSender)
public onlyPrimeKey(repo, msgSender)
{
require (!isKey(repo, bKey), "UR.SBK: used key");
uint caller = getUserNo(repo, msgSender);
User storage user = repo.users[caller];
require(user.backupKey.pubKey == address(0),
"UR.SBK: already set backupKey");
user.backupKey.pubKey = bKey;
repo.userNo[bKey] = caller;
}
function upgradeBackupToPrime(
Repo storage repo,
address msgSender
) public {
User storage user = repo.users[getUserNo(repo, msgSender)];
(user.primeKey.pubKey, user.backupKey.pubKey) =
(user.backupKey.pubKey, user.primeKey.pubKey);
}
// ##############
// ## Read I/O ##
// ##############
// ==== options ====
function counterOfUsers(Repo storage repo) public view returns (uint40) {
return repo.users[0].primeKey.coupon;
}
function getOwner(Repo storage repo) public view returns (address) {
return repo.users[1].primeKey.pubKey;
}
function getBookeeper(Repo storage repo) public view returns (address) {
return repo.users[1].backupKey.pubKey;
}
// ==== register ====
function isKey(Repo storage repo, address key) public view returns (bool) {
return repo.userNo[key] > 0;
}
function getUser(Repo storage repo, address msgSender)
public view returns (User memory)
{
return repo.users[getUserNo(repo, msgSender)];
}
function getUserNo(Repo storage repo, address msgSender)
public view returns(uint40)
{
uint40 user = uint40(repo.userNo[msgSender]);
if (user > 0) return user;
else revert ("UR.getUserNo: not registered");
}
}{
"optimizer": {
"enabled": true,
"runs": 200
},
"outputSelection": {
"*": {
"*": [
"evm.bytecode",
"evm.deployedBytecode",
"devdoc",
"userdoc",
"metadata",
"abi"
]
}
},
"libraries": {
"contracts/lib/EnumerableSet.sol": {
"EnumerableSet": "0xc4f24b69792ff96f0316fab2d7df198af8033930"
},
"contracts/lib/OptionsRepo.sol": {
"OptionsRepo": "0x05c0543826e2c0bfb4182c8d5ad2954adc484940"
},
"contracts/lib/RolesRepo.sol": {
"RolesRepo": "0xbd04750ec4ecc513c81db997f06d615349d6215d"
}
}
}Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"directKeeper","type":"address"},{"indexed":false,"internalType":"address","name":"regCenter","type":"address"},{"indexed":true,"internalType":"address","name":"generalKeeper","type":"address"}],"name":"Init","type":"event"},{"anonymous":false,"inputs":[],"name":"LockContents","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"keeper","type":"address"}],"name":"SetDirectKeeper","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"acct","type":"address"}],"name":"SetOwner","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"acct","type":"address"}],"name":"SetRoleAdmin","type":"event"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"abandonRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"seqOfOpt","type":"uint256"},{"internalType":"uint256","name":"obligor","type":"uint256"}],"name":"addObligorIntoOpt","outputs":[{"internalType":"bool","name":"flag","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"counterOfOptions","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"snOfOpt","type":"bytes32"},{"internalType":"bytes32","name":"snOfCond","type":"bytes32"},{"internalType":"uint256","name":"rightholder","type":"uint256"},{"internalType":"uint256","name":"paid","type":"uint256"},{"internalType":"uint256","name":"par","type":"uint256"}],"name":"createOption","outputs":[{"components":[{"internalType":"uint32","name":"seqOfOpt","type":"uint32"},{"internalType":"uint8","name":"typeOfOpt","type":"uint8"},{"internalType":"uint16","name":"classOfShare","type":"uint16"},{"internalType":"uint32","name":"rate","type":"uint32"},{"internalType":"uint48","name":"issueDate","type":"uint48"},{"internalType":"uint48","name":"triggerDate","type":"uint48"},{"internalType":"uint16","name":"execDays","type":"uint16"},{"internalType":"uint16","name":"closingDays","type":"uint16"},{"internalType":"uint40","name":"obligor","type":"uint40"}],"internalType":"struct OptionsRepo.Head","name":"head","type":"tuple"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"seqOfOpt","type":"uint256"}],"name":"delOption","outputs":[{"internalType":"bool","name":"flag","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getAllOptions","outputs":[{"components":[{"components":[{"internalType":"uint32","name":"seqOfOpt","type":"uint32"},{"internalType":"uint8","name":"typeOfOpt","type":"uint8"},{"internalType":"uint16","name":"classOfShare","type":"uint16"},{"internalType":"uint32","name":"rate","type":"uint32"},{"internalType":"uint48","name":"issueDate","type":"uint48"},{"internalType":"uint48","name":"triggerDate","type":"uint48"},{"internalType":"uint16","name":"execDays","type":"uint16"},{"internalType":"uint16","name":"closingDays","type":"uint16"},{"internalType":"uint40","name":"obligor","type":"uint40"}],"internalType":"struct OptionsRepo.Head","name":"head","type":"tuple"},{"components":[{"internalType":"uint32","name":"seqOfCond","type":"uint32"},{"internalType":"uint8","name":"logicOpr","type":"uint8"},{"internalType":"uint8","name":"compOpr1","type":"uint8"},{"internalType":"uint64","name":"para1","type":"uint64"},{"internalType":"uint8","name":"compOpr2","type":"uint8"},{"internalType":"uint64","name":"para2","type":"uint64"},{"internalType":"uint8","name":"compOpr3","type":"uint8"},{"internalType":"uint64","name":"para3","type":"uint64"}],"internalType":"struct CondsRepo.Cond","name":"cond","type":"tuple"},{"components":[{"internalType":"uint48","name":"closingDeadline","type":"uint48"},{"internalType":"uint40","name":"rightholder","type":"uint40"},{"internalType":"uint64","name":"paid","type":"uint64"},{"internalType":"uint64","name":"par","type":"uint64"},{"internalType":"uint8","name":"state","type":"uint8"},{"internalType":"uint16","name":"para","type":"uint16"},{"internalType":"uint16","name":"argu","type":"uint16"}],"internalType":"struct OptionsRepo.Body","name":"body","type":"tuple"}],"internalType":"struct OptionsRepo.Option[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getDK","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"seqOfOpt","type":"uint256"}],"name":"getObligorsOfOption","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"seqOfOpt","type":"uint256"}],"name":"getOption","outputs":[{"components":[{"components":[{"internalType":"uint32","name":"seqOfOpt","type":"uint32"},{"internalType":"uint8","name":"typeOfOpt","type":"uint8"},{"internalType":"uint16","name":"classOfShare","type":"uint16"},{"internalType":"uint32","name":"rate","type":"uint32"},{"internalType":"uint48","name":"issueDate","type":"uint48"},{"internalType":"uint48","name":"triggerDate","type":"uint48"},{"internalType":"uint16","name":"execDays","type":"uint16"},{"internalType":"uint16","name":"closingDays","type":"uint16"},{"internalType":"uint40","name":"obligor","type":"uint40"}],"internalType":"struct OptionsRepo.Head","name":"head","type":"tuple"},{"components":[{"internalType":"uint32","name":"seqOfCond","type":"uint32"},{"internalType":"uint8","name":"logicOpr","type":"uint8"},{"internalType":"uint8","name":"compOpr1","type":"uint8"},{"internalType":"uint64","name":"para1","type":"uint64"},{"internalType":"uint8","name":"compOpr2","type":"uint8"},{"internalType":"uint64","name":"para2","type":"uint64"},{"internalType":"uint8","name":"compOpr3","type":"uint8"},{"internalType":"uint64","name":"para3","type":"uint64"}],"internalType":"struct CondsRepo.Cond","name":"cond","type":"tuple"},{"components":[{"internalType":"uint48","name":"closingDeadline","type":"uint48"},{"internalType":"uint40","name":"rightholder","type":"uint40"},{"internalType":"uint64","name":"paid","type":"uint64"},{"internalType":"uint64","name":"par","type":"uint64"},{"internalType":"uint8","name":"state","type":"uint8"},{"internalType":"uint16","name":"para","type":"uint16"},{"internalType":"uint16","name":"argu","type":"uint16"}],"internalType":"struct OptionsRepo.Body","name":"body","type":"tuple"}],"internalType":"struct OptionsRepo.Option","name":"option","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleAdmin","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getSeqList","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"acct","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"acct","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"directKeeper","type":"address"},{"internalType":"address","name":"regCenter","type":"address"},{"internalType":"address","name":"generalKeeper","type":"address"}],"name":"init","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"isFinalized","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"seqOfOpt","type":"uint256"},{"internalType":"uint256","name":"acct","type":"uint256"}],"name":"isObligor","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"seqOfOpt","type":"uint256"}],"name":"isOption","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lockContents","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"qtyOfOptions","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"seqOfOpt","type":"uint256"},{"internalType":"uint256","name":"obligor","type":"uint256"}],"name":"removeObligorFromOpt","outputs":[{"internalType":"bool","name":"flag","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"acct","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"acct","type":"address"}],"name":"setDirectKeeper","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"acct","type":"address"}],"name":"setOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"acct","type":"address"}],"name":"setRoleAdmin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"target","type":"address"}],"name":"takeBackKeys","outputs":[],"stateMutability":"nonpayable","type":"function"}]Contract Creation Code
608060405234801561001057600080fd5b50611f68806100206000396000f3fe608060405234801561001057600080fd5b50600436106101a95760003560e01c8063565beabf116100f95780638d4e408311610097578063b3be456a11610071578063b3be456a146103a5578063d547741f146103b8578063f045a769146103cb578063f96ebeb7146103d357600080fd5b80638d4e40831461036a57806391d148541461037f578063ad7631921461039257600080fd5b80636be8496f116100d35780636be8496f14610329578063809bf77b1461033c578063893d20e81461034f5780638bb9c5bf1461035757600080fd5b8063565beabf146102ed57806356da88d1146103035780635a601d611461031657600080fd5b806326d29afa116101665780633da9a600116101405780633da9a600146102a157806340b6c23e146102b45780634b4b0d37146102c757806350627aea146102dc57600080fd5b806326d29afa146102585780632f2ff15d1461027b5780633bce3c301461028e57600080fd5b806306552ff3146101ae5780631368f327146101c357806313af4035146101e55780631ce5e9a6146101f8578063243fbc2114610218578063248a9ca31461022d575b600080fd5b6101c16101bc3660046115aa565b6103f3565b005b6101cb6104e2565b60405163ffffffff90911681526020015b60405180910390f35b6101c16101f3366004611606565b61056d565b61020b610206366004611623565b61061c565b6040516101dc91906117f7565b6102206106b4565b6040516101dc9190611806565b61024061023b366004611623565b61073e565b6040516001600160a01b0390911681526020016101dc565b61026b61026636600461184a565b6107cb565b60405190151581526020016101dc565b6101c161028936600461186c565b61093a565b61026b61029c36600461184a565b6109ac565b6101c16102af366004611623565b6109f5565b6101c16102c2366004611606565b610a64565b6102cf610b04565b6040516101dc919061189c565b6002546001600160a01b0316610240565b6102f5610b8e565b6040519081526020016101dc565b61026b61031136600461184a565b610c14565b61026b610324366004611623565b610d27565b6101c1610337366004611606565b610e68565b61022061034a366004611623565b610efd565b610240610f8e565b6101c1610365366004611623565b611014565b600054600160a01b900460ff1660021461026b565b61026b61038d36600461186c565b611050565b6101c16103a036600461186c565b61108e565b61026b6103b3366004611623565b611136565b6101c16103c636600461186c565b611178565b6101c16111b6565b6103e66103e13660046118df565b611343565b6040516101dc919061191a565b60405163013faf4960e71b8152600060048201526001600160a01b038516602482015273bd04750ec4ecc513c81db997f06d615349d6215d90639fd7a4809060440160006040518083038186803b15801561044d57600080fd5b505af4158015610461573d6000803e3d6000fd5b5050600280546001600160a01b038781166001600160a01b0319928316811790935560038054888316908416811790915560048054888416941684179055604051908152919450919250908716907f11e5c44db2defe55dea9656ae7848e94dc9f3e66eb342a7b84547a68d31c393a9060200160405180910390a450505050565b6040516316cfb4dd60e21b8152600560048201526000907305c0543826e2c0bfb4182c8d5ad2954adc48494090635b3ed3749060240160206040518083038186803b15801561053057600080fd5b505af4158015610544573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105689190611942565b905090565b6040516351e7e80560e01b8152600060048201526001600160a01b038216602482015233604482015273bd04750ec4ecc513c81db997f06d615349d6215d906351e7e8059060640160006040518083038186803b1580156105cd57600080fd5b505af41580156105e1573d6000803e3d6000fd5b50506040516001600160a01b03841692507f167d3e9c1016ab80e58802ca9da10ce5c6a0f4debc46a2e7a2cd9e56899a4fb59150600090a250565b6106246114b0565b604051625e5c6b60e41b815260056004820152602481018390527305c0543826e2c0bfb4182c8d5ad2954adc484940906305e5c6b0906044016103006040518083038186803b15801561067657600080fd5b505af415801561068a573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106ae9190611cd7565b92915050565b60405163364fa04b60e01b8152600560048201526060907305c0543826e2c0bfb4182c8d5ad2954adc4849409063364fa04b9060240160006040518083038186803b15801561070257600080fd5b505af4158015610716573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526105689190810190611d18565b604051639423dd0560e01b8152600060048201819052602482018390529073bd04750ec4ecc513c81db997f06d615349d6215d90639423dd059060440160206040518083038186803b15801561079357600080fd5b505af41580156107a7573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106ae9190611dae565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf7390610815908490684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b15801561082d57600080fd5b505af4158015610841573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108659190611dea565b61088a5760405162461bcd60e51b815260040161088190611e0c565b60405180910390fd5b61089383611136565b156106ae576000838152600660205260409081902090516392f13c3f60e01b815260048101919091526024810183905273c4f24b69792ff96f0316fab2d7df198af8033930906392f13c3f906044015b60206040518083038186803b1580156108fb57600080fd5b505af415801561090f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109339190611dea565b9392505050565b604051633d3c5d2760e01b815273bd04750ec4ecc513c81db997f06d615349d6215d90633d3c5d279061097890600090869086903390600401611e3a565b60006040518083038186803b15801561099057600080fd5b505af41580156109a4573d6000803e3d6000fd5b505050505050565b60405163d848781b60e01b81526005600482015260248101839052604481018290526000907305c0543826e2c0bfb4182c8d5ad2954adc4849409063d848781b906064016108e3565b60405163bf2755e360e01b815273bd04750ec4ecc513c81db997f06d615349d6215d9063bf2755e390610a319060009085903390600401611dcb565b60006040518083038186803b158015610a4957600080fd5b505af4158015610a5d573d6000803e3d6000fd5b5050505050565b6002546001600160a01b03163314610aaf5760405162461bcd60e51b815260206004820152600e60248201526d1050cb9bdb9b1e5112ce881393d560921b6044820152606401610881565b604051636be8496f60e01b81523360048201526001600160a01b03821690636be8496f90602401600060405180830381600087803b158015610af057600080fd5b505af1158015610a5d573d6000803e3d6000fd5b60405163741524b760e11b8152600560048201526060907305c0543826e2c0bfb4182c8d5ad2954adc4849409063e82a496e9060240160006040518083038186803b158015610b5257600080fd5b505af4158015610b66573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526105689190810190611e61565b6040516337e02dad60e11b8152600560048201526000907305c0543826e2c0bfb4182c8d5ad2954adc48494090636fc05b5a9060240160206040518083038186803b158015610bdc57600080fd5b505af4158015610bf0573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105689190611efc565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf7390610c5e908490684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b158015610c7657600080fd5b505af4158015610c8a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610cae9190611dea565b610cca5760405162461bcd60e51b815260040161088190611e0c565b610cd383611136565b156106ae576000838152600660205260409081902090516392fd02df60e01b815260048101919091526024810183905273c4f24b69792ff96f0316fab2d7df198af8033930906392fd02df906044016108e3565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf7390610d71908490684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b158015610d8957600080fd5b505af4158015610d9d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610dc19190611dea565b610ddd5760405162461bcd60e51b815260040161088190611e0c565b604051632009570d60e11b815260056004820152602481018390527305c0543826e2c0bfb4182c8d5ad2954adc48494090634012ae1a906044015b60206040518083038186803b158015610e3057600080fd5b505af4158015610e44573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106ae9190611dea565b6002546001600160a01b03163314610eb35760405162461bcd60e51b815260206004820152600e60248201526d1050cb9bdb9b1e5112ce881393d560921b6044820152606401610881565b600280546001600160a01b0319166001600160a01b0383169081179091556040517fe435cbeab64e4eae6497b9d8289704bd709d624a80704ef3ca9af777534bc6ec90600090a250565b60405163107977b960e21b815260056004820152602481018290526060907305c0543826e2c0bfb4182c8d5ad2954adc484940906341e5dee49060440160006040518083038186803b158015610f5257600080fd5b505af4158015610f66573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526106ae9190810190611d18565b6040516394916c4760e01b81526000600482018190529073bd04750ec4ecc513c81db997f06d615349d6215d906394916c479060240160206040518083038186803b158015610fdc57600080fd5b505af4158015610ff0573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105689190611dae565b604051632d16622360e11b815273bd04750ec4ecc513c81db997f06d615349d6215d90635a2cc44690610a319060009085903390600401611dcb565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf73906108e390849087908790600401611dcb565b60405163f17b815160e01b815273bd04750ec4ecc513c81db997f06d615349d6215d9063f17b8151906110cc90600090869086903390600401611e3a565b60006040518083038186803b1580156110e457600080fd5b505af41580156110f8573d6000803e3d6000fd5b50506040516001600160a01b03841692508491507fc58ebc6bc4ca6dc8356357e625ffa0eb58d0db30d94a31a80a840e1cc74d2df990600090a35050565b60405163a9b823e360e01b815260056004820152602481018290526000907305c0543826e2c0bfb4182c8d5ad2954adc4849409063a9b823e390604401610e18565b60405163303b779f60e01b815273bd04750ec4ecc513c81db997f06d615349d6215d9063303b779f9061097890600090869086903390600401611e3a565b600054600160a01b900460ff166001146112125760405162461bcd60e51b815260206004820152601c60248201527f41432e6c6f636b436f6e74656e74733a2077726f6e67207374617465000000006044820152606401610881565b60405163bf2755e360e01b8152339073bd04750ec4ecc513c81db997f06d615349d6215d9063bf2755e39061125c90600090684174746f726e65797360b81b908690600401611dcb565b60006040518083038186803b15801561127457600080fd5b505af4158015611288573d6000803e3d6000fd5b50506040516351e7e80560e01b815260006004820181905260248201526001600160a01b038416604482015273bd04750ec4ecc513c81db997f06d615349d6215d92506351e7e805915060640160006040518083038186803b1580156112ed57600080fd5b505af4158015611301573d6000803e3d6000fd5b50506000805460ff60a01b1916600160a11b1781556040517f35012b512309c322acf68632cd6cb702ff0e1d776fbe9394e38ffc13b434fe759350909150a150565b61134b611546565b604051632c7ebf7360e01b815273bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf739061139390600090684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b1580156113ab57600080fd5b505af41580156113bf573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906113e39190611dea565b6113ff5760405162461bcd60e51b815260040161088190611e0c565b604051633b29f3dd60e11b8152600560048201526024810187905260448101869052606481018590526084810184905260a481018390527305c0543826e2c0bfb4182c8d5ad2954adc48494090637653e7ba9060c4016101206040518083038186803b15801561146e57600080fd5b505af4158015611482573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906114a69190611f15565b9695505050505050565b60405180606001604052806114c3611546565b8152604080516101008101825260008082526020828101829052928201819052606082018190526080820181905260a0820181905260c0820181905260e082015291019081526040805160e08101825260008082526020828101829052928201819052606082018190526080820181905260a0820181905260c082015291015290565b6040805161012081018252600080825260208201819052918101829052606081018290526080810182905260a0810182905260c0810182905260e0810182905261010081019190915290565b6001600160a01b03811681146115a757600080fd5b50565b600080600080608085870312156115c057600080fd5b84356115cb81611592565b935060208501356115db81611592565b925060408501356115eb81611592565b915060608501356115fb81611592565b939692955090935050565b60006020828403121561161857600080fd5b813561093381611592565b60006020828403121561163557600080fd5b5035919050565b63ffffffff815116825260ff60208201511660208301526040810151611668604084018261ffff169052565b506060810151611680606084018263ffffffff169052565b50608081015161169a608084018265ffffffffffff169052565b5060a08101516116b460a084018265ffffffffffff169052565b5060c08101516116ca60c084018261ffff169052565b5060e08101516116e060e084018261ffff169052565b506101009081015164ffffffffff16910152565b6116ff82825161163c565b602081015163ffffffff81511661012084015260ff60208201511661014084015260ff604082015116610160840152606081015161174a61018085018267ffffffffffffffff169052565b5060808181015160ff9081166101a086015260a08084015167ffffffffffffffff9081166101c088015260c08086015184166101e089015260e0909501518116610200880152604095860151805165ffffffffffff16610220890152602081015164ffffffffff1661024089015295860151811661026088015260608601511661028087015291840151166102a085015282015161ffff9081166102c0850152910151166102e090910152565b61030081016106ae82846116f4565b6020808252825182820181905260009190848201906040850190845b8181101561183e57835183529284019291840191600101611822565b50909695505050505050565b6000806040838503121561185d57600080fd5b50508035926020909101359150565b6000806040838503121561187f57600080fd5b82359150602083013561189181611592565b809150509250929050565b6020808252825182820181905260009190848201906040850190845b8181101561183e576118cb8385516116f4565b9284019261030092909201916001016118b8565b600080600080600060a086880312156118f757600080fd5b505083359560208501359550604085013594606081013594506080013592509050565b61012081016106ae828461163c565b805163ffffffff8116811461193d57600080fd5b919050565b60006020828403121561195457600080fd5b61093382611929565b634e487b7160e01b600052604160045260246000fd5b604051610120810167ffffffffffffffff811182821017156119975761199761195d565b60405290565b6040516060810167ffffffffffffffff811182821017156119975761199761195d565b604051610100810167ffffffffffffffff811182821017156119975761199761195d565b604051601f8201601f1916810167ffffffffffffffff81118282101715611a0d57611a0d61195d565b604052919050565b805160ff8116811461193d57600080fd5b805161ffff8116811461193d57600080fd5b805165ffffffffffff8116811461193d57600080fd5b805164ffffffffff8116811461193d57600080fd5b60006101208284031215611a7657600080fd5b611a7e611973565b9050611a8982611929565b8152611a9760208301611a15565b6020820152611aa860408301611a26565b6040820152611ab960608301611929565b6060820152611aca60808301611a38565b6080820152611adb60a08301611a38565b60a0820152611aec60c08301611a26565b60c0820152611afd60e08301611a26565b60e0820152610100611b10818401611a4e565b9082015292915050565b805167ffffffffffffffff8116811461193d57600080fd5b600060e08284031215611b4457600080fd5b60405160e0810181811067ffffffffffffffff82111715611b6757611b6761195d565b604052905080611b7683611a38565b8152611b8460208401611a4e565b6020820152611b9560408401611b1a565b6040820152611ba660608401611b1a565b6060820152611bb760808401611a15565b6080820152611bc860a08401611a26565b60a0820152611bd960c08401611a26565b60c08201525092915050565b6000818303610300811215611bf957600080fd5b611c0161199d565b9150611c0d8484611a63565b825261010061011f1982011215611c2357600080fd5b50611c2c6119c0565b611c396101208401611929565b8152611c486101408401611a15565b6020820152611c5a6101608401611a15565b6040820152611c6c6101808401611b1a565b6060820152611c7e6101a08401611a15565b6080820152611c906101c08401611b1a565b60a0820152611ca26101e08401611a15565b60c0820152611cb46102008401611b1a565b60e08201526020820152611ccc836102208401611b32565b604082015292915050565b60006103008284031215611cea57600080fd5b6109338383611be5565b600067ffffffffffffffff821115611d0e57611d0e61195d565b5060051b60200190565b60006020808385031215611d2b57600080fd5b825167ffffffffffffffff811115611d4257600080fd5b8301601f81018513611d5357600080fd5b8051611d66611d6182611cf4565b6119e4565b81815260059190911b82018301908381019087831115611d8557600080fd5b928401925b82841015611da357835182529284019290840190611d8a565b979650505050505050565b600060208284031215611dc057600080fd5b815161093381611592565b92835260208301919091526001600160a01b0316604082015260600190565b600060208284031215611dfc57600080fd5b8151801515811461093357600080fd5b6020808252601490820152731050cb9bdb9b1e505d1d1bdc9b995e4e881393d560621b604082015260600190565b93845260208401929092526001600160a01b03908116604084015216606082015260800190565b60006020808385031215611e7457600080fd5b825167ffffffffffffffff811115611e8b57600080fd5b8301601f81018513611e9c57600080fd5b8051611eaa611d6182611cf4565b8181526103009182028301840191848201919088841115611eca57600080fd5b938501935b83851015611ef057611ee18986611be5565b83529384019391850191611ecf565b50979650505050505050565b600060208284031215611f0e57600080fd5b5051919050565b60006101208284031215611f2857600080fd5b6109338383611a6356fea2646970667358221220c3e4ae494832a69b89d2b6ec62571b42bdbb1eb51f64a74150f2dac54cde534164736f6c63430008080033
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106101a95760003560e01c8063565beabf116100f95780638d4e408311610097578063b3be456a11610071578063b3be456a146103a5578063d547741f146103b8578063f045a769146103cb578063f96ebeb7146103d357600080fd5b80638d4e40831461036a57806391d148541461037f578063ad7631921461039257600080fd5b80636be8496f116100d35780636be8496f14610329578063809bf77b1461033c578063893d20e81461034f5780638bb9c5bf1461035757600080fd5b8063565beabf146102ed57806356da88d1146103035780635a601d611461031657600080fd5b806326d29afa116101665780633da9a600116101405780633da9a600146102a157806340b6c23e146102b45780634b4b0d37146102c757806350627aea146102dc57600080fd5b806326d29afa146102585780632f2ff15d1461027b5780633bce3c301461028e57600080fd5b806306552ff3146101ae5780631368f327146101c357806313af4035146101e55780631ce5e9a6146101f8578063243fbc2114610218578063248a9ca31461022d575b600080fd5b6101c16101bc3660046115aa565b6103f3565b005b6101cb6104e2565b60405163ffffffff90911681526020015b60405180910390f35b6101c16101f3366004611606565b61056d565b61020b610206366004611623565b61061c565b6040516101dc91906117f7565b6102206106b4565b6040516101dc9190611806565b61024061023b366004611623565b61073e565b6040516001600160a01b0390911681526020016101dc565b61026b61026636600461184a565b6107cb565b60405190151581526020016101dc565b6101c161028936600461186c565b61093a565b61026b61029c36600461184a565b6109ac565b6101c16102af366004611623565b6109f5565b6101c16102c2366004611606565b610a64565b6102cf610b04565b6040516101dc919061189c565b6002546001600160a01b0316610240565b6102f5610b8e565b6040519081526020016101dc565b61026b61031136600461184a565b610c14565b61026b610324366004611623565b610d27565b6101c1610337366004611606565b610e68565b61022061034a366004611623565b610efd565b610240610f8e565b6101c1610365366004611623565b611014565b600054600160a01b900460ff1660021461026b565b61026b61038d36600461186c565b611050565b6101c16103a036600461186c565b61108e565b61026b6103b3366004611623565b611136565b6101c16103c636600461186c565b611178565b6101c16111b6565b6103e66103e13660046118df565b611343565b6040516101dc919061191a565b60405163013faf4960e71b8152600060048201526001600160a01b038516602482015273bd04750ec4ecc513c81db997f06d615349d6215d90639fd7a4809060440160006040518083038186803b15801561044d57600080fd5b505af4158015610461573d6000803e3d6000fd5b5050600280546001600160a01b038781166001600160a01b0319928316811790935560038054888316908416811790915560048054888416941684179055604051908152919450919250908716907f11e5c44db2defe55dea9656ae7848e94dc9f3e66eb342a7b84547a68d31c393a9060200160405180910390a450505050565b6040516316cfb4dd60e21b8152600560048201526000907305c0543826e2c0bfb4182c8d5ad2954adc48494090635b3ed3749060240160206040518083038186803b15801561053057600080fd5b505af4158015610544573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105689190611942565b905090565b6040516351e7e80560e01b8152600060048201526001600160a01b038216602482015233604482015273bd04750ec4ecc513c81db997f06d615349d6215d906351e7e8059060640160006040518083038186803b1580156105cd57600080fd5b505af41580156105e1573d6000803e3d6000fd5b50506040516001600160a01b03841692507f167d3e9c1016ab80e58802ca9da10ce5c6a0f4debc46a2e7a2cd9e56899a4fb59150600090a250565b6106246114b0565b604051625e5c6b60e41b815260056004820152602481018390527305c0543826e2c0bfb4182c8d5ad2954adc484940906305e5c6b0906044016103006040518083038186803b15801561067657600080fd5b505af415801561068a573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106ae9190611cd7565b92915050565b60405163364fa04b60e01b8152600560048201526060907305c0543826e2c0bfb4182c8d5ad2954adc4849409063364fa04b9060240160006040518083038186803b15801561070257600080fd5b505af4158015610716573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526105689190810190611d18565b604051639423dd0560e01b8152600060048201819052602482018390529073bd04750ec4ecc513c81db997f06d615349d6215d90639423dd059060440160206040518083038186803b15801561079357600080fd5b505af41580156107a7573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106ae9190611dae565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf7390610815908490684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b15801561082d57600080fd5b505af4158015610841573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108659190611dea565b61088a5760405162461bcd60e51b815260040161088190611e0c565b60405180910390fd5b61089383611136565b156106ae576000838152600660205260409081902090516392f13c3f60e01b815260048101919091526024810183905273c4f24b69792ff96f0316fab2d7df198af8033930906392f13c3f906044015b60206040518083038186803b1580156108fb57600080fd5b505af415801561090f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109339190611dea565b9392505050565b604051633d3c5d2760e01b815273bd04750ec4ecc513c81db997f06d615349d6215d90633d3c5d279061097890600090869086903390600401611e3a565b60006040518083038186803b15801561099057600080fd5b505af41580156109a4573d6000803e3d6000fd5b505050505050565b60405163d848781b60e01b81526005600482015260248101839052604481018290526000907305c0543826e2c0bfb4182c8d5ad2954adc4849409063d848781b906064016108e3565b60405163bf2755e360e01b815273bd04750ec4ecc513c81db997f06d615349d6215d9063bf2755e390610a319060009085903390600401611dcb565b60006040518083038186803b158015610a4957600080fd5b505af4158015610a5d573d6000803e3d6000fd5b5050505050565b6002546001600160a01b03163314610aaf5760405162461bcd60e51b815260206004820152600e60248201526d1050cb9bdb9b1e5112ce881393d560921b6044820152606401610881565b604051636be8496f60e01b81523360048201526001600160a01b03821690636be8496f90602401600060405180830381600087803b158015610af057600080fd5b505af1158015610a5d573d6000803e3d6000fd5b60405163741524b760e11b8152600560048201526060907305c0543826e2c0bfb4182c8d5ad2954adc4849409063e82a496e9060240160006040518083038186803b158015610b5257600080fd5b505af4158015610b66573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526105689190810190611e61565b6040516337e02dad60e11b8152600560048201526000907305c0543826e2c0bfb4182c8d5ad2954adc48494090636fc05b5a9060240160206040518083038186803b158015610bdc57600080fd5b505af4158015610bf0573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105689190611efc565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf7390610c5e908490684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b158015610c7657600080fd5b505af4158015610c8a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610cae9190611dea565b610cca5760405162461bcd60e51b815260040161088190611e0c565b610cd383611136565b156106ae576000838152600660205260409081902090516392fd02df60e01b815260048101919091526024810183905273c4f24b69792ff96f0316fab2d7df198af8033930906392fd02df906044016108e3565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf7390610d71908490684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b158015610d8957600080fd5b505af4158015610d9d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610dc19190611dea565b610ddd5760405162461bcd60e51b815260040161088190611e0c565b604051632009570d60e11b815260056004820152602481018390527305c0543826e2c0bfb4182c8d5ad2954adc48494090634012ae1a906044015b60206040518083038186803b158015610e3057600080fd5b505af4158015610e44573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906106ae9190611dea565b6002546001600160a01b03163314610eb35760405162461bcd60e51b815260206004820152600e60248201526d1050cb9bdb9b1e5112ce881393d560921b6044820152606401610881565b600280546001600160a01b0319166001600160a01b0383169081179091556040517fe435cbeab64e4eae6497b9d8289704bd709d624a80704ef3ca9af777534bc6ec90600090a250565b60405163107977b960e21b815260056004820152602481018290526060907305c0543826e2c0bfb4182c8d5ad2954adc484940906341e5dee49060440160006040518083038186803b158015610f5257600080fd5b505af4158015610f66573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526106ae9190810190611d18565b6040516394916c4760e01b81526000600482018190529073bd04750ec4ecc513c81db997f06d615349d6215d906394916c479060240160206040518083038186803b158015610fdc57600080fd5b505af4158015610ff0573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105689190611dae565b604051632d16622360e11b815273bd04750ec4ecc513c81db997f06d615349d6215d90635a2cc44690610a319060009085903390600401611dcb565b604051632c7ebf7360e01b815260009073bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf73906108e390849087908790600401611dcb565b60405163f17b815160e01b815273bd04750ec4ecc513c81db997f06d615349d6215d9063f17b8151906110cc90600090869086903390600401611e3a565b60006040518083038186803b1580156110e457600080fd5b505af41580156110f8573d6000803e3d6000fd5b50506040516001600160a01b03841692508491507fc58ebc6bc4ca6dc8356357e625ffa0eb58d0db30d94a31a80a840e1cc74d2df990600090a35050565b60405163a9b823e360e01b815260056004820152602481018290526000907305c0543826e2c0bfb4182c8d5ad2954adc4849409063a9b823e390604401610e18565b60405163303b779f60e01b815273bd04750ec4ecc513c81db997f06d615349d6215d9063303b779f9061097890600090869086903390600401611e3a565b600054600160a01b900460ff166001146112125760405162461bcd60e51b815260206004820152601c60248201527f41432e6c6f636b436f6e74656e74733a2077726f6e67207374617465000000006044820152606401610881565b60405163bf2755e360e01b8152339073bd04750ec4ecc513c81db997f06d615349d6215d9063bf2755e39061125c90600090684174746f726e65797360b81b908690600401611dcb565b60006040518083038186803b15801561127457600080fd5b505af4158015611288573d6000803e3d6000fd5b50506040516351e7e80560e01b815260006004820181905260248201526001600160a01b038416604482015273bd04750ec4ecc513c81db997f06d615349d6215d92506351e7e805915060640160006040518083038186803b1580156112ed57600080fd5b505af4158015611301573d6000803e3d6000fd5b50506000805460ff60a01b1916600160a11b1781556040517f35012b512309c322acf68632cd6cb702ff0e1d776fbe9394e38ffc13b434fe759350909150a150565b61134b611546565b604051632c7ebf7360e01b815273bd04750ec4ecc513c81db997f06d615349d6215d90632c7ebf739061139390600090684174746f726e65797360b81b903390600401611dcb565b60206040518083038186803b1580156113ab57600080fd5b505af41580156113bf573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906113e39190611dea565b6113ff5760405162461bcd60e51b815260040161088190611e0c565b604051633b29f3dd60e11b8152600560048201526024810187905260448101869052606481018590526084810184905260a481018390527305c0543826e2c0bfb4182c8d5ad2954adc48494090637653e7ba9060c4016101206040518083038186803b15801561146e57600080fd5b505af4158015611482573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906114a69190611f15565b9695505050505050565b60405180606001604052806114c3611546565b8152604080516101008101825260008082526020828101829052928201819052606082018190526080820181905260a0820181905260c0820181905260e082015291019081526040805160e08101825260008082526020828101829052928201819052606082018190526080820181905260a0820181905260c082015291015290565b6040805161012081018252600080825260208201819052918101829052606081018290526080810182905260a0810182905260c0810182905260e0810182905261010081019190915290565b6001600160a01b03811681146115a757600080fd5b50565b600080600080608085870312156115c057600080fd5b84356115cb81611592565b935060208501356115db81611592565b925060408501356115eb81611592565b915060608501356115fb81611592565b939692955090935050565b60006020828403121561161857600080fd5b813561093381611592565b60006020828403121561163557600080fd5b5035919050565b63ffffffff815116825260ff60208201511660208301526040810151611668604084018261ffff169052565b506060810151611680606084018263ffffffff169052565b50608081015161169a608084018265ffffffffffff169052565b5060a08101516116b460a084018265ffffffffffff169052565b5060c08101516116ca60c084018261ffff169052565b5060e08101516116e060e084018261ffff169052565b506101009081015164ffffffffff16910152565b6116ff82825161163c565b602081015163ffffffff81511661012084015260ff60208201511661014084015260ff604082015116610160840152606081015161174a61018085018267ffffffffffffffff169052565b5060808181015160ff9081166101a086015260a08084015167ffffffffffffffff9081166101c088015260c08086015184166101e089015260e0909501518116610200880152604095860151805165ffffffffffff16610220890152602081015164ffffffffff1661024089015295860151811661026088015260608601511661028087015291840151166102a085015282015161ffff9081166102c0850152910151166102e090910152565b61030081016106ae82846116f4565b6020808252825182820181905260009190848201906040850190845b8181101561183e57835183529284019291840191600101611822565b50909695505050505050565b6000806040838503121561185d57600080fd5b50508035926020909101359150565b6000806040838503121561187f57600080fd5b82359150602083013561189181611592565b809150509250929050565b6020808252825182820181905260009190848201906040850190845b8181101561183e576118cb8385516116f4565b9284019261030092909201916001016118b8565b600080600080600060a086880312156118f757600080fd5b505083359560208501359550604085013594606081013594506080013592509050565b61012081016106ae828461163c565b805163ffffffff8116811461193d57600080fd5b919050565b60006020828403121561195457600080fd5b61093382611929565b634e487b7160e01b600052604160045260246000fd5b604051610120810167ffffffffffffffff811182821017156119975761199761195d565b60405290565b6040516060810167ffffffffffffffff811182821017156119975761199761195d565b604051610100810167ffffffffffffffff811182821017156119975761199761195d565b604051601f8201601f1916810167ffffffffffffffff81118282101715611a0d57611a0d61195d565b604052919050565b805160ff8116811461193d57600080fd5b805161ffff8116811461193d57600080fd5b805165ffffffffffff8116811461193d57600080fd5b805164ffffffffff8116811461193d57600080fd5b60006101208284031215611a7657600080fd5b611a7e611973565b9050611a8982611929565b8152611a9760208301611a15565b6020820152611aa860408301611a26565b6040820152611ab960608301611929565b6060820152611aca60808301611a38565b6080820152611adb60a08301611a38565b60a0820152611aec60c08301611a26565b60c0820152611afd60e08301611a26565b60e0820152610100611b10818401611a4e565b9082015292915050565b805167ffffffffffffffff8116811461193d57600080fd5b600060e08284031215611b4457600080fd5b60405160e0810181811067ffffffffffffffff82111715611b6757611b6761195d565b604052905080611b7683611a38565b8152611b8460208401611a4e565b6020820152611b9560408401611b1a565b6040820152611ba660608401611b1a565b6060820152611bb760808401611a15565b6080820152611bc860a08401611a26565b60a0820152611bd960c08401611a26565b60c08201525092915050565b6000818303610300811215611bf957600080fd5b611c0161199d565b9150611c0d8484611a63565b825261010061011f1982011215611c2357600080fd5b50611c2c6119c0565b611c396101208401611929565b8152611c486101408401611a15565b6020820152611c5a6101608401611a15565b6040820152611c6c6101808401611b1a565b6060820152611c7e6101a08401611a15565b6080820152611c906101c08401611b1a565b60a0820152611ca26101e08401611a15565b60c0820152611cb46102008401611b1a565b60e08201526020820152611ccc836102208401611b32565b604082015292915050565b60006103008284031215611cea57600080fd5b6109338383611be5565b600067ffffffffffffffff821115611d0e57611d0e61195d565b5060051b60200190565b60006020808385031215611d2b57600080fd5b825167ffffffffffffffff811115611d4257600080fd5b8301601f81018513611d5357600080fd5b8051611d66611d6182611cf4565b6119e4565b81815260059190911b82018301908381019087831115611d8557600080fd5b928401925b82841015611da357835182529284019290840190611d8a565b979650505050505050565b600060208284031215611dc057600080fd5b815161093381611592565b92835260208301919091526001600160a01b0316604082015260600190565b600060208284031215611dfc57600080fd5b8151801515811461093357600080fd5b6020808252601490820152731050cb9bdb9b1e505d1d1bdc9b995e4e881393d560621b604082015260600190565b93845260208401929092526001600160a01b03908116604084015216606082015260800190565b60006020808385031215611e7457600080fd5b825167ffffffffffffffff811115611e8b57600080fd5b8301601f81018513611e9c57600080fd5b8051611eaa611d6182611cf4565b8181526103009182028301840191848201919088841115611eca57600080fd5b938501935b83851015611ef057611ee18986611be5565b83529384019391850191611ecf565b50979650505050505050565b600060208284031215611f0e57600080fd5b5051919050565b60006101208284031215611f2857600080fd5b6109338383611a6356fea2646970667358221220c3e4ae494832a69b89d2b6ec62571b42bdbb1eb51f64a74150f2dac54cde534164736f6c63430008080033
Loading...
Loading
Loading...
Loading
Loading...
Loading
Net Worth in USD
$0.00
Net Worth in ETH
0
Multichain Portfolio | 35 Chains
| Chain | Token | Portfolio % | Price | Amount | Value |
|---|
Loading...
Loading
Loading...
Loading
Loading...
Loading
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.