编译器
0.8.19+commit.7dd6d404
文件 1 的 11:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 2 的 11:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 3 的 11:ERC20.sol
pragma solidity ^0.8.0;
import "./IERC20.sol";
import "./extensions/IERC20Metadata.sol";
import "../../utils/Context.sol";
contract ERC20 is Context, IERC20, IERC20Metadata {
mapping(address => uint256) private _balances;
mapping(address => mapping(address => uint256)) private _allowances;
uint256 private _totalSupply;
string private _name;
string private _symbol;
constructor(string memory name_, string memory symbol_) {
_name = name_;
_symbol = symbol_;
}
function name() public view virtual override returns (string memory) {
return _name;
}
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
function decimals() public view virtual override returns (uint8) {
return 18;
}
function totalSupply() public view virtual override returns (uint256) {
return _totalSupply;
}
function balanceOf(address account) public view virtual override returns (uint256) {
return _balances[account];
}
function transfer(address to, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_transfer(owner, to, amount);
return true;
}
function allowance(address owner, address spender) public view virtual override returns (uint256) {
return _allowances[owner][spender];
}
function approve(address spender, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_approve(owner, spender, amount);
return true;
}
function transferFrom(address from, address to, uint256 amount) public virtual override returns (bool) {
address spender = _msgSender();
_spendAllowance(from, spender, amount);
_transfer(from, to, amount);
return true;
}
function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
address owner = _msgSender();
_approve(owner, spender, allowance(owner, spender) + addedValue);
return true;
}
function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
address owner = _msgSender();
uint256 currentAllowance = allowance(owner, spender);
require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
unchecked {
_approve(owner, spender, currentAllowance - subtractedValue);
}
return true;
}
function _transfer(address from, address to, uint256 amount) internal virtual {
require(from != address(0), "ERC20: transfer from the zero address");
require(to != address(0), "ERC20: transfer to the zero address");
_beforeTokenTransfer(from, to, amount);
uint256 fromBalance = _balances[from];
require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
unchecked {
_balances[from] = fromBalance - amount;
_balances[to] += amount;
}
emit Transfer(from, to, amount);
_afterTokenTransfer(from, to, amount);
}
function _mint(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: mint to the zero address");
_beforeTokenTransfer(address(0), account, amount);
_totalSupply += amount;
unchecked {
_balances[account] += amount;
}
emit Transfer(address(0), account, amount);
_afterTokenTransfer(address(0), account, amount);
}
function _burn(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: burn from the zero address");
_beforeTokenTransfer(account, address(0), amount);
uint256 accountBalance = _balances[account];
require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
unchecked {
_balances[account] = accountBalance - amount;
_totalSupply -= amount;
}
emit Transfer(account, address(0), amount);
_afterTokenTransfer(account, address(0), amount);
}
function _approve(address owner, address spender, uint256 amount) internal virtual {
require(owner != address(0), "ERC20: approve from the zero address");
require(spender != address(0), "ERC20: approve to the zero address");
_allowances[owner][spender] = amount;
emit Approval(owner, spender, amount);
}
function _spendAllowance(address owner, address spender, uint256 amount) internal virtual {
uint256 currentAllowance = allowance(owner, spender);
if (currentAllowance != type(uint256).max) {
require(currentAllowance >= amount, "ERC20: insufficient allowance");
unchecked {
_approve(owner, spender, currentAllowance - amount);
}
}
}
function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual {}
function _afterTokenTransfer(address from, address to, uint256 amount) internal virtual {}
}
文件 4 的 11:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address to, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(address from, address to, uint256 amount) external returns (bool);
}
文件 5 的 11:IERC20Metadata.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
interface IERC20Metadata is IERC20 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
}
文件 6 的 11:IERC20Permit.sol
pragma solidity ^0.8.0;
interface IERC20Permit {
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
function nonces(address owner) external view returns (uint256);
function DOMAIN_SEPARATOR() external view returns (bytes32);
}
文件 7 的 11:IterableMapping.sol
pragma solidity ^0.8.19;
library IterableMapping {
struct Map {
address[] keys;
mapping(address => uint) values;
mapping(address => uint) indexOf;
mapping(address => bool) inserted;
}
function get(Map storage map, address key) public view returns (uint value) {
assembly {
mstore(0x0, key)
mstore(0x20, add(map.slot, 1))
value := sload(keccak256(0x0, 64))
}
}
function getKeyAtIndex(Map storage map, uint index) public view returns (address value) {
assembly {
mstore(0x0, map.slot)
value := sload(add(keccak256(0x0, 32), index))
}
}
function size(Map storage map) public view returns (uint length) {
assembly {
length := sload(map.slot)
}
}
function set(
Map storage map,
address key,
uint val
) public {
assembly {
mstore(0x0, key)
mstore(0x20, add(map.slot, 3))
let insertHash := keccak256(0x0, 64)
let cond := sload(insertHash)
if cond {
mstore(0x20, add(map.slot, 1))
sstore(keccak256(0x0, 64), val)
}
if iszero(cond) {
sstore(insertHash, 1)
mstore(0x20, add(map.slot, 1))
sstore(keccak256(0x0, 64), val)
mstore(0x20, add(map.slot, 2))
let keysLength := sload(map.slot)
sstore(keccak256(0x0, 64), keysLength)
mstore(0x0, map.slot)
sstore(add(keccak256(0x0, 32), keysLength), key)
sstore(map.slot, add(keysLength, 1))
}
}
}
function remove(Map storage map, address key) public {
if (!map.inserted[key]) {
return;
}
delete map.inserted[key];
delete map.values[key];
assembly {
let keysSlot := map.slot
let keysLength := sload(keysSlot)
mstore(0x0, shr(96, shl(96, key)))
mstore(0x20, add(keysSlot, 2))
let index := sload(keccak256(0x0, 64))
mstore(0x60, keysSlot)
let lastKey := sload(add(keccak256(0x60, 32), sub(keysLength, 1)))
mstore(0x60, keysSlot)
sstore(add(keccak256(0x60, 32), index), lastKey)
mstore(0x0, lastKey)
sstore(keccak256(0x0, 64), index)
mstore(0x60, 0)
}
map.keys.pop();
delete map.indexOf[key];
}
}
文件 8 的 11:Math.sol
pragma solidity ^0.8.0;
library Math {
enum Rounding {
Down,
Up,
Zero
}
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a > b ? a : b;
}
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
function average(uint256 a, uint256 b) internal pure returns (uint256) {
return (a & b) + (a ^ b) / 2;
}
function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
return a == 0 ? 0 : (a - 1) / b + 1;
}
function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) {
unchecked {
uint256 prod0;
uint256 prod1;
assembly {
let mm := mulmod(x, y, not(0))
prod0 := mul(x, y)
prod1 := sub(sub(mm, prod0), lt(mm, prod0))
}
if (prod1 == 0) {
return prod0 / denominator;
}
require(denominator > prod1, "Math: mulDiv overflow");
uint256 remainder;
assembly {
remainder := mulmod(x, y, denominator)
prod1 := sub(prod1, gt(remainder, prod0))
prod0 := sub(prod0, remainder)
}
uint256 twos = denominator & (~denominator + 1);
assembly {
denominator := div(denominator, twos)
prod0 := div(prod0, twos)
twos := add(div(sub(0, twos), twos), 1)
}
prod0 |= prod1 * twos;
uint256 inverse = (3 * denominator) ^ 2;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
result = prod0 * inverse;
return result;
}
}
function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) {
uint256 result = mulDiv(x, y, denominator);
if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
result += 1;
}
return result;
}
function sqrt(uint256 a) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
uint256 result = 1 << (log2(a) >> 1);
unchecked {
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
return min(result, a / result);
}
}
function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = sqrt(a);
return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
}
}
function log2(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 128;
}
if (value >> 64 > 0) {
value >>= 64;
result += 64;
}
if (value >> 32 > 0) {
value >>= 32;
result += 32;
}
if (value >> 16 > 0) {
value >>= 16;
result += 16;
}
if (value >> 8 > 0) {
value >>= 8;
result += 8;
}
if (value >> 4 > 0) {
value >>= 4;
result += 4;
}
if (value >> 2 > 0) {
value >>= 2;
result += 2;
}
if (value >> 1 > 0) {
result += 1;
}
}
return result;
}
function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log2(value);
return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
}
}
function log10(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >= 10 ** 64) {
value /= 10 ** 64;
result += 64;
}
if (value >= 10 ** 32) {
value /= 10 ** 32;
result += 32;
}
if (value >= 10 ** 16) {
value /= 10 ** 16;
result += 16;
}
if (value >= 10 ** 8) {
value /= 10 ** 8;
result += 8;
}
if (value >= 10 ** 4) {
value /= 10 ** 4;
result += 4;
}
if (value >= 10 ** 2) {
value /= 10 ** 2;
result += 2;
}
if (value >= 10 ** 1) {
result += 1;
}
}
return result;
}
function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log10(value);
return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0);
}
}
function log256(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 16;
}
if (value >> 64 > 0) {
value >>= 64;
result += 8;
}
if (value >> 32 > 0) {
value >>= 32;
result += 4;
}
if (value >> 16 > 0) {
value >>= 16;
result += 2;
}
if (value >> 8 > 0) {
result += 1;
}
}
return result;
}
function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log256(value);
return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0);
}
}
}
文件 9 的 11:SafeERC20.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/IERC20Permit.sol";
import "../../../utils/Address.sol";
library SafeERC20 {
using Address for address;
function safeTransfer(IERC20 token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
function safeApprove(IERC20 token, address spender, uint256 value) internal {
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
uint256 oldAllowance = token.allowance(address(this), spender);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
}
function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
}
}
function forceApprove(IERC20 token, address spender, uint256 value) internal {
bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
if (!_callOptionalReturnBool(token, approvalCall)) {
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
_callOptionalReturn(token, approvalCall);
}
}
function safePermit(
IERC20Permit token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) {
(bool success, bytes memory returndata) = address(token).call(data);
return
success && (returndata.length == 0 || abi.decode(returndata, (bool))) && Address.isContract(address(token));
}
}
文件 10 的 11:SafeMath.sol
pragma solidity ^0.8.0;
library SafeMath {
function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
uint256 c = a + b;
if (c < a) return (false, 0);
return (true, c);
}
}
function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b > a) return (false, 0);
return (true, a - b);
}
}
function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (a == 0) return (true, 0);
uint256 c = a * b;
if (c / a != b) return (false, 0);
return (true, c);
}
}
function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b == 0) return (false, 0);
return (true, a / b);
}
}
function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b == 0) return (false, 0);
return (true, a % b);
}
}
function add(uint256 a, uint256 b) internal pure returns (uint256) {
return a + b;
}
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
return a - b;
}
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
return a * b;
}
function div(uint256 a, uint256 b) internal pure returns (uint256) {
return a / b;
}
function mod(uint256 a, uint256 b) internal pure returns (uint256) {
return a % b;
}
function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
unchecked {
require(b <= a, errorMessage);
return a - b;
}
}
function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
unchecked {
require(b > 0, errorMessage);
return a / b;
}
}
function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
unchecked {
require(b > 0, errorMessage);
return a % b;
}
}
}
文件 11 的 11:UXGroupShares.sol
pragma solidity ^0.8.19;
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@openzeppelin/contracts/utils/math/SafeMath.sol";
import "@openzeppelin/contracts/utils/math/Math.sol";
import "./libs/IterableMapping.sol";
interface IUXGroupAccount {
function owner() external view returns (address);
}
contract UXGroupShares {
using SafeMath for uint256;
using IterableMapping for IterableMapping.Map;
struct SubjectData {
address SubjectAddress;
uint256 Val;
}
address public protocolFeeDestination;
uint256 public protocolFeePercent;
uint256 public subjectFeePercent;
IERC20 public settlementToken;
address public settlementTokenAddress;
mapping(address => mapping(address => uint256)) public sharesBalance;
mapping(address => uint256) public sharesSupply;
mapping(address => IterableMapping.Map) private sharesHoders;
mapping(address => uint256) public sharesCapitals;
mapping(address => address) public groupOwners;
mapping(address => bool) public superOperators;
event AuthorizedOperator(address indexed operator, address indexed holder);
event RevokedOperator(address indexed operator, address indexed holder);
event SetGroupOwners(address groupAccount, address groupOwner);
event TopUpCapital(address operator, address groupAccount, uint256 price);
event Trade(address trader, address subject, bool isBuy, uint256 shareAmount, uint256 tokenAmount, uint256 protocolTokenAmount, uint256 subjectTokenAmount, uint256 supply, uint256 capital);
modifier onlyAdmin() {
require(superOperators[msg.sender], "Not super operator");
_;
}
constructor(address _settlementTokenAddress) {
superOperators[msg.sender] = true;
settlementTokenAddress = _settlementTokenAddress;
settlementToken = IERC20(_settlementTokenAddress);
protocolFeeDestination = msg.sender;
}
function setSettlementToken(address _settlementTokenAddress) public onlyAdmin {
settlementTokenAddress = _settlementTokenAddress;
settlementToken = IERC20(_settlementTokenAddress);
}
function setFeeDestination(address _feeDestination) public onlyAdmin {
protocolFeeDestination = _feeDestination;
}
function setProtocolFeePercent(uint256 _feePercent) public onlyAdmin {
protocolFeePercent = _feePercent;
}
function setSubjectFeePercent(uint256 _feePercent) public onlyAdmin {
subjectFeePercent = _feePercent;
}
function setSharesSubjectGroupOwner(address _uxgroupAccount) public {
require(_uxgroupAccount != address(0), "UXGroupAccount address not set");
IUXGroupAccount groupAccount = IUXGroupAccount(_uxgroupAccount);
address groupOwner = groupAccount.owner();
require(groupOwner != address(0), "UXGroupAccount owner nonexistent");
groupOwners[_uxgroupAccount] = groupOwner;
emit SetGroupOwners(_uxgroupAccount, groupOwner);
}
function getTokenBalance(address tokenAddress, address _uxgroupAccount) public view returns (uint256) {
IERC20 tokenContract = IERC20(tokenAddress);
uint256 balance = tokenContract.balanceOf(_uxgroupAccount);
return balance;
}
function getEthBalance(address _uxgroupAccount) external view returns (uint256) {
uint256 balance = _uxgroupAccount.balance;
return balance;
}
function getHolders(address _uxgroupAccount) public view returns (uint256) {
return sharesHoders[_uxgroupAccount].size();
}
function multiGetHolders(address[] memory sharesSubjects) public view returns (SubjectData[] memory res) {
require(sharesSubjects.length > 0, "sharesSubjects.len is invalid ");
res = new SubjectData[](sharesSubjects.length);
for (uint256 i = 0; i < sharesSubjects.length; i++) {
SubjectData memory one;
one.SubjectAddress = sharesSubjects[i];
one.Val = getHolders(sharesSubjects[i]);
res[i] = one;
}
}
function getCapital(address _uxgroupAccount) public view returns (uint256) {
return sharesCapitals[_uxgroupAccount];
}
function multiGetCapital(address[] memory sharesSubjects) public view returns (SubjectData[] memory res) {
require(sharesSubjects.length > 0, "sharesSubjects.len is invalid ");
res = new SubjectData[](sharesSubjects.length);
for (uint256 i = 0; i < sharesSubjects.length; i++) {
SubjectData memory one;
one.SubjectAddress = sharesSubjects[i];
one.Val = getCapital(sharesSubjects[i]);
res[i] = one;
}
}
function getBaseNumber(uint256 holders,uint256 supply,uint256 amount) public pure returns (uint256) {
if(supply == 0){
return 1 ether;
}
uint256 number = 2 * holders * supply;
uint baseNumer = Math.log2(number,Math.Rounding.Up);
return baseNumer * amount;
}
function getPrice(uint256 supply, uint256 amount, address sharesSubject) public view returns (uint256) {
uint256 holders = getHolders(sharesSubject);
if(supply == 0){
return 1 ether;
}
uint256 baseNumer = getBaseNumber(holders, supply, amount);
if(baseNumer < 1){
baseNumer = 1;
}
uint256 summation = baseNumer * 100;
return summation * 1 ether;
}
function getBuyPrice(address sharesSubject, uint256 amount) public view returns (uint256) {
return getPrice(sharesSupply[sharesSubject], amount, sharesSubject);
}
function multiGetBuyPrice(address[] memory sharesSubjects) public view returns (SubjectData[] memory res) {
require(sharesSubjects.length > 0, "sharesSubjects.len is invalid ");
res = new SubjectData[](sharesSubjects.length);
for (uint256 i = 0; i < sharesSubjects.length; i++) {
SubjectData memory one;
one.SubjectAddress = sharesSubjects[i];
one.Val = getBuyPrice(sharesSubjects[i], 1);
res[i] = one;
}
}
function getSellPrice(address sharesSubject, uint256 amount) public view returns (uint256) {
uint256 capital = sharesCapitals[sharesSubject];
if(capital == 0){
return 0;
}
uint256 sellPrice = getPrice(sharesSupply[sharesSubject] - amount, amount, sharesSubject);
if(sellPrice > capital){
sellPrice = capital;
}
return sellPrice;
}
function multiGetSellPrice(address[] memory sharesSubjects) public view returns (SubjectData[] memory res) {
require(sharesSubjects.length > 0, "sharesSubjects.len is invalid ");
res = new SubjectData[](sharesSubjects.length);
for (uint256 i = 0; i < sharesSubjects.length; i++) {
SubjectData memory one;
one.SubjectAddress = sharesSubjects[i];
one.Val = getSellPrice(sharesSubjects[i], 1);
res[i] = one;
}
}
function multiGetSupply(address[] memory sharesSubjects) public view returns (SubjectData[] memory res) {
require(sharesSubjects.length > 0, "sharesSubjects.len is invalid ");
res = new SubjectData[](sharesSubjects.length);
for (uint256 i = 0; i < sharesSubjects.length; i++) {
SubjectData memory one;
one.SubjectAddress = sharesSubjects[i];
one.Val = sharesSupply[sharesSubjects[i]];
res[i] = one;
}
}
function getHoldersListBySubject(address sharesSubject) public view returns (SubjectData[] memory res) {
uint256 len = sharesHoders[sharesSubject].size();
if(len < 1){
res = new SubjectData[](len);
return res;
}
IterableMapping.Map storage mappingManager = sharesHoders[sharesSubject];
res = new SubjectData[](len);
for (uint256 i = 0; i < len; i++) {
SubjectData memory one;
one.SubjectAddress = mappingManager.getKeyAtIndex(i);
one.Val = mappingManager.get(one.SubjectAddress);
res[i] = one;
}
}
function topUpCapital(address sharesSubject, uint256 price) public {
uint256 tokenBalance = getTokenBalance(settlementTokenAddress,msg.sender);
require(tokenBalance >= price, "Insufficient payment");
bool success0 = settlementToken.transferFrom(msg.sender,address(this), price);
require(success0, "Unable to send funds");
sharesCapitals[sharesSubject] = sharesCapitals[sharesSubject] + price;
emit TopUpCapital(msg.sender, sharesSubject, price);
}
function getBuyPriceAfterFee(address sharesSubject, uint256 amount) public view returns (uint256) {
uint256 price = getBuyPrice(sharesSubject, amount);
uint256 protocolFee = price * protocolFeePercent / 100;
uint256 subjectFee = price * subjectFeePercent / 100;
return price + protocolFee + subjectFee;
}
function getSellPriceAfterFee(address sharesSubject, uint256 amount) public view returns (uint256) {
uint256 price = getSellPrice(sharesSubject, amount);
uint256 protocolFee = price * protocolFeePercent / 100;
uint256 subjectFee = price * subjectFeePercent / 100;
return price - protocolFee - subjectFee;
}
function buyShares(address sharesSubject, uint256 amount) public payable {
uint256 supply = sharesSupply[sharesSubject];
if(supply == 0){
setSharesSubjectGroupOwner(sharesSubject);
}
require(supply > 0 || groupOwners[sharesSubject] == msg.sender, "Only the shares' group owner can buy the first share");
uint256 price = getBuyPrice(sharesSubject, amount);
uint256 protocolFee = price * protocolFeePercent / 100;
uint256 subjectFee = price * subjectFeePercent / 100;
uint256 tokenBalance = getTokenBalance(settlementTokenAddress,msg.sender);
require(tokenBalance >= price + protocolFee + subjectFee, "Insufficient payment");
sharesBalance[sharesSubject][msg.sender] = sharesBalance[sharesSubject][msg.sender] + amount;
sharesSupply[sharesSubject] = supply + amount;
IterableMapping.Map storage mappingManager = sharesHoders[sharesSubject];
mappingManager.set(msg.sender, sharesBalance[sharesSubject][msg.sender]);
sharesCapitals[sharesSubject] = sharesCapitals[sharesSubject] + price;
uint256 capitail = sharesCapitals[sharesSubject];
emit Trade(msg.sender, sharesSubject, true, amount, price, protocolFee, subjectFee, supply + amount, capitail);
(bool success0 ) = settlementToken.transferFrom(msg.sender,address(this), price+protocolFee+subjectFee);
SafeERC20.safeTransfer(settlementToken, protocolFeeDestination, protocolFee);
SafeERC20.safeTransfer(settlementToken, sharesSubject, subjectFee);
require(success0, "Unable to send funds");
}
function sellShares(address sharesSubject, uint256 amount) public payable {
uint256 supply = sharesSupply[sharesSubject];
require(supply > amount, "Cannot sell the last share");
uint256 price = getSellPrice(sharesSubject, amount);
require(price > 0, "Cannot sell share, no funds");
require(sharesCapitals[sharesSubject] - price >= 0, "Cannot sell share, no funds");
uint256 protocolFee = price * protocolFeePercent / 100;
uint256 subjectFee = price * subjectFeePercent / 100;
require(sharesBalance[sharesSubject][msg.sender] >= amount, "Insufficient shares");
sharesBalance[sharesSubject][msg.sender] = sharesBalance[sharesSubject][msg.sender] - amount;
sharesSupply[sharesSubject] = supply - amount;
IterableMapping.Map storage mappingManager = sharesHoders[sharesSubject];
if(sharesBalance[sharesSubject][msg.sender] > 0){
mappingManager.set(msg.sender, sharesBalance[sharesSubject][msg.sender]);
}else{
mappingManager.remove(msg.sender);
}
sharesCapitals[sharesSubject] = sharesCapitals[sharesSubject] - price;
uint256 capitail = sharesCapitals[sharesSubject];
emit Trade(msg.sender, sharesSubject, false, amount, price, protocolFee, subjectFee, supply - amount, capitail);
SafeERC20.safeTransfer(settlementToken, msg.sender, price - protocolFee - subjectFee);
SafeERC20.safeTransfer(settlementToken, protocolFeeDestination, protocolFee);
SafeERC20.safeTransfer(settlementToken, sharesSubject, subjectFee);
}
function authorizeOperator(address _operator) external onlyAdmin {
superOperators[_operator] = true;
emit AuthorizedOperator(_operator, msg.sender);
}
function revokeOperator(address _operator) external onlyAdmin {
superOperators[_operator] = false;
emit RevokedOperator(_operator, msg.sender);
}
receive() external payable {}
function withdrawETH(address recipient, uint256 amount) public onlyAdmin {
require(amount > 0, "amount is zero");
uint256 balance = address(this).balance;
require(balance >= amount, "balance must be greater than amount");
payable(recipient).transfer(amount);
}
}
{
"compilationTarget": {
"contracts/UXGroupShares.sol": "UXGroupShares"
},
"evmVersion": "paris",
"libraries": {
"contracts/libs/IterableMapping.sol:IterableMapping": "0x40f153cba689b03289ba7af07e6da325426b7f75"
},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_settlementTokenAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":true,"internalType":"address","name":"holder","type":"address"}],"name":"AuthorizedOperator","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":true,"internalType":"address","name":"holder","type":"address"}],"name":"RevokedOperator","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"groupAccount","type":"address"},{"indexed":false,"internalType":"address","name":"groupOwner","type":"address"}],"name":"SetGroupOwners","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"address","name":"groupAccount","type":"address"},{"indexed":false,"internalType":"uint256","name":"price","type":"uint256"}],"name":"TopUpCapital","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"trader","type":"address"},{"indexed":false,"internalType":"address","name":"subject","type":"address"},{"indexed":false,"internalType":"bool","name":"isBuy","type":"bool"},{"indexed":false,"internalType":"uint256","name":"shareAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"protocolTokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"subjectTokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"supply","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"capital","type":"uint256"}],"name":"Trade","type":"event"},{"inputs":[{"internalType":"address","name":"_operator","type":"address"}],"name":"authorizeOperator","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"buyShares","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"holders","type":"uint256"},{"internalType":"uint256","name":"supply","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"getBaseNumber","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"getBuyPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"getBuyPriceAfterFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_uxgroupAccount","type":"address"}],"name":"getCapital","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_uxgroupAccount","type":"address"}],"name":"getEthBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_uxgroupAccount","type":"address"}],"name":"getHolders","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"}],"name":"getHoldersListBySubject","outputs":[{"components":[{"internalType":"address","name":"SubjectAddress","type":"address"},{"internalType":"uint256","name":"Val","type":"uint256"}],"internalType":"struct UXGroupShares.SubjectData[]","name":"res","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"supply","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address","name":"sharesSubject","type":"address"}],"name":"getPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"getSellPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"getSellPriceAfterFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenAddress","type":"address"},{"internalType":"address","name":"_uxgroupAccount","type":"address"}],"name":"getTokenBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"groupOwners","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"sharesSubjects","type":"address[]"}],"name":"multiGetBuyPrice","outputs":[{"components":[{"internalType":"address","name":"SubjectAddress","type":"address"},{"internalType":"uint256","name":"Val","type":"uint256"}],"internalType":"struct UXGroupShares.SubjectData[]","name":"res","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"sharesSubjects","type":"address[]"}],"name":"multiGetCapital","outputs":[{"components":[{"internalType":"address","name":"SubjectAddress","type":"address"},{"internalType":"uint256","name":"Val","type":"uint256"}],"internalType":"struct UXGroupShares.SubjectData[]","name":"res","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"sharesSubjects","type":"address[]"}],"name":"multiGetHolders","outputs":[{"components":[{"internalType":"address","name":"SubjectAddress","type":"address"},{"internalType":"uint256","name":"Val","type":"uint256"}],"internalType":"struct UXGroupShares.SubjectData[]","name":"res","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"sharesSubjects","type":"address[]"}],"name":"multiGetSellPrice","outputs":[{"components":[{"internalType":"address","name":"SubjectAddress","type":"address"},{"internalType":"uint256","name":"Val","type":"uint256"}],"internalType":"struct UXGroupShares.SubjectData[]","name":"res","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"sharesSubjects","type":"address[]"}],"name":"multiGetSupply","outputs":[{"components":[{"internalType":"address","name":"SubjectAddress","type":"address"},{"internalType":"uint256","name":"Val","type":"uint256"}],"internalType":"struct UXGroupShares.SubjectData[]","name":"res","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"protocolFeeDestination","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"protocolFeePercent","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_operator","type":"address"}],"name":"revokeOperator","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"sellShares","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"_feeDestination","type":"address"}],"name":"setFeeDestination","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_feePercent","type":"uint256"}],"name":"setProtocolFeePercent","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_settlementTokenAddress","type":"address"}],"name":"setSettlementToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_uxgroupAccount","type":"address"}],"name":"setSharesSubjectGroupOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_feePercent","type":"uint256"}],"name":"setSubjectFeePercent","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"settlementToken","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"settlementTokenAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"sharesBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"sharesCapitals","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"sharesSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"subjectFeePercent","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"superOperators","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"sharesSubject","type":"address"},{"internalType":"uint256","name":"price","type":"uint256"}],"name":"topUpCapital","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"withdrawETH","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]