Address Contract Verified
Address
0xb533687Ef77459093368c43E95f8Df1C2B5a1F7a
Balance
0 ETH
Nonce
2
Code Size
16582 bytes
Creator
0xa05cFFE2...454E at tx 0xc56dc411...6d7c82
Indexed Transactions
0
Contract Bytecode
16582 bytes
0x6080604052600436106102e85760003560e01c80636f9a880e11610190578063a9059cbb116100dc578063cc1776d311610095578063dd62ed3e1161006f578063dd62ed3e14610a71578063f144e62014610a91578063f2fde38b14610ab1578063f84e426514610ad157600080fd5b8063cc1776d3146109c8578063db408e1a14610a11578063dbe66ca014610a4157600080fd5b8063a9059cbb146108e9578063aa5f7e2614610909578063ad56c13c14610929578063b2a9c0c014610973578063c077ab4b14610993578063c0f306ef146109a857600080fd5b806392b4639011610149578063a457c2d711610123578063a457c2d714610855578063a5949bcf14610875578063a8aa1b3114610895578063a8b9d240146108c957600080fd5b806392b46390146107f257806395d89b41146108125780639e93ad8e1461082757600080fd5b80636f9a880e1461075557806370a0823114610775578063715018a61461079557806371778e7d146107aa578063751039fc146107bf5780638da5cb5b146107d457600080fd5b806331e79db01161024f57806356340e811161020857806364a81f1d116101e257806364a81f1d146106735780636843cd84146106935780636cdc3677146106b35780636d7adcad146106d357600080fd5b806356340e81146105b15780635cfb534e146106335780635ecc20221461065357600080fd5b806331e79db01461049a57806336496bf7146104ba57806339509351146104da57806345006628146104fa5780634e71d92d1461051a5780634f7041a51461052f57600080fd5b80631af71da6116102a15780631af71da6146103ef5780631b03ddc11461040f5780631ebcb3df1461042f57806323b872dd1461044957806330bb4cff14610469578063313ce5671461047e57600080fd5b8063048dec38146102f457806306f200031461031657806306fdde03146103295780630758d92414610354578063095ea7b3146103a057806318160ddd146103d057600080fd5b366102ef57005b600080fd5b34801561030057600080fd5b5061031461030f3660046139af565b610af1565b005b6103146103243660046139af565b610b8b565b34801561033557600080fd5b5061033e610ed4565b60405161034b91906139d3565b60405180910390f35b34801561036057600080fd5b506103887f0000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488d81565b6040516001600160a01b03909116815260200161034b565b3480156103ac57600080fd5b506103c06103bb366004613a21565b610f66565b604051901515815260200161034b565b3480156103dc57600080fd5b506002545b60405190815260200161034b565b3480156103fb57600080fd5b506103e161040a366004613a4d565b610f80565b34801561041b57600080fd5b50600c54610388906001600160a01b031681565b34801561043b57600080fd5b506010546103c09060ff1681565b34801561045557600080fd5b506103c0610464366004613a66565b611147565b34801561047557600080fd5b506103e161116b565b34801561048a57600080fd5b506040516012815260200161034b565b3480156104a657600080fd5b506103146104b53660046139af565b6111de565b3480156104c657600080fd5b506103e16104d53660046139af565b61126b565b3480156104e657600080fd5b506103c06104f5366004613a21565b611440565b34801561050657600080fd5b50610314610515366004613ac2565b611462565b34801561052657600080fd5b50610314611616565b34801561053b57600080fd5b50600a546105789065ffffffffffff80821691600160301b8104821691600160601b8204811691600160901b8104821691600160c01b9091041685565b6040805165ffffffffffff968716815294861660208601529285169284019290925283166060830152909116608082015260a00161034b565b3480156105bd57600080fd5b50600f546105f9906001600160601b0380821691600160601b810490911690600160c01b810465ffffffffffff1690600160f01b900460ff1684565b604080516001600160601b03958616815294909316602085015265ffffffffffff909116918301919091521515606082015260800161034b565b34801561063f57600080fd5b5061031461064e366004613b24565b61168e565b34801561065f57600080fd5b5061031461066e366004613b41565b6116cb565b34801561067f57600080fd5b5061031461068e366004613b7a565b611819565b34801561069f57600080fd5b506103e16106ae3660046139af565b611941565b3480156106bf57600080fd5b506103146106ce366004613ac2565b6119b1565b3480156106df57600080fd5b50600d54600e5461071a916001600160401b0380821692600160401b8304821692600160801b8104831692600160c01b909104169060ff1685565b604080516001600160401b0396871681529486166020860152928516928401929092529092166060820152901515608082015260a00161034b565b34801561076157600080fd5b50600954610388906001600160a01b031681565b34801561078157600080fd5b506103e16107903660046139af565b611b65565b3480156107a157600080fd5b50610314611b80565b3480156107b657600080fd5b506103e1611bf4565b3480156107cb57600080fd5b50610314611c3e565b3480156107e057600080fd5b506005546001600160a01b0316610388565b3480156107fe57600080fd5b5061031461080d3660046139af565b611d46565b34801561081e57600080fd5b5061033e611dd7565b34801561083357600080fd5b5061083d61271081565b6040516001600160401b03909116815260200161034b565b34801561086157600080fd5b506103c0610870366004613a21565b611de6565b34801561088157600080fd5b50600854610388906001600160a01b031681565b3480156108a157600080fd5b506103887f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e3581565b3480156108d557600080fd5b506103e16108e43660046139af565b611e61565b3480156108f557600080fd5b506103c0610904366004613a21565b611e94565b34801561091557600080fd5b50610314610924366004613a4d565b611ea2565b34801561093557600080fd5b506109496109443660046139af565b611f5d565b604080516001600160a01b039095168552602085019390935291830152606082015260800161034b565b34801561097f57600080fd5b5061031461098e366004613a4d565b611fe1565b34801561099f57600080fd5b506103e161212c565b3480156109b457600080fd5b506103146109c33660046139af565b6122d2565b3480156109d457600080fd5b50600b546105789065ffffffffffff80821691600160301b8104821691600160601b8204811691600160901b8104821691600160c01b9091041685565b348015610a1d57600080fd5b506103c0610a2c3660046139af565b60076020526000908152604090205460ff1681565b348015610a4d57600080fd5b506103c0610a5c3660046139af565b60066020526000908152604090205460ff1681565b348015610a7d57600080fd5b506103e1610a8c366004613ba3565b61232e565b348015610a9d57600080fd5b50610314610aac366004613b7a565b612359565b348015610abd57600080fd5b50610314610acc3660046139af565b61246a565b348015610add57600080fd5b50610314610aec366004613b41565b612555565b6005546001600160a01b03163314610b245760405162461bcd60e51b8152600401610b1b90613bd1565b60405180910390fd5b6001600160a01b038116610b695760405162461bcd60e51b815260206004820152600c60248201526b7a65726f206164647265737360a01b6044820152606401610b1b565b600880546001600160a01b0319166001600160a01b0392909216919091179055565b6005546001600160a01b03163314610bb55760405162461bcd60e51b8152600401610b1b90613bd1565b600047118015610bcd57506000610bcb30611b65565b115b610bd657600080fd5b7f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b031663d0e30db0476040518263ffffffff1660e01b81526004016000604051808303818588803b158015610c3157600080fd5b505af1158015610c45573d6000803e3d6000fd5b5050505050610c7d307f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e35610c7830611b65565b612678565b6040516370a0823160e01b81523060048201527f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc26001600160a01b03169063a9059cbb907f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e359083906370a0823190602401602060405180830381865afa158015610d0b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d2f9190613c06565b6040516001600160e01b031960e085901b1681526001600160a01b03909216600483015260248201526044016020604051808303816000875af1158015610d7a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d9e9190613c1f565b506040516335313c2160e11b81526001600160a01b0382811660048301527f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e351690636a627842906024016020604051808303816000875af1158015610e07573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e2b9190613c06565b5043601355600f805460ff60f01b1916600160f01b179055604080516350d25bcd60e01b815290516001600160a01b037f0000000000000000000000005f4ec3df9cbd43714fe2740f5e3616155c5b841916916350d25bcd91600482810192602092919082900301816000875af1158015610eaa573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610ece9190613c06565b60155550565b606060038054610ee390613c3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610f0f90613c3c565b8015610f5c5780601f10610f3157610100808354040283529160200191610f5c565b820191906000526020600020905b815481529060010190602001808311610f3f57829003601f168201915b5050505050905090565b600033610f7481858561281d565b60019150505b92915050565b600081600003610f9257506000919050565b6040805160028082526060820183526000926020830190803683370190505090507f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc281600081518110610fe757610fe7613c8c565b60200260200101906001600160a01b031690816001600160a01b031681525050308160018151811061101b5761101b613c8c565b6001600160a01b03928316602091820292909201015260405163d06ca61f60e01b81526000917f0000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488d169063d06ca61f9061107a9087908690600401613ce6565b600060405180830381865afa158015611097573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526110bf9190810190613cff565b600a54909150612710906110e390600160c01b900465ffffffffffff166032613dd2565b65ffffffffffff16826001815181106110fe576110fe613c8c565b60200260200101516111109190613df8565b61111a9190613e25565b8160018151811061112d5761112d613c8c565b602002602001015161113f9190613e39565b949350505050565b600033611155858285612941565b6111608585856129bb565b506001949350505050565b600c54604080516342d359d760e11b815290516000926001600160a01b0316916385a6b3ae9160048083019260209291908290030181865afa1580156111b5573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111d99190613c06565b905090565b6005546001600160a01b031633146112085760405162461bcd60e51b8152600401610b1b90613bd1565b600c5460405163031e79db60e41b81526001600160a01b038381166004830152909116906331e79db0906024015b600060405180830381600087803b15801561125057600080fd5b505af1158015611264573d6000803e3d6000fd5b5050505050565b60008061127783611e61565b90508060000361128a5750600092915050565b6040805160028082526060820183526000926020830190803683370190505090507f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2816000815181106112df576112df613c8c565b60200260200101906001600160a01b031690816001600160a01b031681525050308160018151811061131357611313613c8c565b6001600160a01b03928316602091820292909201015260405163d06ca61f60e01b81526000917f0000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488d169063d06ca61f906113729086908690600401613ce6565b600060405180830381865afa15801561138f573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f191682016040526113b79190810190613cff565b600a54909150612710906113db90600160c01b900465ffffffffffff166032613dd2565b65ffffffffffff16826001815181106113f6576113f6613c8c565b60200260200101516114089190613df8565b6114129190613e25565b8160018151811061142557611425613c8c565b60200260200101516114379190613e39565b95945050505050565b600033610f74818585611453838361232e565b61145d9190613e4c565b61281d565b6005546001600160a01b0316331461148c5760405162461bcd60e51b8152600401610b1b90613bd1565b6040805160a0810182526000608082015265ffffffffffff8681168252858116928201929092528382166020820152908216606082015281836114cf8688613dd2565b6114d99190613dd2565b6114e39190613dd2565b65ffffffffffff16608082018190526103e810156115385760405162461bcd60e51b81526020600482015260126024820152714b656570207461782062656c6f772031302560701b6044820152606401610b1b565b608081015160405165ffffffffffff90911681527fa6255338a5f732d64ceba7f4c18182567f9d1067eb984b46d478b37d72a52d119060200160405180910390a18051600b805460208401516040850151606086015160809096015165ffffffffffff908116600160c01b0265ffffffffffff60c01b19978216600160901b0265ffffffffffff60901b19938316600160601b0293909316600160601b600160c01b0319948316600160301b026001600160601b03199096169290971691909117939093179190911693909317929092179290921617905550505050565b600c5460405163bc4c4b3760e01b8152336004820152600060248201526001600160a01b039091169063bc4c4b37906044016020604051808303816000875af1158015611667573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061168b9190613c1f565b50565b6005546001600160a01b031633146116b85760405162461bcd60e51b8152600401610b1b90613bd1565b6012805460ff1916911515919091179055565b6005546001600160a01b031633146116f55760405162461bcd60e51b8152600401610b1b90613bd1565b6001600160a01b03821661173a5760405162461bcd60e51b815260206004820152600c60248201526b5a65726f204164647265737360a01b6044820152606401610b1b565b806117b5577f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b0316826001600160a01b0316036117b55760405162461bcd60e51b815260206004820152601260248201527121b0b73737ba103932b6b7bb32903830b4b960711b6044820152606401610b1b565b6001600160a01b038216600081815260076020908152604091829020805460ff19168515159081179091558251938452908301527f0712a8411c7892e205c3d35c5d2beabe76bb0b484bec43945f2024cd28eaae8d91015b60405180910390a15050565b6005546001600160a01b031633146118435760405162461bcd60e51b8152600401610b1b90613bd1565b61184f6012600a613f43565b6103e861185b60025490565b611866906005613df8565b6118709190613e25565b61187a9190613e25565b816001600160601b031610156118bc5760405162461bcd60e51b8152602060048201526007602482015266546f6f206c6f7760c81b6044820152606401610b1b565b6118c86012600a613f43565b6118db906001600160601b038316613df8565b600f8054600160601b600160c01b031916600160601b6001600160601b0393841681029190911791829055604051910490911681527fde064515fae8f8bb6d8ff19d2c6ba704322def7494147d8a971266430ade0788906020015b60405180910390a150565b600c5460405163156dbbf560e31b81526001600160a01b038381166004830152600092169063ab6ddfa8906024015b602060405180830381865afa15801561198d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610f7a9190613c06565b6005546001600160a01b031633146119db5760405162461bcd60e51b8152600401610b1b90613bd1565b6040805160a0810182526000608082015265ffffffffffff868116825285811692820192909252838216602082015290821660608201528183611a1e8688613dd2565b611a289190613dd2565b611a329190613dd2565b65ffffffffffff16608082018190526103e81015611a875760405162461bcd60e51b81526020600482015260126024820152714b656570207461782062656c6f772031302560701b6044820152606401610b1b565b608081015160405165ffffffffffff90911681527f7a758dc8e99047b028278b3e2ff1416d8493a7aacee7a5dc30b6bf93270eccce9060200160405180910390a18051600a805460208401516040850151606086015160809096015165ffffffffffff908116600160c01b0265ffffffffffff60c01b19978216600160901b0265ffffffffffff60901b19938316600160601b0293909316600160601b600160c01b0319948316600160301b026001600160601b03199096169290971691909117939093179190911693909317929092179290921617905550505050565b6001600160a01b031660009081526020819052604090205490565b6005546001600160a01b03163314611baa5760405162461bcd60e51b8152600401610b1b90613bd1565b6005546040516000916001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600580546001600160a01b0319169055565b600c546040805163ad7a672f60e01b815290516000926001600160a01b03169163ad7a672f9160048083019260209291908290030181865afa1580156111b5573d6000803e3d6000fd5b6005546001600160a01b03163314611c685760405162461bcd60e51b8152600401610b1b90613bd1565b6010805460ff191690556040805160808101825260008082526020820181905291810182905260608101919091526000611ca160025490565b6001600160601b03811680845260208401819052600f805460408087015160608801516001600160c01b03199093168517600160601b9095029490941766ffffffffffffff60c01b1916600160c01b65ffffffffffff9095169490940260ff60f01b191693909317600160f01b91151591909102179055519091507fa4ffae85e880608d5d4365c2b682786545d136145537788e7e0940dff9f0b98c90600090a15050565b6005546001600160a01b03163314611d705760405162461bcd60e51b8152600401610b1b90613bd1565b6001600160a01b038116611db55760405162461bcd60e51b815260206004820152600c60248201526b7a65726f206164647265737360a01b6044820152606401610b1b565b600980546001600160a01b0319166001600160a01b0392909216919091179055565b606060048054610ee390613c3c565b60003381611df4828661232e565b905083811015611e545760405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f77604482015264207a65726f60d81b6064820152608401610b1b565b611160828686840361281d565b600c546040516302a2e74960e61b81526001600160a01b038381166004830152600092169063a8b9d24090602401611970565b600033610f748185856129bb565b600c5460405163dcb95ed960e01b81523360048201526000916001600160a01b03169063dcb95ed9906024016020604051808303816000875af1158015611eed573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611f119190613c06565b90508015611f2857611f24818333612b29565b5050565b60405162461bcd60e51b815260206004820152600a6024820152694e6f207265776172647360b01b6044820152606401610b1b565b600c5460405163fbcbc0f160e01b81526001600160a01b038381166004830152600092839283928392169063fbcbc0f190602401608060405180830381865afa158015611fae573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611fd29190613f52565b93509350935093509193509193565b6005546001600160a01b0316331461200b5760405162461bcd60e51b8152600401610b1b90613bd1565b620186a061201860025490565b612023906001613df8565b61202d9190613e25565b81101561209a5760405162461bcd60e51b815260206004820152603560248201527f5377617020616d6f756e742063616e6e6f74206265206c6f776572207468616e60448201527410181718181892903a37ba30b61039bab838363c9760591b6064820152608401610b1b565b6103e86120a660025490565b6120b1906005613df8565b6120bb9190613e25565b8111156121275760405162461bcd60e51b815260206004820152603460248201527f5377617020616d6f756e742063616e6e6f742062652068696768657220746861604482015273371018171a92903a37ba30b61039bab838363c9760611b6064820152608401610b1b565b601455565b60006305f5e100670de0b6b3a764000061214761dead611b65565b6002546121549190613e39565b61217d7f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e35611b65565b6015547f0000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488d6001600160a01b031663ad5c46486040518163ffffffff1660e01b8152600401602060405180830381865afa1580156121de573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122029190613f91565b6040516370a0823160e01b81526001600160a01b037f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e358116600483015291909116906370a0823190602401602060405180830381865afa15801561226a573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061228e9190613c06565b6122a090670de0b6b3a7640000613df8565b6122aa9190613df8565b6122b49190613e25565b6122be9190613df8565b6122c89190613e25565b6111d99190613e25565b6005546001600160a01b031633146122fc5760405162461bcd60e51b8152600401610b1b90613bd1565b600c5460405163c0f306ef60e01b81526001600160a01b0383811660048301529091169063c0f306ef90602401611236565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b6005546001600160a01b031633146123835760405162461bcd60e51b8152600401610b1b90613bd1565b61238f6012600a613f43565b6103e861239b60025490565b6123a6906005613df8565b6123b09190613e25565b6123ba9190613e25565b816001600160601b031610156123fc5760405162461bcd60e51b8152602060048201526007602482015266546f6f206c6f7760c81b6044820152606401610b1b565b6124086012600a613f43565b61241b906001600160601b038316613df8565b600f80546001600160601b0319166001600160601b039290921691821790556040519081527f6710da7d4acedae09cb83751ae24c150719ef67dcbc1e02049f171d13c6b44e690602001611936565b6005546001600160a01b031633146124945760405162461bcd60e51b8152600401610b1b90613bd1565b6001600160a01b0381166124f95760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610b1b565b6005546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600580546001600160a01b0319166001600160a01b0392909216919091179055565b6005546001600160a01b0316331461257f5760405162461bcd60e51b8152600401610b1b90613bd1565b6001600160a01b0382166125c45760405162461bcd60e51b815260206004820152600c60248201526b5a65726f204164647265737360a01b6044820152606401610b1b565b306001600160a01b0383160361261c5760405162461bcd60e51b815260206004820152601a60248201527f43616e6e6f7420756e6578636c7564656420636f6e74726163740000000000006044820152606401610b1b565b6001600160a01b038216600081815260066020908152604091829020805460ff19168515159081179091558251938452908301527fc4eea32423e96d678d53f47ddd9b7a5103eea02606d7daa81c77038c54dc8edb910161180d565b6001600160a01b0383166126dc5760405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f206164604482015264647265737360d81b6064820152608401610b1b565b6001600160a01b03821661273e5760405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608401610b1b565b6001600160a01b038316600090815260208190526040902054818110156127b65760405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e7420657863656564732062604482015265616c616e636560d01b6064820152608401610b1b565b6001600160a01b03848116600081815260208181526040808320878703905593871680835291849020805487019055925185815290927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef910160405180910390a350505050565b6001600160a01b03831661287f5760405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f206164646044820152637265737360e01b6064820152608401610b1b565b6001600160a01b0382166128e05760405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608401610b1b565b6001600160a01b0383811660008181526001602090815260408083209487168084529482529182902085905590518481527f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925910160405180910390a3505050565b600061294d848461232e565b905060001981146129b557818110156129a85760405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152606401610b1b565b6129b5848484840361281d565b50505050565b6001600160a01b03831660009081526006602052604090205460ff161580156129fd57506001600160a01b03821660009081526006602052604090205460ff16155b15612a2557612a0d838383612c4e565b612a188383836130a1565b612a229082613e39565b90505b612a30838383612678565b600c546001600160a01b031663e30443bc83612a4b81611b65565b6040516001600160e01b031960e085901b1681526001600160a01b0390921660048301526024820152604401600060405180830381600087803b158015612a9157600080fd5b505af1158015612aa5573d6000803e3d6000fd5b5050600c546001600160a01b0316915063e30443bc905084612ac681611b65565b6040516001600160e01b031960e085901b1681526001600160a01b0390921660048301526024820152604401600060405180830381600087803b158015612b0c57600080fd5b505af1158015612b20573d6000803e3d6000fd5b50505050505050565b6040805160028082526060820183526000926020830190803683370190505090507f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc281600081518110612b7e57612b7e613c8c565b60200260200101906001600160a01b031690816001600160a01b0316815250503081600181518110612bb257612bb2613c8c565b6001600160a01b03928316602091820292909201015260405163b6f9de9560e01b81527f0000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488d9091169063b6f9de95908690612c16908790869088904290600401613fae565b6000604051808303818588803b158015612c2f57600080fd5b505af1158015612c43573d6000803e3d6000fd5b505050505050505050565b60105460ff1615612fdf576001600160a01b03821660009081526007602052604081205460ff1690612c7f84611b65565b60408051608081018252600f546001600160601b038082168352600160601b8204166020830152600160c01b810465ffffffffffff1692820192909252600160f01b90910460ff16151560608201529091507f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b03908116908716148015612d0b575082155b15612ee05780516001600160601b0316841115612d545760405162461bcd60e51b815260206004820152600760248201526626b0bc102a3c3760c91b6044820152606401610b1b565b60208101516001600160601b0316612d6c8386613e4c565b1115612da75760405162461bcd60e51b815260206004820152600a60248201526913585e0815d85b1b195d60b21b6044820152606401610b1b565b806060015115612edb57601354612dbf906004613e4c565b4310612e3757600060608201528051600f80546020840151604085015160ff60f01b1965ffffffffffff909116600160c01b021666ffffffffffffff60c01b196001600160601b03928316600160601b026001600160c01b031990941692909516919091179190911792909216919091179055612fdb565b601354612e45906001613e4c565b4311612e7f5760405162461bcd60e51b81526020600482015260096024820152684561726c792062757960b81b6044820152606401610b1b565b806040015165ffffffffffff163a1115612edb5760405162461bcd60e51b815260206004820152601860248201527f4761732070726963652065786365656473206c696d69742e00000000000000006044820152606401610b1b565b612fdb565b7f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b0316856001600160a01b0316148015612f3a57506001600160a01b03861660009081526007602052604090205460ff16155b15612f835780516001600160601b0316841115612edb5760405162461bcd60e51b815260206004820152600760248201526626b0bc102a3c3760c91b6044820152606401610b1b565b82612fdb5760208101516001600160601b0316612fa08386613e4c565b1115612fdb5760405162461bcd60e51b815260206004820152600a60248201526913585e0815d85b1b195d60b21b6044820152606401610b1b565b5050505b60125460ff1615613075577f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b0316826001600160a01b03160361307a576001600160a01b03831660009081526011602052604090205443116130755760405162461bcd60e51b815260206004820152600860248201526720b73a349026a2ab60c11b6044820152606401610b1b565b505050565b506001600160a01b0316600090815260116020526040808220439081905532835291205550565b60006014546130af30611b65565b101580156130ef57507f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b0316846001600160a01b031614155b156130fc576130fc6134e0565b6040805160a08101825260008082526020820181905291810182905260608101829052608081018290527f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b0316856001600160a01b0316036131b957506040805160a081018252600b5465ffffffffffff8082168352600160301b820481166020840152600160601b8204811693830193909352600160901b810483166060830152600160c01b90049091166080820152613248565b7f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b0316866001600160a01b03160361324857506040805160a081018252600a5465ffffffffffff8082168352600160301b820481166020840152600160601b8204811693830193909352600160901b810483166060830152600160c01b900490911660808201525b608081015165ffffffffffff16156134ce576040805160a081018252600d546001600160401b038082168352600160401b820481166020840152600160801b8204811693830193909352600160c01b90049091166060820152600e5460ff161515608080830191909152820151612710906132cb9065ffffffffffff1687613df8565b6132d59190613e25565b9250633b9aca00826080015165ffffffffffff16836040015165ffffffffffff16856133019190613fe3565b61330b919061400e565b613315919061400e565b816020018181516133269190614034565b6001600160401b031690525060808201518251633b9aca009165ffffffffffff90811691613355911686613fe3565b61335f919061400e565b613369919061400e565b81518290613378908390614034565b6001600160401b031690525060808201516020830151633b9aca009165ffffffffffff908116916133aa911686613fe3565b6133b4919061400e565b6133be919061400e565b816040018181516133cf9190614034565b6001600160401b031690525060808201516060830151633b9aca009165ffffffffffff90811691613401911686613fe3565b61340b919061400e565b613415919061400e565b816060018181516134269190614034565b6001600160401b039081169091528251600d80546020860151604087015160608801518616600160c01b026001600160c01b03918716600160801b02919091166001600160801b03928716600160401b026fffffffffffffffffffffffffffffffff199094169590961694909417919091178116939093179190911790556080830151600e805491151560ff199092169190911790556134cc9150889030908616612678565b505b506001600160801b0316949350505050565b60006134eb30611b65565b6040805160a081018252600d546001600160401b03808216808452600160401b8304821660208501819052600160801b84048316958501869052600160c01b90930490911660608401819052600e5460ff161515608085015294955091936000939092909161355a9190614034565b6135649190614034565b61356e9190614034565b6001600160401b03169050821580613584575080155b1561358e57505050565b6014805461359b91613df8565b8311156135b257601480546135af91613df8565b92505b60208201516001600160401b0316156136ae5760008183602001516001600160401b0316856135e19190613df8565b6135eb9190613e25565b9050613618307f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e3583612678565b7f000000000000000000000000e79bac885fc86b861b9ca4866c022c85bb801e356001600160a01b031663fff6cae96040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561367357600080fd5b505af1925050508015613684575060015b5061368f8185613e39565b935082602001516001600160401b0316826136aa9190613e39565b9150505b8215613840576136bd83613876565b604082015147906000906001600160401b03161561374f5760095460408501516001600160a01b039091169084906136fe906001600160401b031685613df8565b6137089190613e25565b604051600081818185875af1925050503d8060008114613744576040519150601f19603f3d011682016040523d82523d6000602084013e613749565b606091505b50909150505b60608401516001600160401b0316156137dc57600c5460608501516001600160a01b0390911690849061378b906001600160401b031685613df8565b6137959190613e25565b604051600081818185875af1925050503d80600081146137d1576040519150601f19603f3d011682016040523d82523d6000602084013e6137d6565b606091505b50909150505b479150811561383d576008546040516001600160a01b03909116908390600081818185875af1925050503d8060008114613832576040519150601f19603f3d011682016040523d82523d6000602084013e613837565b606091505b50909150505b50505b506000602082018190528082526040820181905260608201819052600d5560800151600e805460ff191691151591909117905550565b60408051600280825260608201835260009260208301908036833701905050905030816000815181106138ab576138ab613c8c565b60200260200101906001600160a01b031690816001600160a01b0316815250507f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2816001815181106138ff576138ff613c8c565b6001600160a01b03928316602091820292909201015260405163791ac94760e01b81527f0000000000000000000000007a250d5630b4cf539739df2c5dacb4c659f2488d9091169063791ac94790613964908590600090869030904290600401614054565b600060405180830381600087803b15801561397e57600080fd5b505af1158015613992573d6000803e3d6000fd5b505050505050565b6001600160a01b038116811461168b57600080fd5b6000602082840312156139c157600080fd5b81356139cc8161399a565b9392505050565b600060208083528351808285015260005b81811015613a00578581018301518582016040015282016139e4565b506000604082860101526040601f19601f8301168501019250505092915050565b60008060408385031215613a3457600080fd5b8235613a3f8161399a565b946020939093013593505050565b600060208284031215613a5f57600080fd5b5035919050565b600080600060608486031215613a7b57600080fd5b8335613a868161399a565b92506020840135613a968161399a565b929592945050506040919091013590565b803565ffffffffffff81168114613abd57600080fd5b919050565b60008060008060808587031215613ad857600080fd5b613ae185613aa7565b9350613aef60208601613aa7565b9250613afd60408601613aa7565b9150613b0b60608601613aa7565b905092959194509250565b801515811461168b57600080fd5b600060208284031215613b3657600080fd5b81356139cc81613b16565b60008060408385031215613b5457600080fd5b8235613b5f8161399a565b91506020830135613b6f81613b16565b809150509250929050565b600060208284031215613b8c57600080fd5b81356001600160601b03811681146139cc57600080fd5b60008060408385031215613bb657600080fd5b8235613bc18161399a565b91506020830135613b6f8161399a565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b600060208284031215613c1857600080fd5b5051919050565b600060208284031215613c3157600080fd5b81516139cc81613b16565b600181811c90821680613c5057607f821691505b602082108103613c7057634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052604160045260246000fd5b634e487b7160e01b600052603260045260246000fd5b600081518084526020808501945080840160005b83811015613cdb5781516001600160a01b031687529582019590820190600101613cb6565b509495945050505050565b82815260406020820152600061113f6040830184613ca2565b60006020808385031215613d1257600080fd5b82516001600160401b0380821115613d2957600080fd5b818501915085601f830112613d3d57600080fd5b815181811115613d4f57613d4f613c76565b8060051b604051601f19603f83011681018181108582111715613d7457613d74613c76565b604052918252848201925083810185019188831115613d9257600080fd5b938501935b82851015613db057845184529385019392850192613d97565b98975050505050505050565b634e487b7160e01b600052601160045260246000fd5b65ffffffffffff818116838216019080821115613df157613df1613dbc565b5092915050565b8082028115828204841417610f7a57610f7a613dbc565b634e487b7160e01b600052601260045260246000fd5b600082613e3457613e34613e0f565b500490565b81810381811115610f7a57610f7a613dbc565b80820180821115610f7a57610f7a613dbc565b600181815b80851115613e9a578160001904821115613e8057613e80613dbc565b80851615613e8d57918102915b93841c9390800290613e64565b509250929050565b600082613eb157506001610f7a565b81613ebe57506000610f7a565b8160018114613ed45760028114613ede57613efa565b6001915050610f7a565b60ff841115613eef57613eef613dbc565b50506001821b610f7a565b5060208310610133831016604e8410600b8410161715613f1d575081810a610f7a565b613f278383613e5f565b8060001904821115613f3b57613f3b613dbc565b029392505050565b60006139cc60ff841683613ea2565b60008060008060808587031215613f6857600080fd5b8451613f738161399a565b60208601516040870151606090970151919890975090945092505050565b600060208284031215613fa357600080fd5b81516139cc8161399a565b848152608060208201526000613fc76080830186613ca2565b6001600160a01b03949094166040830152506060015292915050565b6001600160801b0381811683821602808216919082811461400657614006613dbc565b505092915050565b60006001600160801b038084168061402857614028613e0f565b92169190910492915050565b6001600160401b03818116838216019080821115613df157613df1613dbc565b85815284602082015260a06040820152600061407360a0830186613ca2565b6001600160a01b039490941660608301525060800152939250505056fea264697066735822122047b4ee7f349e2e938e0863beb4de9a3f10cdc37dc595cb870c400a46cd0b72c964736f6c63430008130033
Verified Source Code Full Match
Compiler: v0.8.19+commit.7dd6d404
EVM: paris
Optimization: Yes (200 runs)
Token.sol 1371 lines
pragma solidity 0.8.19;
// SPDX-License-Identifier: MIT
import "./StructLibrary.sol";
import "./Context.sol";
import "./IERC20.sol";
import "./DEXInterfaces.sol";
interface IERC20Metadata is IERC20{
/**
* @dev Returns the name of the token.
*/
function name() external view returns (string memory);
/**
* @dev Returns the symbol of the token.
*/
function symbol() external view returns (string memory);
/**
* @dev Returns the decimals places of the token.
*/
function decimals() external view returns (uint8);
}
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;
/**
* @dev Sets the values for {name} and {symbol}.
*
* All two of these values are immutable: they can only be set once during
* construction.
*/
constructor(string memory name_, string memory symbol_) {
_name = name_;
_symbol = symbol_;
}
/**
* @dev Returns the name of the token.
*/
function name() public view virtual override returns (string memory) {
return _name;
}
/**
* @dev Returns the symbol of the token, usually a shorter version of the
* name.
*/
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
/**
* @dev Returns the number of decimals used to get its user representation.
* For example, if `decimals` equals `2`, a balance of `505` tokens should
* be displayed to a user as `5.05` (`505 / 10 ** 2`).
*
* Tokens usually opt for a value of 18, imitating the relationship between
* Ether and Wei. This is the default value returned by this function, unless
* it's overridden.
*
* NOTE: This information is only used for _display_ purposes: it in
* no way affects any of the arithmetic of the contract, including
* {IERC20-balanceOf} and {IERC20-transfer}.
*/
function decimals() public view virtual override returns (uint8) {
return 18;
}
/**
* @dev See {IERC20-totalSupply}.
*/
function totalSupply() public view virtual override returns (uint256) {
return _totalSupply;
}
/**
* @dev See {IERC20-balanceOf}.
*/
function balanceOf(address account) public view virtual override returns (uint256) {
return _balances[account];
}
/**
* @dev See {IERC20-transfer}.
*
* Requirements:
*
* - `to` cannot be the zero address.
* - the caller must have a balance of at least `amount`.
*/
function transfer(address to, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_transfer(owner, to, amount);
return true;
}
/**
* @dev See {IERC20-allowance}.
*/
function allowance(address owner, address spender) public view virtual override returns (uint256) {
return _allowances[owner][spender];
}
/**
* @dev See {IERC20-approve}.
*
* NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
* `transferFrom`. This is semantically equivalent to an infinite approval.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/
function approve(address spender, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_approve(owner, spender, amount);
return true;
}
/**
* @dev See {IERC20-transferFrom}.
*
* Emits an {Approval} event indicating the updated allowance. This is not
* required by the EIP. See the note at the beginning of {ERC20}.
*
* NOTE: Does not update the allowance if the current allowance
* is the maximum `uint256`.
*
* Requirements:
*
* - `from` and `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
* - the caller must have allowance for ``from``'s tokens of at least
* `amount`.
*/
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;
}
/**
* @dev Atomically increases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/
function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
address owner = _msgSender();
_approve(owner, spender, allowance(owner, spender) + addedValue);
return true;
}
/**
* @dev Atomically decreases the allowance granted to `spender` by the caller.
*
* This is an alternative to {approve} that can be used as a mitigation for
* problems described in {IERC20-approve}.
*
* Emits an {Approval} event indicating the updated allowance.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `spender` must have allowance for the caller of at least
* `subtractedValue`.
*/
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;
}
/**
* @dev Moves `amount` of tokens from `from` to `to`.
*
* This internal function is equivalent to {transfer}, and can be used to
* e.g. implement automatic token fees, slashing mechanisms, etc.
*
* Emits a {Transfer} event.
*
* Requirements:
*
* - `from` cannot be the zero address.
* - `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
*/
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");
uint256 fromBalance = _balances[from];
require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
unchecked {
_balances[from] = fromBalance - amount;
// Overflow not possible: the sum of all balances is capped by totalSupply, and the sum is preserved by
// decrementing then incrementing.
_balances[to] += amount;
}
emit Transfer(from, to, amount);
}
/** @dev Creates `amount` tokens and assigns them to `account`, increasing
* the total supply.
*
* Emits a {Transfer} event with `from` set to the zero address.
*
* Requirements:
*
* - `account` cannot be the zero address.
*/
function _mint(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: mint to the zero address");
_totalSupply += amount;
unchecked {
// Overflow not possible: balance + amount is at most totalSupply + amount, which is checked above.
_balances[account] += amount;
}
emit Transfer(address(0), account, amount);
}
/**
* @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
*
* This internal function is equivalent to `approve`, and can be used to
* e.g. set automatic allowances for certain subsystems, etc.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `owner` cannot be the zero address.
* - `spender` cannot be the zero address.
*/
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);
}
/**
* @dev Updates `owner` s allowance for `spender` based on spent `amount`.
*
* Does not update the allowance amount in case of infinite allowance.
* Revert if not enough allowance is available.
*
* Might emit an {Approval} event.
*/
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);
}
}
}
}
contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor () {
address msgSender = _msgSender();
_owner = msgSender;
emit OwnershipTransferred(address(0), msgSender);
}
function owner() public view returns (address) {
return _owner;
}
modifier onlyOwner() {
require(_owner == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() external virtual onlyOwner {
emit OwnershipTransferred(_owner, address(0));
_owner = address(0);
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
emit OwnershipTransferred(_owner, newOwner);
_owner = newOwner;
}
}
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);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but also transferring `value` wei to `target`.
*
* Requirements:
*
* - the calling contract must have an ETH balance of at least `value`.
* - the called Solidity function must be `payable`.
*
* _Available since v3.1._
*/
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");
}
/**
* @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
* with `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
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);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
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);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
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);
}
/**
* @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
* the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
*
* _Available since v4.8._
*/
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
// only check isContract if the call was successful and the return data is empty
// otherwise we already know that it was a contract
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
/**
* @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
* revert reason or using the provided one.
*
* _Available since v4.3._
*/
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 {
// Look for revert reason and bubble it up if present
if (returndata.length > 0) {
// The easiest way to bubble the revert reason is using memory via assembly
/// @solidity memory-safe-assembly
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
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 _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) {
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
function safeApprove(IERC20 token, address spender, uint256 value) internal {
// safeApprove should only be called when setting an initial allowance,
// or when resetting it to zero. To increase and decrease it, use
// 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
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));
}
}
library SafeMath {
function add(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a + b;
require(c >= a, 'SafeMath: addition overflow');
return c;
}
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
return sub(a, b, 'SafeMath: subtraction overflow');
}
function sub(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
require(b <= a, errorMessage);
uint256 c = a - b;
return c;
}
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
// Gas optimization: this is cheaper than requiring 'a' not being zero, but the
// benefit is lost if 'b' is also tested.
// See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
if (a == 0) {
return 0;
}
uint256 c = a * b;
require(c / a == b, 'SafeMath: multiplication overflow');
return c;
}
function div(uint256 a, uint256 b) internal pure returns (uint256) {
return div(a, b, 'SafeMath: division by zero');
}
function div(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
require(b > 0, errorMessage);
uint256 c = a / b;
// assert(a == b * c + a % b); // There is no case in which this doesn't hold
return c;
}
function mod(uint256 a, uint256 b) internal pure returns (uint256) {
return mod(a, b, 'SafeMath: modulo by zero');
}
function mod(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
require(b != 0, errorMessage);
return a % b;
}
function min(uint256 x, uint256 y) internal pure returns (uint256 z) {
z = x < y ? x : y;
}
function sqrt(uint256 y) internal pure returns (uint256 z) {
if (y > 3) {
z = y;
uint256 x = y / 2 + 1;
while (x < z) {
z = x;
x = (y / x + x) / 2;
}
} else if (y != 0) {
z = 1;
}
}
}
library SafeMathInt {
int256 private constant MIN_INT256 = int256(1) << 255;
int256 private constant MAX_INT256 = ~(int256(1) << 255);
/**
* @dev Multiplies two int256 variables and fails on overflow.
*/
function mul(int256 a, int256 b) internal pure returns (int256) {
int256 c = a * b;
// Detect overflow when multiplying MIN_INT256 with -1
require(c != MIN_INT256 || (a & MIN_INT256) != (b & MIN_INT256));
require((b == 0) || (c / b == a));
return c;
}
/**
* @dev Division of two int256 variables and fails on overflow.
*/
function div(int256 a, int256 b) internal pure returns (int256) {
// Prevent overflow when dividing MIN_INT256 by -1
require(b != -1 || a != MIN_INT256);
// Solidity already throws when dividing by 0.
return a / b;
}
/**
* @dev Subtracts two int256 variables and fails on overflow.
*/
function sub(int256 a, int256 b) internal pure returns (int256) {
int256 c = a - b;
require((b >= 0 && c <= a) || (b < 0 && c > a));
return c;
}
/**
* @dev Adds two int256 variables and fails on overflow.
*/
function add(int256 a, int256 b) internal pure returns (int256) {
int256 c = a + b;
require((b >= 0 && c >= a) || (b < 0 && c < a));
return c;
}
/**
* @dev Converts to absolute value, and fails on overflow.
*/
function abs(int256 a) internal pure returns (int256) {
require(a != MIN_INT256);
return a < 0 ? -a : a;
}
function toUint256Safe(int256 a) internal pure returns (uint256) {
require(a >= 0);
return uint256(a);
}
}
library SafeMathUint {
function toInt256Safe(uint256 a) internal pure returns (int256) {
int256 b = int256(a);
require(b >= 0);
return b;
}
}
interface DividendPayingContractOptionalInterface {
function withdrawableDividendOf(address _owner) external view returns(uint256);
function withdrawnDividendOf(address _owner) external view returns(uint256);
function accumulativeDividendOf(address _owner) external view returns(uint256);
}
interface DividendPayingContractInterface {
function dividendOf(address _owner) external view returns(uint256);
function distributeDividends() external payable;
function withdrawDividend() external;
event DividendsDistributed(
address indexed from,
uint256 weiAmount
);
event DividendWithdrawn(
address indexed to,
uint256 weiAmount
);
}
contract DividendPayingContract is DividendPayingContractInterface, DividendPayingContractOptionalInterface, Ownable {
using SafeMath for uint256;
using SafeMathUint for uint256;
using SafeMathInt for int256;
uint256 constant internal magnitude = 2**128;
uint256 internal magnifiedDividendPerShare;
mapping(address => int256) internal magnifiedDividendCorrections;
mapping(address => uint256) internal withdrawnDividends;
mapping (address => uint256) public holderBalance;
uint256 public totalBalance;
uint256 public totalDividendsDistributed;
receive() external payable {
distributeDividends();
}
function distributeDividends() public override payable {
if(totalBalance > 0 && msg.value > 0){
magnifiedDividendPerShare = magnifiedDividendPerShare.add(
(msg.value).mul(magnitude) / totalBalance
);
emit DividendsDistributed(msg.sender, msg.value);
totalDividendsDistributed = totalDividendsDistributed.add(msg.value);
}
}
function withdrawDividend() external virtual override {
_withdrawDividendOfUser(payable(msg.sender));
}
function _withdrawDividendOfUser(address payable user) internal returns (uint256) {
uint256 _withdrawableDividend = withdrawableDividendOf(user);
if (_withdrawableDividend > 0) {
withdrawnDividends[user] = withdrawnDividends[user].add(_withdrawableDividend);
emit DividendWithdrawn(user, _withdrawableDividend);
(bool success,) = user.call{value: _withdrawableDividend}("");
if(!success) {
withdrawnDividends[user] = withdrawnDividends[user].sub(_withdrawableDividend);
return 0;
}
return _withdrawableDividend;
}
return 0;
}
function withdrawDividendOfUserForCompound(address payable user) external onlyOwner returns (uint256 _withdrawableDividend) {
_withdrawableDividend = withdrawableDividendOf(user);
if (_withdrawableDividend > 0) {
withdrawnDividends[user] = withdrawnDividends[user] + _withdrawableDividend;
emit DividendWithdrawn(user, _withdrawableDividend);
}
(bool success,) = owner().call{value: _withdrawableDividend}("");
if(!success) {
withdrawnDividends[user] = withdrawnDividends[user].sub(_withdrawableDividend);
return 0;
}
}
function dividendOf(address _owner) external view override returns(uint256) {
return withdrawableDividendOf(_owner);
}
function withdrawableDividendOf(address _owner) public view override returns(uint256) {
return accumulativeDividendOf(_owner).sub(withdrawnDividends[_owner]);
}
function withdrawnDividendOf(address _owner) external view override returns(uint256) {
return withdrawnDividends[_owner];
}
function accumulativeDividendOf(address _owner) public view override returns(uint256) {
return magnifiedDividendPerShare.mul(holderBalance[_owner]).toInt256Safe()
.add(magnifiedDividendCorrections[_owner]).toUint256Safe() / magnitude;
}
function _increase(address account, uint256 value) internal {
magnifiedDividendCorrections[account] = magnifiedDividendCorrections[account]
.sub( (magnifiedDividendPerShare.mul(value)).toInt256Safe() );
}
function _reduce(address account, uint256 value) internal {
magnifiedDividendCorrections[account] = magnifiedDividendCorrections[account]
.add( (magnifiedDividendPerShare.mul(value)).toInt256Safe() );
}
function _setBalance(address account, uint256 newBalance) internal {
uint256 currentBalance = holderBalance[account];
holderBalance[account] = newBalance;
if(newBalance > currentBalance) {
uint256 increaseAmount = newBalance.sub(currentBalance);
_increase(account, increaseAmount);
totalBalance += increaseAmount;
} else if(newBalance < currentBalance) {
uint256 reduceAmount = currentBalance.sub(newBalance);
_reduce(account, reduceAmount);
totalBalance -= reduceAmount;
}
}
}
contract RevShare is DividendPayingContract {
event Claim(address indexed account, uint256 amount, bool indexed automatic);
mapping (address => bool) public excludedFromDividends;
constructor() {}
function getAccount(address _account)
public view returns (
address account,
uint256 withdrawableDividends,
uint256 totalDividends,
uint256 balance) {
account = _account;
withdrawableDividends = withdrawableDividendOf(account);
totalDividends = accumulativeDividendOf(account);
balance = holderBalance[account];
}
function setBalance(address payable account, uint256 newBalance) external onlyOwner {
if(excludedFromDividends[account]) {
return;
}
_setBalance(account, newBalance);
processAccount(account, true);
}
function processAccount(address payable account, bool automatic) public onlyOwner returns (bool) {
uint256 amount = _withdrawDividendOfUser(account);
if(amount > 0) {
emit Claim(account, amount, automatic);
return true;
}
return false;
}
function getTotalDividendsDistributed() external view returns (uint256) {
return totalDividendsDistributed;
}
function dividendTokenBalanceOf(address account) public view returns (uint256) {
return holderBalance[account];
}
function getNumberOfDividends() external view returns(uint256) {
return totalBalance;
}
function excludeFromDividends(address account) external onlyOwner {
excludedFromDividends[account] = true;
_setBalance(account, 0);
}
function includeInDividends(address account) external onlyOwner {
require(excludedFromDividends[account]);
excludedFromDividends[account] = false;
_setBalance(account, IERC20(owner()).balanceOf(account)); // sets balance back to token balance
}
}
interface IWETH {
function deposit() external payable;
function transfer(address to, uint value) external returns (bool);
function withdraw(uint) external;
}
interface IPriceFeed {
function latestAnswer() external returns (int256);
}
contract ASG is ERC20, Ownable {
mapping (address => bool) public excludedFromFees;
mapping (address => bool) public exFromLimits;
address public marketingReceiver;
address public devReceiver;
Tax public buyTax;
Tax public sellTax;
RevShare public revShare;
TokenDistributionForTax public tokensForTax;
TxLimits public transactionLimits;
bool public transactionLimitsActive = true;
mapping(address => uint256) private _holderLastTransferBlock; // MEV protection
bool private antiSandwichEnabled = true;
uint256 private launchBlock;
uint256 private swapTokensAtAmount;
address public immutable pair;
IDexRouter public immutable dexRouter;
IWETH private immutable WETH;
IPriceFeed internal immutable priceFeed;
uint256 private latestEthPrice = 0;
uint64 public constant FEE_DIVISOR = 10000;
// structs
struct TxLimits {
uint96 transactionLimit;
uint96 walletLimit;
uint48 gasLimit;
bool gasLimitActive;
}
struct Tax {
uint48 mktingTax;
uint48 devTax;
uint48 lpTax;
uint48 revShareTax;
uint48 totalTax;
}
struct TokenDistributionForTax {
uint64 tokensForMkting;
uint64 tokensForLP;
uint64 tokensForDev;
uint64 tokensForRevShare;
bool gasSaver;
}
// events
event UpdatedTransactionLimit(uint newMax);
event UpdatedWalletLimit(uint newMax);
event SetExcludedFromFees(address _address, bool _isExcluded);
event SetExcludedFromLimits(address _address, bool _isExcluded);
event RemovedLimits();
event BuyTaxUpdated(uint newAmount);
event SellTaxUpdated(uint newAmount);
// constructor
constructor(StructLibrary.Params memory params)
ERC20(params.name, params.symbol)
{
_mint(address(this), params.totalSupply * 1e18 * params.liquidityPercentage / 10000);
uint256 leftoverTokens = (params.totalSupply * 1e18) - balanceOf(address(this));
if(leftoverTokens > 0){
_mint(msg.sender, leftoverTokens);
}
address _v2Router;
address _priceFeed;
if(block.chainid == 1){
_v2Router = 0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D; // ETH: Uniswap V2
_priceFeed = 0x5f4eC3Df9cbd43714FE2740f5E3616155c5b8419;
} else if(block.chainid == 5){
_v2Router = 0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D; // Goerli ETH: Uniswap V2
_priceFeed = 0xD4a33860578De61DBAbDc8BFdb98FD742fA7028e;
} else if(block.chainid == 56){
_v2Router = 0x10ED43C718714eb63d5aA57B78B54704E256024E; // BNB Chain: PCS V2
_priceFeed = 0x0567F2323251f0Aab15c8dFb1967E4e8A7D42aeE;
} else if(block.chainid == 97){
_v2Router = 0xD99D1c33F9fC3444f8101754aBC46c52416550D1; // BNB Chain Testnet: PCS V2
_priceFeed = 0x2514895c72f50D8bd4B4F9b1110F0D6bD2c97526;
} else if(block.chainid == 42161){
_v2Router = 0x1b02dA8Cb0d097eB8D57A175b88c7D8b47997506; // Arbitrum: SushiSwap
_priceFeed = 0x639Fe6ab55C921f74e7fac1ee960C0B6293ba612;
} else {
revert("Chain not configured");
}
priceFeed = IPriceFeed(_priceFeed);
require(priceFeed.latestAnswer() > 0, "wrong price feed");
revShare = new RevShare();
dexRouter = IDexRouter(_v2Router);
transactionLimits.transactionLimit = uint96(totalSupply() * params.transactionLimit / 10000);
transactionLimits.walletLimit = uint96(totalSupply() * params.walletLimit / 10000);
transactionLimits.gasLimit = params.gasLimit * 1 gwei;
swapTokensAtAmount = totalSupply() * 25 / 100000;
marketingReceiver = params.marketingReceiver; // update
devReceiver = params.devReceiver; // update
buyTax.mktingTax = params.marketingTaxBuy;
buyTax.lpTax = params.lpTaxBuy;
buyTax.devTax = params.devTaxBuy;
buyTax.revShareTax = params.revShareTaxBuy;
buyTax.totalTax = buyTax.mktingTax + buyTax.lpTax + buyTax.devTax + buyTax.revShareTax;
sellTax.mktingTax = params.marketingTaxSell;
sellTax.lpTax = params.lpTaxSell;
sellTax.devTax = params.devTaxSell;
sellTax.revShareTax = params.revShareTaxSell;
sellTax.totalTax = sellTax.mktingTax + sellTax.lpTax + sellTax.devTax + buyTax.revShareTax;
tokensForTax.gasSaver = true;
WETH = IWETH(dexRouter.WETH());
pair = IDexFactory(dexRouter.factory()).createPair(address(this), address(WETH));
exFromLimits[pair] = true;
exFromLimits[msg.sender] = true;
exFromLimits[address(this)] = true;
exFromLimits[_v2Router] = true;
excludedFromFees[msg.sender] = true;
excludedFromFees[address(this)] = true;
excludedFromFees[address(0xdead)] = true;
excludedFromFees[_v2Router] = true;
revShare.excludeFromDividends(address(this));
revShare.excludeFromDividends(address(pair));
revShare.excludeFromDividends(address(0xdead));
_approve(address(this), address(dexRouter), type(uint256).max);
}
function _transfer(
address from,
address to,
uint256 amount
) internal virtual override {
if(!excludedFromFees[from] && !excludedFromFees[to]){
handleLimits(from, to, amount);
amount -= handleFee(from, to, amount);
}
super._transfer(from,to,amount);
revShare.setBalance(payable(to), balanceOf(to));
revShare.setBalance(payable(from), balanceOf(from));
}
function handleLimits(address from, address to, uint256 amount) internal {
if(transactionLimitsActive){
bool exFromLimitsTo = exFromLimits[to];
uint256 balanceOfTo = balanceOf(to);
TxLimits memory _transactionLimits = transactionLimits;
// buy
if (from == pair && !exFromLimitsTo) {
require(amount <= _transactionLimits.transactionLimit, "Max Txn");
require(amount + balanceOfTo <= _transactionLimits.walletLimit, "Max Wallet");
if(_transactionLimits.gasLimitActive){
if(block.number >= launchBlock + 4){
_transactionLimits.gasLimitActive = false;
transactionLimits = _transactionLimits;
} else {
if(block.number <= launchBlock + 1){
revert("Early buy");
}
require(tx.gasprice <= _transactionLimits.gasLimit, "Gas price exceeds limit.");
}
}
}
// sell
else if (to == pair && !exFromLimits[from]) {
require(amount <= _transactionLimits.transactionLimit, "Max Txn");
}
else if(!exFromLimitsTo) {
require(amount + balanceOfTo <= _transactionLimits.walletLimit, "Max Wallet");
}
}
if (antiSandwichEnabled){
if(to == pair){
require(_holderLastTransferBlock[from] < block.number, "Anti MEV");
} else {
_holderLastTransferBlock[to] = block.number;
_holderLastTransferBlock[tx.origin] = block.number;
}
}
}
function handleFee(address from, address to, uint256 amount) internal returns (uint256){
if(balanceOf(address(this)) >= swapTokensAtAmount && from != pair) {
convertTaxes();
}
uint128 tax = 0;
Tax memory taxes;
if (to == pair){
taxes = sellTax;
} else if (from == pair){
taxes = buyTax;
}
if(taxes.totalTax > 0){
TokenDistributionForTax memory tokensForTaxUpdate = tokensForTax;
tax = uint128(amount * taxes.totalTax / FEE_DIVISOR);
tokensForTaxUpdate.tokensForLP += uint64(tax * taxes.lpTax / taxes.totalTax / 1e9);
tokensForTaxUpdate.tokensForMkting += uint64(tax * taxes.mktingTax / taxes.totalTax / 1e9);
tokensForTaxUpdate.tokensForDev += uint64(tax * taxes.devTax / taxes.totalTax / 1e9);
tokensForTaxUpdate.tokensForRevShare += uint64(tax * taxes.revShareTax / taxes.totalTax / 1e9);
tokensForTax = tokensForTaxUpdate;
super._transfer(from, address(this), tax);
}
return tax;
}
function swapTokensForETH(uint256 tokenAmount) private {
address[] memory path = new address[](2);
path[0] = address(this);
path[1] = address(WETH);
dexRouter.swapExactTokensForETHSupportingFeeOnTransferTokens(
tokenAmount,
0,
path,
address(this),
block.timestamp
);
}
function convertTaxes() private {
uint256 contractBalance = balanceOf(address(this));
TokenDistributionForTax memory tokensForTaxMem = tokensForTax;
uint256 totalTokensToSwap = tokensForTaxMem.tokensForLP + tokensForTaxMem.tokensForMkting + tokensForTaxMem.tokensForDev + tokensForTaxMem.tokensForRevShare;
if(contractBalance == 0 || totalTokensToSwap == 0) {return;}
if(contractBalance > swapTokensAtAmount * 20){
contractBalance = swapTokensAtAmount * 20;
}
if(tokensForTaxMem.tokensForLP > 0){
uint256 lpTokens = contractBalance * tokensForTaxMem.tokensForLP / totalTokensToSwap;
super._transfer(address(this), pair, lpTokens);
try ILpPair(pair).sync(){} catch {}
contractBalance -= lpTokens;
totalTokensToSwap -= tokensForTaxMem.tokensForLP;
}
if(contractBalance > 0){
swapTokensForETH(contractBalance);
uint256 ethBalance = address(this).balance;
bool success;
if(tokensForTaxMem.tokensForDev > 0){
(success,) = devReceiver.call{value: ethBalance * tokensForTaxMem.tokensForDev / totalTokensToSwap}("");
}
if(tokensForTaxMem.tokensForRevShare > 0){
(success,) = address(revShare).call{value: ethBalance * tokensForTaxMem.tokensForRevShare/ totalTokensToSwap}("");
}
ethBalance = address(this).balance;
if(ethBalance > 0){
(success,) = marketingReceiver.call{value: ethBalance}("");
}
}
tokensForTaxMem.tokensForLP = 0;
tokensForTaxMem.tokensForMkting = 0;
tokensForTaxMem.tokensForDev = 0;
tokensForTaxMem.tokensForRevShare = 0;
tokensForTax = tokensForTaxMem;
}
// owner functions
function changeExcludedFromFees(address _address, bool _isExcluded) external onlyOwner {
require(_address != address(0), "Zero Address");
require(_address != address(this), "Cannot unexcluded contract");
excludedFromFees[_address] = _isExcluded;
emit SetExcludedFromFees(_address, _isExcluded);
}
function changeExcludedFromLimits(address _address, bool _isExcluded) external onlyOwner {
require(_address != address(0), "Zero Address");
if(!_isExcluded){
require(_address != pair, "Cannot remove pair");
}
exFromLimits[_address] = _isExcluded;
emit SetExcludedFromLimits(_address, _isExcluded);
}
function updateTransactionLimit(uint96 newNumInTokens) external onlyOwner {
require(newNumInTokens >= (totalSupply() * 5 / 1000)/(10**decimals()), "Too low");
transactionLimits.transactionLimit = uint96(newNumInTokens * (10**decimals()));
emit UpdatedTransactionLimit(transactionLimits.transactionLimit);
}
function updateWalletLimit(uint96 newNumInTokens) external onlyOwner {
require(newNumInTokens >= (totalSupply() * 5 / 1000)/(10**decimals()), "Too low");
transactionLimits.walletLimit = uint96(newNumInTokens * (10**decimals()));
emit UpdatedWalletLimit(transactionLimits.walletLimit);
}
function changeSwapTokensAmount(uint256 newAmount) external onlyOwner {
require(newAmount >= (totalSupply() * 1) / 100000, "Swap amount cannot be lower than 0.001% total supply.");
require(newAmount <= (totalSupply() * 5) / 1000, "Swap amount cannot be higher than 0.5% total supply.");
swapTokensAtAmount = newAmount;
}
function updateSellTax(uint48 _mktingTax, uint48 _lpTax, uint48 _devTax, uint48 _revShareTax) external onlyOwner {
Tax memory taxes;
taxes.mktingTax = _mktingTax;
taxes.lpTax = _lpTax;
taxes.devTax = _devTax;
taxes.revShareTax = _revShareTax;
taxes.totalTax = _mktingTax + _lpTax + _devTax + _revShareTax;
require(taxes.totalTax <= 1000, "Keep tax below 10%");
emit SellTaxUpdated(taxes.totalTax);
sellTax = taxes;
}
function setBuyTax(uint48 _mktingTax, uint48 _lpTax, uint48 _devTax, uint48 _revShareTax) external onlyOwner {
Tax memory taxes;
taxes.mktingTax = _mktingTax;
taxes.lpTax = _lpTax;
taxes.devTax = _devTax;
taxes.revShareTax = _revShareTax;
taxes.totalTax = _mktingTax + _lpTax + _devTax + _revShareTax;
require(taxes.totalTax <= 1000, "Keep tax below 10%");
emit BuyTaxUpdated(taxes.totalTax);
buyTax = taxes;
}
function removeLimits() external onlyOwner {
transactionLimitsActive = false;
TxLimits memory _transactionLimits;
uint256 supply = totalSupply();
_transactionLimits.transactionLimit = uint96(supply);
_transactionLimits.walletLimit = uint96(supply);
transactionLimits = _transactionLimits;
emit RemovedLimits();
}
function updateAntiMevEnabled(bool _enabled) external onlyOwner {
antiSandwichEnabled = _enabled;
}
function changeMarketingAddress(address _address) external onlyOwner {
require(_address != address(0), "zero address");
marketingReceiver = _address;
}
function changeDevAddress(address _address) external onlyOwner {
require(_address != address(0), "zero address");
devReceiver = _address;
}
receive() payable external {}
// dividend functions
function claim() external {
revShare.processAccount(payable(msg.sender), false);
}
function getTotalDividendsDistributed() external view returns (uint256) {
return revShare.totalDividendsDistributed();
}
function withdrawableDividendOf(address account) public view returns(uint256) {
return revShare.withdrawableDividendOf(account);
}
function dividendTokenBalanceOf(address account) public view returns (uint256) {
return revShare.holderBalance(account);
}
function getAccountDividendsInfo(address account)
external view returns (
address,
uint256,
uint256,
uint256) {
return revShare.getAccount(account);
}
function getNumberOfDividends() external view returns(uint256) {
return revShare.totalBalance();
}
function excludeFromDividends(address _wallet) external onlyOwner {
revShare.excludeFromDividends(_wallet);
}
function includeInDividends(address _wallet) external onlyOwner {
revShare.includeInDividends(_wallet);
}
function compound(uint256 minOutput) external {
uint256 amountEthForCompound = revShare.withdrawDividendOfUserForCompound(payable(msg.sender));
if(amountEthForCompound > 0){
buyBackTokens(amountEthForCompound, minOutput, msg.sender);
} else {
revert("No rewards");
}
}
function buyBackTokens(uint256 ethAmountInWei, uint256 minOut, address to) internal {
// generate the uniswap pair path of weth -> eth
address[] memory path = new address[](2);
path[0] = address(WETH);
path[1] = address(this);
// make the swap
dexRouter.swapExactETHForTokensSupportingFeeOnTransferTokens{value: ethAmountInWei}(
minOut,
path,
address(to),
block.timestamp
);
}
// helper views
function getCompoundOutputByEthAddress(uint256 rewardAmount) external view returns(uint256) {
if(rewardAmount == 0){
return 0;
}
address[] memory path = new address[](2);
path[0] = address(WETH);
path[1] = address(this);
uint256[] memory amounts = dexRouter.getAmountsOut(rewardAmount, path);
return amounts[1] - (amounts[1] * (buyTax.totalTax + 50) / FEE_DIVISOR);
}
function getCompoundOutputByAddress(address wallet) external view returns(uint256) {
uint256 rewardAmount = withdrawableDividendOf(wallet);
if(rewardAmount == 0){
return 0;
}
address[] memory path = new address[](2);
path[0] = address(WETH);
path[1] = address(this);
uint256[] memory amounts = dexRouter.getAmountsOut(rewardAmount, path);
return amounts[1] - (amounts[1] * (buyTax.totalTax + 50) / FEE_DIVISOR);
}
function addLp(address _to) external onlyOwner payable {
require(address(this).balance > 0 && balanceOf(address(this)) > 0);
WETH.deposit{value: address(this).balance}();
super._transfer(address(this), address(pair), balanceOf(address(this)));
IERC20(address(WETH)).transfer(address(pair), IERC20(address(WETH)).balanceOf(address(this)));
ILpPair(pair).mint(_to);
launchBlock = block.number;
transactionLimits.gasLimitActive = true;
latestEthPrice = uint256(priceFeed.latestAnswer());
}
function getMcap() public view returns (uint256){
return (IERC20(dexRouter.WETH()).balanceOf(address(pair)) * 1e18 * latestEthPrice / balanceOf(address(pair)) * (totalSupply()-balanceOf(address(0xdead))) / 1e18 / 1e8);
}
}
IERC20.sol 78 lines
pragma solidity 0.8.19;
// SPDX-License-Identifier: MIT
interface IERC20 {
/**
* @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 `recipient`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address recipient, 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 `sender` to `recipient` 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 sender,
address recipient,
uint256 amount
) external returns (bool);
/**
* @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 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 value);
}
Context.sol 14 lines
pragma solidity 0.8.19;
// SPDX-License-Identifier: MIT
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
return msg.data;
}
}
DEXInterfaces.sol 27 lines
pragma solidity 0.8.19;
// SPDX-License-Identifier: MIT
interface ILpPair {
function sync() external;
function mint(address to) external returns (uint liquidity);
}
interface IDexRouter {
function factory() external pure returns (address);
function WETH() external pure returns (address);
function swapExactTokensForETHSupportingFeeOnTransferTokens(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline) external;
function swapExactETHForTokensSupportingFeeOnTransferTokens(
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external payable;
function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts);
function getAmountsIn(uint amountOut, address[] calldata path) external view returns (uint[] memory amounts);
}
interface IDexFactory {
function createPair(address tokenA, address tokenB) external returns (address pair);
}
StructLibrary.sol 25 lines
pragma solidity 0.8.19;
// SPDX-License-Identifier: MIT
library StructLibrary {
struct Params {
string name;
string symbol;
uint112 totalSupply;
uint48 liquidityPercentage;
address marketingReceiver;
address devReceiver;
uint48 marketingTaxBuy;
uint48 devTaxBuy;
uint48 lpTaxBuy;
uint48 revShareTaxBuy;
uint48 marketingTaxSell;
uint48 devTaxSell;
uint48 lpTaxSell;
uint48 revShareTaxSell;
uint128 transactionLimit;
uint128 walletLimit;
uint48 gasLimit;
}
}
Read Contract
FEE_DIVISOR 0x9e93ad8e → uint64
allowance 0xdd62ed3e → uint256
balanceOf 0x70a08231 → uint256
buyTax 0x4f7041a5 → uint48, uint48, uint48, uint48, uint48
decimals 0x313ce567 → uint8
devReceiver 0x6f9a880e → address
dexRouter 0x0758d924 → address
dividendTokenBalanceOf 0x6843cd84 → uint256
exFromLimits 0xdb408e1a → bool
excludedFromFees 0xdbe66ca0 → bool
getAccountDividendsInfo 0xad56c13c → address, uint256, uint256, uint256
getCompoundOutputByAddress 0x36496bf7 → uint256
getCompoundOutputByEthAddress 0x1af71da6 → uint256
getMcap 0xc077ab4b → uint256
getNumberOfDividends 0x71778e7d → uint256
getTotalDividendsDistributed 0x30bb4cff → uint256
marketingReceiver 0xa5949bcf → address
name 0x06fdde03 → string
owner 0x8da5cb5b → address
pair 0xa8aa1b31 → address
revShare 0x1b03ddc1 → address
sellTax 0xcc1776d3 → uint48, uint48, uint48, uint48, uint48
symbol 0x95d89b41 → string
tokensForTax 0x6d7adcad → uint64, uint64, uint64, uint64, bool
totalSupply 0x18160ddd → uint256
transactionLimits 0x56340e81 → uint96, uint96, uint48, bool
transactionLimitsActive 0x1ebcb3df → bool
withdrawableDividendOf 0xa8b9d240 → uint256
Write Contract 23 functions
These functions modify contract state and require a wallet transaction to execute.
addLp 0x06f20003
address _to
approve 0x095ea7b3
address spender
uint256 amount
returns: bool
changeDevAddress 0x92b46390
address _address
changeExcludedFromFees 0xf84e4265
address _address
bool _isExcluded
changeExcludedFromLimits 0x5ecc2022
address _address
bool _isExcluded
changeMarketingAddress 0x048dec38
address _address
changeSwapTokensAmount 0xb2a9c0c0
uint256 newAmount
claim 0x4e71d92d
No parameters
compound 0xaa5f7e26
uint256 minOutput
decreaseAllowance 0xa457c2d7
address spender
uint256 subtractedValue
returns: bool
excludeFromDividends 0x31e79db0
address _wallet
includeInDividends 0xc0f306ef
address _wallet
increaseAllowance 0x39509351
address spender
uint256 addedValue
returns: bool
removeLimits 0x751039fc
No parameters
renounceOwnership 0x715018a6
No parameters
setBuyTax 0x6cdc3677
uint48 _mktingTax
uint48 _lpTax
uint48 _devTax
uint48 _revShareTax
transfer 0xa9059cbb
address to
uint256 amount
returns: bool
transferFrom 0x23b872dd
address from
address to
uint256 amount
returns: bool
transferOwnership 0xf2fde38b
address newOwner
updateAntiMevEnabled 0x5cfb534e
bool _enabled
updateSellTax 0x45006628
uint48 _mktingTax
uint48 _lpTax
uint48 _devTax
uint48 _revShareTax
updateTransactionLimit 0xf144e620
uint96 newNumInTokens
updateWalletLimit 0x64a81f1d
uint96 newNumInTokens
Recent Transactions
No transactions found for this address