Address Contract Partially Verified
Address
0x5557f095556FeB36C725f1Ffe94A97631aDEb770
Balance
0 ETH
Nonce
4
Code Size
14851 bytes
Creator
0xF416a7Ac...6Ff1 at tx 0x83d24cc6...f733e8
Indexed Transactions
Index loading...
Contract Bytecode
14851 bytes
0x608060405234801561001057600080fd5b50600436106103835760003560e01c80638516d41d116101de578063b5fc714f1161010f578063ce05bdf9116100ad578063d79875eb1161007c578063d79875eb146105d7578063effa2a7d146105ea578063f2fde38b146105f2578063fbfdd8741461060557610383565b8063ce05bdf91461059c578063d021f22a146105bf578063d4c30dd8146105c7578063d60a2baf146105cf57610383565b8063c0c53b8b116100e9578063c0c53b8b14610589578063c369bb2a1461059c578063cbc4762c146105a4578063ce02ee08146105ac57610383565b8063b5fc714f14610566578063b70598361461056e578063c00007b01461057657610383565b80639be420b11161017c578063a7a42c8c11610156578063a7a42c8c14610530578063ac4b3c0414610538578063accab5501461054b578063af14052c1461055e57610383565b80639be420b11461047f578063a20924e814610520578063a5e54dd01461052857610383565b80638e71e1b4116101b85780638e71e1b4146105005780638ee624a91461050857806395bdb59f146105105780639b4e1e6c1461051857610383565b80638516d41d146104e85780638da5cb5b146104f05780638dca98b3146104f857610383565b8063500ad5de116102b85780636e827ff611610256578063750142e611610230578063750142e6146104ba578063807f05d9146104d0578063817b1cd2146104d8578063828c1e0a146104e057610383565b80636e827ff61461049757806370a082311461049f578063715018a6146104b257610383565b806361d027b31161029257806361d027b314610477578063644712201461047f57806365731f01146104875780636a53452d1461048f57610383565b8063500ad5de1461045f578063502b8c701461046757806351c4c59c1461046f57610383565b80633776dea6116103255780634298c73f116102ff5780634298c73f1461041e5780634b341aed146104315780634bbdb6f9146104445780634e71d92d1461045757610383565b80633776dea614610406578063413d9c3a1461040e57806342546ce41461041657610383565b80631c994d07116103615780631c994d07146103d057806321258602146103d85780632a276796146103e05780632e1a7d4d146103f357610383565b8063059a500c146103885780631758078b1461039d57806319d96d67146103bb575b600080fd5b61039b610396366004613356565b61060d565b005b6103a5610abe565b6040516103b291906133e0565b60405180910390f35b6103c3610acd565b6040516103b2919061397c565b6103c3610ad9565b6103c3610ade565b61039b6103ee3660046132d4565b610ae4565b6103c3610401366004613356565b610b51565b6103c3610fc2565b6103c3610fce565b6103a5610fd5565b6103c361042c3660046132b8565b610ff9565b6103c361043f3660046132b8565b6110cf565b61039b6104523660046132b8565b61111e565b61039b6111ac565b6103a5611423565b6103c3611432565b6103c3611439565b6103a561143f565b6103c361144e565b61039b611453565b6103c3611510565b6103c3611515565b6103c36104ad3660046132b8565b61151c565b61039b611660565b6104c26116b6565b6040516103b292919061399c565b6103a56117e3565b6103c3611807565b6103c361189f565b6103c36118a5565b6103a56118aa565b61039b6118b9565b6103a561196c565b6103a5611990565b6103c361199f565b6103c36119a5565b6103a56119ac565b6103a56119d0565b6103a56119e8565b6104c26105463660046132b8565b6119f7565b61039b610559366004613356565b611a7b565b61039b611ae4565b6103c3611bf6565b6103c3611bfb565b6104c26105843660046132b8565b611c00565b61039b61059736600461330c565b611d45565b6103c3611e2b565b6103c3611e30565b61039b6105ba3660046132b8565b611e37565b6103c3611ec5565b6103a5611ecb565b6103c3611eda565b6104c26105e5366004613386565b611ee0565b6103a561267b565b61039b6106003660046132b8565b61268a565b6103c3612709565b7f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1616001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561066657600080fd5b505afa15801561067a573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061069e919061336e565b600054146106c75760405162461bcd60e51b81526004016106be90613873565b60405180910390fd5b6106cf61270f565b6107046001600160a01b037f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a16116333084612728565b600061070f82612786565b336000908152601160205260409020600401549091506107415733600090815260116020526040902061074190612858565b6040805180820182526001600160d01b038316815265ffffffffffff421660208083019190915233600090815260119091529190912061078091612862565b600061079c61271061079685633b9aca00612917565b9061293b565b905060006107b1612710610796846041612917565b905042621a5e00600c5401111561089857600a54600b546040516340c10f1960e01b81526001600160a01b03928316926340c10f19926107f8929116908590600401613416565b600060405180830381600087803b15801561081257600080fd5b505af1158015610826573d6000803e3d6000fd5b5050600a546001600160a01b031691506340c10f19905033610848858561295b565b6040518363ffffffff1660e01b8152600401610865929190613416565b600060405180830381600087803b15801561087f57600080fd5b505af1158015610893573d6000803e3d6000fd5b505050505b6040516305dc812160e31b81526001600160a01b037f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be761690632ee40908906108e690339087906004016133f4565b600060405180830381600087803b15801561090057600080fd5b505af1158015610914573d6000803e3d6000fd5b50506040516305dc812160e31b81526001600160a01b037f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f169250632ee40908915061096690339087906004016133f4565b600060405180830381600087803b15801561098057600080fd5b505af1158015610994573d6000803e3d6000fd5b505033600081815260116020526040908190206003015490519193507f90890809c654f11d6e72a28fa60149770a0d11ec6c92319d6ceb2bb0a4ea1a1592506109df9188919061399c565b60405180910390a27f0ea809b2fd10635ec1f83398c49b948d3250250ff7a7f7f5423261597a72454f7f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f6001600160a01b031663817b1cd26040518163ffffffff1660e01b815260040160206040518083038186803b158015610a6157600080fd5b505afa158015610a75573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a99919061336e565b42604051610aa892919061399c565b60405180910390a1505050610abb612971565b50565b6006546001600160a01b031681565b678ac7230489e8000081565b601681565b600c5481565b610b0233610af06118aa565b6001600160a01b0316146101aa612978565b6010805460ff191682151517908190556040517f4a66defdf194d0aacb040af6cc101a703a08c8418d3abb20001432096dfd85ab91610b469160ff9091169061346d565b60405180910390a150565b60007f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1616001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015610bac57600080fd5b505afa158015610bc0573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610be4919061336e565b60005414610c045760405162461bcd60e51b81526004016106be90613873565b610c0c61270f565b604051634b341aed60e01b81526000906001600160a01b037f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be761690634b341aed90610c5b9033906004016133e0565b60206040518083038186803b158015610c7357600080fd5b505afa158015610c87573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610cab919061336e565b905080831115610ccd5760405162461bcd60e51b81526004016106be90613556565b825b8015610de5573360009081526011602052604090206003015415610de0573360009081526011602052604081206001810154610d0b919061298a565b60105490915060ff16610d4f57610d25426276a70061295b565b816020015165ffffffffffff1610610d4f5760405162461bcd60e51b81526004016106be90613609565b80516001600160d01b0316821015610dad57336000908152601160205260409020600101548151610da49190610d8e906001600160d01b03168561295b565b3360009081526011602052604090209190612a2e565b60009150610dde565b8051610dc39083906001600160d01b031661295b565b336000908152601160205260409020909250610dde90612acf565b505b610ccf565b610dee84612b2f565b9250610e246001600160a01b037f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a161163385612be2565b60405163398d1d8360e11b81526001600160a01b037f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be76169063731a3b0690610e729033908890600401613416565b600060405180830381600087803b158015610e8c57600080fd5b505af1158015610ea0573d6000803e3d6000fd5b505060405163398d1d8360e11b81526001600160a01b037f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f16925063731a3b069150610ef29033908890600401613416565b600060405180830381600087803b158015610f0c57600080fd5b505af1158015610f20573d6000803e3d6000fd5b505033600081815260116020526040908190206003015490519193507fdf273cb619d95419a9cd0ec88123a0538c85064229baa6363788f743fff90deb9250610f6b9187919061399c565b60405180910390a27f0ea809b2fd10635ec1f83398c49b948d3250250ff7a7f7f5423261597a72454f610f9c611807565b42604051610fab92919061399c565b60405180910390a15050610fbd612971565b919050565b671bc16d674ec8000081565b6276a70081565b7f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a16181565b60007f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be766001600160a01b031663817b1cd26040518163ffffffff1660e01b815260040160206040518083038186803b15801561105457600080fd5b505afa158015611068573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061108c919061336e565b61109857506000610fbd565b6001600160a01b03821660009081526011602052604081206110bd906276a700612c06565b90506110c881612b2f565b9392505050565b6001600160a01b0381166000908152601160205260408120600401546110f757506000610fbd565b6001600160a01b038216600090815260116020526040812061111891612c06565b92915050565b61112a33610af06118aa565b6001600160a01b0381166111505760405162461bcd60e51b81526004016106be906135c2565b600980546001600160a01b0319166001600160a01b03838116919091179182905542600f556008546040517f7a069918400ffda59d7a796750418b338c2b9aedc03f9b8beea61dcab2375f2093610b469392831692169061342f565b6111b461270f565b6000806111c033611c00565b915091507f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f6001600160a01b0316639470b0bd337f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f6001600160a01b0316634b341aed336040518263ffffffff1660e01b815260040161124091906133e0565b60206040518083038186803b15801561125857600080fd5b505afa15801561126c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611290919061336e565b6040518363ffffffff1660e01b81526004016112ad929190613416565b600060405180830381600087803b1580156112c757600080fd5b505af11580156112db573d6000803e3d6000fd5b5050604051634b341aed60e01b81526001600160a01b037f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be76169250639470b0bd915033908390634b341aed906113359084906004016133e0565b60206040518083038186803b15801561134d57600080fd5b505afa158015611361573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611385919061336e565b6040518363ffffffff1660e01b81526004016113a2929190613416565b600060405180830381600087803b1580156113bc57600080fd5b505af11580156113d0573d6000803e3d6000fd5b50505050336001600160a01b03167f987d620f307ff6b94d58743cb7a7509f24071586a77759b77c2d4e29f75a2f9a838360405161140f92919061399c565b60405180910390a25050611421612971565b565b6004546001600160a01b031681565b620186a081565b60005481565b600b546001600160a01b031681565b603781565b61145f33610af06118aa565b6009546001600160a01b03166114875760405162461bcd60e51b81526004016106be9061358b565b62015180600f540142116114ad5760405162461bcd60e51b81526004016106be906136bf565b60098054600880546001600160a01b038084166001600160a01b0319928316179283905592169092556040517feb2a36bf0d2229c3264985b6cd847a8caf18d163047b6e482a86b275f6a452db926115069216906133e0565b60405180910390a1565b600381565b620f424081565b60007f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be766001600160a01b031663817b1cd26040518163ffffffff1660e01b815260040160206040518083038186803b15801561157757600080fd5b505afa15801561158b573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906115af919061336e565b6115bb57506000610fbd565b604051634b341aed60e01b8152611118906001600160a01b037f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be761690634b341aed9061160b9086906004016133e0565b60206040518083038186803b15801561162357600080fd5b505afa158015611637573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061165b919061336e565b612b2f565b61166c33610af06118aa565b6002546040516000916001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600280546001600160a01b0319169055565b6000807f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f6001600160a01b0316637d1fcbfa6040518163ffffffff1660e01b815260040160206040518083038186803b15801561171257600080fd5b505afa158015611726573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061174a919061336e565b91507f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be766001600160a01b0316637d1fcbfa6040518163ffffffff1660e01b815260040160206040518083038186803b1580156117a557600080fd5b505afa1580156117b9573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906117dd919061336e565b90509091565b7f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be7681565b60007f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be766001600160a01b031663817b1cd26040518163ffffffff1660e01b815260040160206040518083038186803b15801561186257600080fd5b505afa158015611876573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061189a919061336e565b905090565b600f5481565b604181565b6002546001600160a01b031690565b6118c533610af06118aa565b6007546001600160a01b03166118ed5760405162461bcd60e51b81526004016106be9061358b565b62015180600e540142116119135760405162461bcd60e51b81526004016106be906136bf565b60078054600680546001600160a01b038084166001600160a01b0319928316179283905592169092556040517fbe5003d33bd2d69fc79a2a2f03ba32b80a39cf1c474d45398690260f81729b0e926115069216906133e0565b7f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a16181565b6005546001600160a01b031681565b600e5481565b621a5e0081565b7f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f81565b7364aa3364f17a4d01c6f1751fd97c2bd3d7e7f1d581565b6008546001600160a01b031681565b6001600160a01b0381166000908152601160205260408120600401548190611a2457506000905080611a76565b6001600160a01b03831660009081526011602052604081206001810154611a4b919061298a565b9050611a6381600001516001600160d01b0316612b2f565b9250806020015165ffffffffffff169150505b915091565b611a8733610af06118aa565b671bc16d674ec80000811115611aaf5760405162461bcd60e51b81526004016106be90613939565b600d8190556040517faf71fdc062ce1aae819fa72e0a5ce8deb553584d0bba52f70a15fa2a84d85e4790610b4690839061397c565b60007f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1616001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611b3f57600080fd5b505afa158015611b53573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b77919061336e565b9050600054811415611ba95760015462015180420311611ba95760405162461bcd60e51b81526004016106be906134c2565b42600155611bb681612cbe565b7f11c6bf55864ff83827df712625d7a80e5583eef0264921025e7cd22003a2151160005482604051611be992919061399c565b60405180910390a1600055565b600a81565b602d81565b6000807f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f6001600160a01b031663c00007b0846040518263ffffffff1660e01b8152600401611c4f91906133e0565b60206040518083038186803b158015611c6757600080fd5b505afa158015611c7b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611c9f919061336e565b604051630c00007b60e41b81529092506001600160a01b037f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be76169063c00007b090611cee9086906004016133e0565b60206040518083038186803b158015611d0657600080fd5b505afa158015611d1a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611d3e919061336e565b9050915091565b611d5133610af06118aa565b6001600160a01b038316611d775760405162461bcd60e51b81526004016106be906138f7565b6001600160a01b038216611d9d5760405162461bcd60e51b81526004016106be90613759565b6001600160a01b038116611dc35760405162461bcd60e51b81526004016106be9061358b565b600b546001600160a01b031615611dec5760405162461bcd60e51b81526004016106be90613478565b600580546001600160a01b039485166001600160a01b031991821617909155600b80549385169382169390931790925560068054919093169116179055565b602381565b6201518081565b611e4333610af06118aa565b6001600160a01b038116611e695760405162461bcd60e51b81526004016106be9061358b565b600780546001600160a01b0319166001600160a01b03838116919091179182905542600e556006546040517f465c9b82d9a3c350b621b44f2222f8fdb9f149cfe253ebec1729d9adf02e755c93610b469392831692169061342f565b600d5481565b600a546001600160a01b031681565b61271081565b600080611eeb61270f565b6008546001600160a01b03163314611f155760405162461bcd60e51b81526004016106be9061351f565b600480546040516370a0823160e01b81526000926001600160a01b037f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1618116936370a0823193611f6a939190921691016133e0565b60206040518083038186803b158015611f8257600080fd5b505afa158015611f96573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611fba919061336e565b90506000611fce606461079684602d612917565b90506000611fe26064610796856016612917565b90506000611ff66064610796866003612917565b60048054604051630f41a04d60e11b81529293506001600160a01b031691631e83409a91612046917f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a16191016133e0565b600060405180830381600087803b15801561206057600080fd5b505af1158015612074573d6000803e3d6000fd5b50506006546001600160a01b037f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1618116935063095ea7b39250166120b8868661319d565b6040518363ffffffff1660e01b81526004016120d5929190613416565b602060405180830381600087803b1580156120ef57600080fd5b505af1158015612103573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061212791906132f0565b506006546040516301b119ff60e01b81526001600160a01b03909116906301b119ff9061215a9086908c9060040161399c565b602060405180830381600087803b15801561217457600080fd5b505af1158015612188573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906121ac919061336e565b60065460405163fedabd0d60e01b81529197506001600160a01b03169063fedabd0d906121df9085908b9060040161399c565b602060405180830381600087803b1580156121f957600080fd5b505af115801561220d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612231919061336e565b600b54909550612267906001600160a01b031661225460646107968a600a612917565b600a546001600160a01b03169190612be2565b600a546040516370a0823160e01b81526000916001600160a01b0316906370a08231906122989030906004016133e0565b60206040518083038186803b1580156122b057600080fd5b505afa1580156122c4573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122e8919061336e565b90507fb90306ad06b2a6ff86ddc9327db583062895ef6540e62dc50add009db5b356eb81604051612319919061397c565b60405180910390a1600a54604051630852cd8d60e31b81526001600160a01b03909116906342966c689061235190849060040161397c565b600060405180830381600087803b15801561236b57600080fd5b505af115801561237f573d6000803e3d6000fd5b50505050600061239e606461079660378a61291790919063ffffffff16565b905060006123b260646107968a6023612917565b60405163095ea7b360e01b81529091507364aa3364f17a4d01c6f1751fd97c2bd3d7e7f1d59063095ea7b39061240e907f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f908690600401613416565b602060405180830381600087803b15801561242857600080fd5b505af115801561243c573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061246091906132f0565b506040516370fd458160e11b81526001600160a01b037f0000000000000000000000005053aa3a263a9abcdad20f4977fc436ff5026c9f169063e1fa8b02906124af9085903090600401613985565b600060405180830381600087803b1580156124c957600080fd5b505af11580156124dd573d6000803e3d6000fd5b505060055461250d92507364aa3364f17a4d01c6f1751fd97c2bd3d7e7f1d591506001600160a01b031683612be2565b600560009054906101000a90046001600160a01b03166001600160a01b031663d264e05e6040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561255d57600080fd5b505af1158015612571573d6000803e3d6000fd5b5050600b546040516370a0823160e01b815261262b93506001600160a01b0390911691507364aa3364f17a4d01c6f1751fd97c2bd3d7e7f1d5906370a08231906125bf9030906004016133e0565b60206040518083038186803b1580156125d757600080fd5b505afa1580156125eb573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061260f919061336e565b7364aa3364f17a4d01c6f1751fd97c2bd3d7e7f1d59190612be2565b600b54612665906001600160a01b037f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1618116911686612be2565b50505050505050612674612971565b9250929050565b6009546001600160a01b031681565b61269633610af06118aa565b6126ad6001600160a01b03821615156101ab612978565b6002546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600280546001600160a01b0319166001600160a01b0392909216919091179055565b60015481565b61272160026003541415610190612978565b6002600355565b612780846323b872dd60e01b85858560405160240161274993929190613449565b60408051601f198184030181529190526020810180516001600160e01b03166001600160e01b0319909316929092179091526131af565b50505050565b60008061282f7f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1616001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b1580156127e557600080fd5b505afa1580156127f9573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061281d919061336e565b61079685678ac7230489e80000612917565b90506001600160d01b038111156111185760405162461bcd60e51b81526004016106be90613651565b6001600490910155565b60048201805460018082019092556040805180820182528481526000602080830182815285835288825293909120915180518354919092015165ffffffffffff16600160d01b026001600160d01b039283166001600160d01b031990921691909117909116178155905190830155908301546128eb576002830181905560018301819055612909565b60028301805460009081526020859052604090206001018290558190555b505060030180546001019055565b60008282026110c884158061293457508385838161293157fe5b04145b6003612978565b600061294a8215156004612978565b81838161295357fe5b049392505050565b600061296b838311156001612978565b50900390565b6001600355565b81612986576129868161324e565b5050565b6129926132a1565b816129af5760405162461bcd60e51b81526004016106be90613790565b60008281526020848152604091829020825160808101845281546001600160d01b038116948201948552600160d01b900465ffffffffffff16606082015292835260010154908201819052151580612a0a5750836001015483145b612a265760405162461bcd60e51b81526004016106be906137fe565b519392505050565b6001600160d01b03811115612a555760405162461bcd60e51b81526004016106be90613703565b600082815260208490526040902060048401548310612a865760405162461bcd60e51b81526004016106be90613688565b80546001600160d01b0316612aad5760405162461bcd60e51b81526004016106be9061382c565b80546001600160d01b0319166001600160d01b03929092169190911790555050565b6001810154612af05760405162461bcd60e51b81526004016106be906137c7565b60018082018054600081815260208590526040812093840180548455938190559092556003830180546000190190555461298657600060028301555050565b6000611118612b3c611807565b610796847f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1616001600160a01b03166370a08231306040518263ffffffff1660e01b8152600401612b8c91906133e0565b60206040518083038186803b158015612ba457600080fd5b505afa158015612bb8573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612bdc919061336e565b90612917565b612c018363a9059cbb60e01b8484604051602401612749929190613416565b505050565b60018201546000905b8015612cb75760008181526020858152604091829020825160808101845281546001600160d01b038116948201948552600160d01b900465ffffffffffff1660608201529283526001015490820152831580612c8357508051602001518490612c8190429065ffffffffffff1661295b565b115b15612ca657805151612c9f9084906001600160d01b031661319d565b9250612cac565b50612cb7565b602001519050612c0f565b5092915050565b612cc661270f565b6040516370a0823160e01b81526000906001600160a01b037f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a16116906370a0823190612d159030906004016133e0565b60206040518083038186803b158015612d2d57600080fd5b505afa158015612d41573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612d65919061336e565b9050600054821115612e285742600c556006546001600160a01b0316612d9d5760405162461bcd60e51b81526004016106be906138c0565b6000612dc083610796670de0b6b3a764000060005461291790919063ffffffff16565b90506000612de4612ddd670de0b6b3a76400006107968686612917565b849061295b565b600454909150612e21906001600160a01b037f000000000000000000000000d46ba6d942050d489dbd938a2c909a5d5039a1618116911683612be2565b505061312e565b42621a5e00600c5401111561312e576000612e616000548410612e4e57620186a0612e53565b620f42405b61079684633b9aca00612917565b600a546040516340c10f1960e01b81529192506001600160a01b0316906340c10f1990612e949030908590600401613416565b600060405180830381600087803b158015612eae57600080fd5b505af1158015612ec2573d6000803e3d6000fd5b505050506000612ee1606461079660378561291790919063ffffffff16565b90506000612ef56064610796856023612917565b600a5460405163095ea7b360e01b81529192506001600160a01b03169063095ea7b390612f48907f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be76908690600401613416565b602060405180830381600087803b158015612f6257600080fd5b505af1158015612f76573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612f9a91906132f0565b50600554600a54612fb8916001600160a01b03918216911683612be2565b6040516370fd458160e11b81526001600160a01b037f0000000000000000000000007661f376bec43c0de357d80658973bb84af3be76169063e1fa8b02906130069085903090600401613985565b600060405180830381600087803b15801561302057600080fd5b505af1158015613034573d6000803e3d6000fd5b50505050600560009054906101000a90046001600160a01b03166001600160a01b031663d264e05e6040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561308857600080fd5b505af115801561309c573d6000803e3d6000fd5b5050600b54600a546040516370a0823160e01b815261312a94506001600160a01b0392831693509116906370a08231906130da9030906004016133e0565b60206040518083038186803b1580156130f257600080fd5b505afa158015613106573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612254919061336e565b5050505b600a54600d546040516340c10f1960e01b81526001600160a01b03909216916340c10f199161316291339190600401613416565b600060405180830381600087803b15801561317c57600080fd5b505af1158015613190573d6000803e3d6000fd5b5050505050610abb612971565b60008282016110c88482101583612978565b600080836001600160a01b0316836040516131ca91906133a7565b6000604051808303816000865af19150503d8060008114613207576040519150601f19603f3d011682016040523d82523d6000602084013e61320c565b606091505b50915091506000821415613224573d6000803e3d6000fd5b61278081516000148061324657508180602001905181019061324691906132f0565b6101a2612978565b62461bcd60e51b6000908152602060045260076024526642414c23000030600a808404818106603090810160081b95839006959095019082900491820690940160101b939093010160c81b604452606490fd5b604080518082019091526000808252602082015290565b6000602082840312156132c9578081fd5b81356110c8816139aa565b6000602082840312156132e5578081fd5b81356110c8816139bf565b600060208284031215613301578081fd5b81516110c8816139bf565b600080600060608486031215613320578182fd5b833561332b816139aa565b9250602084013561333b816139aa565b9150604084013561334b816139aa565b809150509250925092565b600060208284031215613367578081fd5b5035919050565b60006020828403121561337f578081fd5b5051919050565b60008060408385031215613398578182fd5b50508035926020909101359150565b60008251815b818110156133c757602081860181015185830152016133ad565b818111156133d55782828501525b509190910192915050565b6001600160a01b0391909116815260200190565b6001600160a01b039290921682526001600160d01b0316602082015260400190565b6001600160a01b03929092168252602082015260400190565b6001600160a01b0392831681529116602082015260400190565b6001600160a01b039384168152919092166020820152604081019190915260600190565b901515815260200190565b6020808252602a908201527f456c61737469635661756c743a20636f6e747261637420616c726561647920696040820152691b9a5d1a585b1a5e995960b21b606082015260800190565b6020808252603a908201527f414d504c526562617365723a207265626173652063616e206f6e6c792062652060408201527f63616c6c6564206f6e636520657665727920323420686f757273000000000000606082015260800190565b6020808252601a908201527f456c61737469635661756c743a20756e617574686f72697a6564000000000000604082015260600190565b6020808252818101527f456c61737469635661756c743a204e6f7420656e6f7567682062616c616e6365604082015260600190565b6020808252601c908201527f456c61737469635661756c743a20696e76616c69642074726164657200000000604082015260600190565b60208082526027908201527f456c61737469635661756c743a20696e76616c696420617574686f72697a6564604082015266103a3930b232b960c91b606082015260800190565b60208082526028908201527f456c61737469635661756c743a204e6f20756e6c6f636b6564206465706f73696040820152671d1cc8199bdd5b9960c21b606082015260800190565b6020808252601f908201527f577261707065723a207761616d706c20737570706c79206f766572666c6f7700604082015260600190565b6020808252601e908201527f496e76616c69642049443a20494420646f6573206e6f742065786973742e0000604082015260600190565b60208082526024908201527f456c61737469635661756c743a20547261646572206368616e676520636f6f6c6040820152633237bbb760e11b606082015260800190565b60208082526036908201527f496e76616c696420616d6f756e743a20416d6f756e742065786365656473206d60408201527530bc34b6bab6903232b837b9b4ba1030b6b7bab73a1760511b606082015260800190565b6020808252601e908201527f456c61737469635661756c743a20696e76616c69642074726561737572790000604082015260600190565b6020808252601e908201527f496e76616c69642049443a2049442063616e6e6f74206265207a65726f2e0000604082015260600190565b6020808252601f908201527f4c69737420697320656d7074792c2063616e6e6f7420706f7020686561642e00604082015260600190565b6020808252601490820152732737b232903237b2b9903737ba1032bc34b9ba1760611b604082015260600190565b60208082526027908201527f496e76616c696420616d6f756e743a204465706f73697420646f6573206e6f746040820152661032bc34b9ba1760c91b606082015260800190565b6020808252602d908201527f414d504c526562617365723a204f7065726174696f6e20756e617661696c616260408201526c6c65206d69642d72656261736560981b606082015260800190565b6020808252601c908201527f456c61737469635661756c743a20747261646572206e6f742073657400000000604082015260600190565b60208082526022908201527f456c61737469635661756c743a20696e76616c6964207374616b696e6720706f6040820152611bdb60f21b606082015260800190565b60208082526023908201527f456c61737469635661756c743a20696e76616c69642072656261736520726577604082015262185c9960ea1b606082015260800190565b90815260200190565b9182526001600160a01b0316602082015260400190565b918252602082015260400190565b6001600160a01b0381168114610abb57600080fd5b8015158114610abb57600080fdfea2646970667358221220d04f71026cd501339df8f78d4619e7c1823c4e976504538e85dafb2c0b6779f964736f6c63430007060033
Verified Source Code Partial Match
Compiler: v0.7.6+commit.7338295f
EVM: istanbul
Optimization: Yes (200 runs)
ElasticVault.sol 1513 lines
// SPDX-License-Identifier: MIT
pragma solidity 0.7.6;
pragma abicoder v2;
// solhint-disable
/**
* @dev Reverts if `condition` is false, with a revert reason containing `errorCode`. Only codes up to 999 are
* supported.
*/
function _require(bool condition, uint256 errorCode) pure {
if (!condition) _revert(errorCode);
}
/**
* @dev Reverts with a revert reason containing `errorCode`. Only codes up to 999 are supported.
*/
function _revert(uint256 errorCode) pure {
// We're going to dynamically create a revert string based on the error code, with the following format:
// 'BAL#{errorCode}'
// where the code is left-padded with zeroes to three digits (so they range from 000 to 999).
//
// We don't have revert strings embedded in the contract to save bytecode size: it takes much less space to store a
// number (8 to 16 bits) than the individual string characters.
//
// The dynamic string creation algorithm that follows could be implemented in Solidity, but assembly allows for a
// much denser implementation, again saving bytecode size. Given this function unconditionally reverts, this is a
// safe place to rely on it without worrying about how its usage might affect e.g. memory contents.
assembly {
// First, we need to compute the ASCII representation of the error code. We assume that it is in the 0-999
// range, so we only need to convert three digits. To convert the digits to ASCII, we add 0x30, the value for
// the '0' character.
let units := add(mod(errorCode, 10), 0x30)
errorCode := div(errorCode, 10)
let tenths := add(mod(errorCode, 10), 0x30)
errorCode := div(errorCode, 10)
let hundreds := add(mod(errorCode, 10), 0x30)
// With the individual characters, we can now construct the full string. The "BAL#" part is a known constant
// (0x42414c23): we simply shift this by 24 (to provide space for the 3 bytes of the error code), and add the
// characters to it, each shifted by a multiple of 8.
// The revert reason is then shifted left by 200 bits (256 minus the length of the string, 7 characters * 8 bits
// per character = 56) to locate it in the most significant part of the 256 slot (the beginning of a byte
// array).
let revertReason := shl(200, add(0x42414c23000000, add(add(units, shl(8, tenths)), shl(16, hundreds))))
// We can now encode the reason in memory, which can be safely overwritten as we're about to revert. The encoded
// message will have the following layout:
// [ revert reason identifier ] [ string location offset ] [ string length ] [ string contents ]
// The Solidity revert reason identifier is 0x08c739a0, the function selector of the Error(string) function. We
// also write zeroes to the next 28 bytes of memory, but those are about to be overwritten.
mstore(0x0, 0x08c379a000000000000000000000000000000000000000000000000000000000)
// Next is the offset to the location of the string, which will be placed immediately after (20 bytes away).
mstore(0x04, 0x0000000000000000000000000000000000000000000000000000000000000020)
// The string length is fixed: 7 characters.
mstore(0x24, 7)
// Finally, the string itself is stored.
mstore(0x44, revertReason)
// Even if the string is only 7 bytes long, we need to return a full 32 byte slot containing it. The length of
// the encoded message is therefore 4 + 32 + 32 + 32 = 100.
revert(0, 100)
}
}
library Errors {
// Math
uint256 internal constant ADD_OVERFLOW = 0;
uint256 internal constant SUB_OVERFLOW = 1;
uint256 internal constant SUB_UNDERFLOW = 2;
uint256 internal constant MUL_OVERFLOW = 3;
uint256 internal constant ZERO_DIVISION = 4;
uint256 internal constant DIV_INTERNAL = 5;
uint256 internal constant X_OUT_OF_BOUNDS = 6;
uint256 internal constant Y_OUT_OF_BOUNDS = 7;
uint256 internal constant PRODUCT_OUT_OF_BOUNDS = 8;
uint256 internal constant INVALID_EXPONENT = 9;
// Input
uint256 internal constant OUT_OF_BOUNDS = 100;
uint256 internal constant UNSORTED_ARRAY = 101;
uint256 internal constant UNSORTED_TOKENS = 102;
uint256 internal constant INPUT_LENGTH_MISMATCH = 103;
uint256 internal constant ZERO_TOKEN = 104;
// Shared pools
uint256 internal constant MIN_TOKENS = 200;
uint256 internal constant MAX_TOKENS = 201;
uint256 internal constant MAX_SWAP_FEE_PERCENTAGE = 202;
uint256 internal constant MIN_SWAP_FEE_PERCENTAGE = 203;
uint256 internal constant MINIMUM_BPT = 204;
uint256 internal constant CALLER_NOT_VAULT = 205;
uint256 internal constant UNINITIALIZED = 206;
uint256 internal constant BPT_IN_MAX_AMOUNT = 207;
uint256 internal constant BPT_OUT_MIN_AMOUNT = 208;
uint256 internal constant EXPIRED_PERMIT = 209;
uint256 internal constant NOT_TWO_TOKENS = 210;
// Pools
uint256 internal constant MIN_AMP = 300;
uint256 internal constant MAX_AMP = 301;
uint256 internal constant MIN_WEIGHT = 302;
uint256 internal constant MAX_STABLE_TOKENS = 303;
uint256 internal constant MAX_IN_RATIO = 304;
uint256 internal constant MAX_OUT_RATIO = 305;
uint256 internal constant MIN_BPT_IN_FOR_TOKEN_OUT = 306;
uint256 internal constant MAX_OUT_BPT_FOR_TOKEN_IN = 307;
uint256 internal constant NORMALIZED_WEIGHT_INVARIANT = 308;
uint256 internal constant INVALID_TOKEN = 309;
uint256 internal constant UNHANDLED_JOIN_KIND = 310;
uint256 internal constant ZERO_INVARIANT = 311;
uint256 internal constant ORACLE_INVALID_SECONDS_QUERY = 312;
uint256 internal constant ORACLE_NOT_INITIALIZED = 313;
uint256 internal constant ORACLE_QUERY_TOO_OLD = 314;
uint256 internal constant ORACLE_INVALID_INDEX = 315;
uint256 internal constant ORACLE_BAD_SECS = 316;
uint256 internal constant AMP_END_TIME_TOO_CLOSE = 317;
uint256 internal constant AMP_ONGOING_UPDATE = 318;
uint256 internal constant AMP_RATE_TOO_HIGH = 319;
uint256 internal constant AMP_NO_ONGOING_UPDATE = 320;
uint256 internal constant STABLE_INVARIANT_DIDNT_CONVERGE = 321;
uint256 internal constant STABLE_GET_BALANCE_DIDNT_CONVERGE = 322;
uint256 internal constant RELAYER_NOT_CONTRACT = 323;
uint256 internal constant BASE_POOL_RELAYER_NOT_CALLED = 324;
uint256 internal constant REBALANCING_RELAYER_REENTERED = 325;
uint256 internal constant GRADUAL_UPDATE_TIME_TRAVEL = 326;
uint256 internal constant SWAPS_DISABLED = 327;
uint256 internal constant CALLER_IS_NOT_LBP_OWNER = 328;
uint256 internal constant PRICE_RATE_OVERFLOW = 329;
uint256 internal constant INVALID_JOIN_EXIT_KIND_WHILE_SWAPS_DISABLED = 330;
uint256 internal constant WEIGHT_CHANGE_TOO_FAST = 331;
uint256 internal constant LOWER_GREATER_THAN_UPPER_TARGET = 332;
uint256 internal constant UPPER_TARGET_TOO_HIGH = 333;
uint256 internal constant UNHANDLED_BY_LINEAR_POOL = 334;
uint256 internal constant OUT_OF_TARGET_RANGE = 335;
// Lib
uint256 internal constant REENTRANCY = 400;
uint256 internal constant SENDER_NOT_ALLOWED = 401;
uint256 internal constant PAUSED = 402;
uint256 internal constant PAUSE_WINDOW_EXPIRED = 403;
uint256 internal constant MAX_PAUSE_WINDOW_DURATION = 404;
uint256 internal constant MAX_BUFFER_PERIOD_DURATION = 405;
uint256 internal constant INSUFFICIENT_BALANCE = 406;
uint256 internal constant INSUFFICIENT_ALLOWANCE = 407;
uint256 internal constant ERC20_TRANSFER_FROM_ZERO_ADDRESS = 408;
uint256 internal constant ERC20_TRANSFER_TO_ZERO_ADDRESS = 409;
uint256 internal constant ERC20_MINT_TO_ZERO_ADDRESS = 410;
uint256 internal constant ERC20_BURN_FROM_ZERO_ADDRESS = 411;
uint256 internal constant ERC20_APPROVE_FROM_ZERO_ADDRESS = 412;
uint256 internal constant ERC20_APPROVE_TO_ZERO_ADDRESS = 413;
uint256 internal constant ERC20_TRANSFER_EXCEEDS_ALLOWANCE = 414;
uint256 internal constant ERC20_DECREASED_ALLOWANCE_BELOW_ZERO = 415;
uint256 internal constant ERC20_TRANSFER_EXCEEDS_BALANCE = 416;
uint256 internal constant ERC20_BURN_EXCEEDS_ALLOWANCE = 417;
uint256 internal constant SAFE_ERC20_CALL_FAILED = 418;
uint256 internal constant ADDRESS_INSUFFICIENT_BALANCE = 419;
uint256 internal constant ADDRESS_CANNOT_SEND_VALUE = 420;
uint256 internal constant SAFE_CAST_VALUE_CANT_FIT_INT256 = 421;
uint256 internal constant GRANT_SENDER_NOT_ADMIN = 422;
uint256 internal constant REVOKE_SENDER_NOT_ADMIN = 423;
uint256 internal constant RENOUNCE_SENDER_NOT_ALLOWED = 424;
uint256 internal constant BUFFER_PERIOD_EXPIRED = 425;
uint256 internal constant CALLER_IS_NOT_OWNER = 426;
uint256 internal constant NEW_OWNER_IS_ZERO = 427;
uint256 internal constant CODE_DEPLOYMENT_FAILED = 428;
uint256 internal constant CALL_TO_NON_CONTRACT = 429;
uint256 internal constant LOW_LEVEL_CALL_FAILED = 430;
// Vault
uint256 internal constant INVALID_POOL_ID = 500;
uint256 internal constant CALLER_NOT_POOL = 501;
uint256 internal constant SENDER_NOT_ASSET_MANAGER = 502;
uint256 internal constant USER_DOESNT_ALLOW_RELAYER = 503;
uint256 internal constant INVALID_SIGNATURE = 504;
uint256 internal constant EXIT_BELOW_MIN = 505;
uint256 internal constant JOIN_ABOVE_MAX = 506;
uint256 internal constant SWAP_LIMIT = 507;
uint256 internal constant SWAP_DEADLINE = 508;
uint256 internal constant CANNOT_SWAP_SAME_TOKEN = 509;
uint256 internal constant UNKNOWN_AMOUNT_IN_FIRST_SWAP = 510;
uint256 internal constant MALCONSTRUCTED_MULTIHOP_SWAP = 511;
uint256 internal constant INTERNAL_BALANCE_OVERFLOW = 512;
uint256 internal constant INSUFFICIENT_INTERNAL_BALANCE = 513;
uint256 internal constant INVALID_ETH_INTERNAL_BALANCE = 514;
uint256 internal constant INVALID_POST_LOAN_BALANCE = 515;
uint256 internal constant INSUFFICIENT_ETH = 516;
uint256 internal constant UNALLOCATED_ETH = 517;
uint256 internal constant ETH_TRANSFER = 518;
uint256 internal constant CANNOT_USE_ETH_SENTINEL = 519;
uint256 internal constant TOKENS_MISMATCH = 520;
uint256 internal constant TOKEN_NOT_REGISTERED = 521;
uint256 internal constant TOKEN_ALREADY_REGISTERED = 522;
uint256 internal constant TOKENS_ALREADY_SET = 523;
uint256 internal constant TOKENS_LENGTH_MUST_BE_2 = 524;
uint256 internal constant NONZERO_TOKEN_BALANCE = 525;
uint256 internal constant BALANCE_TOTAL_OVERFLOW = 526;
uint256 internal constant POOL_NO_TOKENS = 527;
uint256 internal constant INSUFFICIENT_FLASH_LOAN_BALANCE = 528;
// Fees
uint256 internal constant SWAP_FEE_PERCENTAGE_TOO_HIGH = 600;
uint256 internal constant FLASH_LOAN_FEE_PERCENTAGE_TOO_HIGH = 601;
uint256 internal constant INSUFFICIENT_FLASH_LOAN_FEE_AMOUNT = 602;
}
/**
* @dev Wrappers over Solidity's arithmetic operations with added overflow checks.
* Adapted from OpenZeppelin's SafeMath library
*/
library Math {
/**
* @dev Returns the addition of two unsigned integers of 256 bits, reverting on overflow.
*/
function add(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a + b;
_require(c >= a, Errors.ADD_OVERFLOW);
return c;
}
/**
* @dev Returns the addition of two signed integers, reverting 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), Errors.ADD_OVERFLOW);
return c;
}
/**
* @dev Returns the subtraction of two unsigned integers of 256 bits, reverting on overflow.
*/
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
_require(b <= a, Errors.SUB_OVERFLOW);
uint256 c = a - b;
return c;
}
/**
* @dev Returns the subtraction of two signed integers, reverting 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), Errors.SUB_OVERFLOW);
return c;
}
/**
* @dev Returns the largest of two numbers of 256 bits.
*/
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a >= b ? a : b;
}
/**
* @dev Returns the smallest of two numbers of 256 bits.
*/
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a * b;
_require(a == 0 || c / a == b, Errors.MUL_OVERFLOW);
return c;
}
function div(
uint256 a,
uint256 b,
bool roundUp
) internal pure returns (uint256) {
return roundUp ? divUp(a, b) : divDown(a, b);
}
function divDown(uint256 a, uint256 b) internal pure returns (uint256) {
_require(b != 0, Errors.ZERO_DIVISION);
return a / b;
}
function divUp(uint256 a, uint256 b) internal pure returns (uint256) {
_require(b != 0, Errors.ZERO_DIVISION);
if (a == 0) {
return 0;
} else {
return 1 + (a - 1) / b;
}
}
}
/**
* @dev Collection of functions related to the address type
*/
library Address {
/**
* @dev Returns true if `account` is a contract.
*
* [IMPORTANT]
* ====
* It is unsafe to assume that an address for which this function returns
* false is an externally-owned account (EOA) and not a contract.
*
* Among others, `isContract` will return false for the following
* types of addresses:
*
* - an externally-owned account
* - a contract in construction
* - an address where a contract will be created
* - an address where a contract lived, but was destroyed
* ====
*/
function isContract(address account) internal view returns (bool) {
// This method relies on extcodesize, which returns 0 for contracts in
// construction, since the code is only stored at the end of the
// constructor execution.
uint256 size;
// solhint-disable-next-line no-inline-assembly
assembly {
size := extcodesize(account)
}
return size > 0;
}
/**
* @dev Replacement for Solidity's `transfer`: sends `amount` wei to
* `recipient`, forwarding all available gas and reverting on errors.
*
* https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
* of certain opcodes, possibly making contracts go over the 2300 gas limit
* imposed by `transfer`, making them unable to receive funds via
* `transfer`. {sendValue} removes this limitation.
*
* https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
*
* IMPORTANT: because control is transferred to `recipient`, care must be
* taken to not create reentrancy vulnerabilities. Consider using
* {ReentrancyGuard} or the
* https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/
function sendValue(address payable recipient, uint256 amount) internal {
_require(address(this).balance >= amount, Errors.ADDRESS_INSUFFICIENT_BALANCE);
// solhint-disable-next-line avoid-low-level-calls, avoid-call-value
(bool success, ) = recipient.call{ value: amount }("");
_require(success, Errors.ADDRESS_CANNOT_SEND_VALUE);
}
/**
* @dev Performs a Solidity function call using a low level `call`. A
* plain `call` is an unsafe replacement for a function call: use this
* function instead.
*
* If `target` reverts with a revert reason, it is bubbled up by this
* function (like regular Solidity function calls).
*
* Returns the raw returned data. To convert to the expected return value,
* use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
*
* Requirements:
*
* - `target` must be a contract.
* - calling `target` with `data` must not revert.
*
* _Available since v3.1._
*/
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
_require(isContract(target), Errors.CALL_TO_NON_CONTRACT);
(bool success, bytes memory returndata) = target.call(data);
return verifyCallResult(success, returndata);
}
/**
* @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
* revert reason using the provided one.
*
* _Available since v4.3._
*/
function verifyCallResult(bool success, bytes memory returndata) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
// 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
// solhint-disable-next-line no-inline-assembly
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
_revert(Errors.LOW_LEVEL_CALL_FAILED);
}
}
}
}
/**
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* By default, the owner account will be the one that deploys the contract. This
* can later be changed with {transferOwnership}.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/
abstract contract Ownable {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
constructor() {
_owner = msg.sender;
emit OwnershipTransferred(address(0), msg.sender);
}
/**
* @dev Returns the address of the current owner.
*/
function owner() public view virtual returns (address) {
return _owner;
}
/**
* @dev Throws if called by any account other than the owner.
*/
modifier onlyOwner() {
_require(owner() == msg.sender, Errors.CALLER_IS_NOT_OWNER);
_;
}
/**
* @dev Leaves the contract without owner. It will not be possible to call
* `onlyOwner` functions anymore. Can only be called by the current owner.
*
* NOTE: Renouncing ownership will leave the contract without an owner,
* thereby removing any functionality that is only available to the owner.
*/
function renounceOwnership() public virtual onlyOwner {
emit OwnershipTransferred(_owner, address(0));
_owner = address(0);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Can only be called by the current owner.
*/
function transferOwnership(address newOwner) public virtual onlyOwner {
_require(newOwner != address(0), Errors.NEW_OWNER_IS_ZERO);
emit OwnershipTransferred(_owner, newOwner);
_owner = newOwner;
}
}
/**
* @dev Contract module that helps prevent reentrant calls to a function.
*
* Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
* available, which can be applied to functions to make sure there are no nested
* (reentrant) calls to them.
*
* Note that because there is a single `nonReentrant` guard, functions marked as
* `nonReentrant` may not call one another. This can be worked around by making
* those functions `private`, and then adding `external` `nonReentrant` entry
* points to them.
*
* TIP: If you would like to learn more about reentrancy and alternative ways
* to protect against it, check out our blog post
* https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
*/
abstract contract ReentrancyGuard {
// Booleans are more expensive than uint256 or any type that takes up a full
// word because each write operation emits an extra SLOAD to first read the
// slot's contents, replace the bits taken up by the boolean, and then write
// back. This is the compiler's defense against contract upgrades and
// pointer aliasing, and it cannot be disabled.
// The values being non-zero value makes deployment a bit more expensive,
// but in exchange the refund on every call to nonReentrant will be lower in
// amount. Since refunds are capped to a percentage of the total
// transaction's gas, it is best to keep them low in cases like this one, to
// increase the likelihood of the full refund coming into effect.
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
/**
* @dev Prevents a contract from calling itself, directly or indirectly.
* Calling a `nonReentrant` function from another `nonReentrant`
* function is not supported. It is possible to prevent this from happening
* by making the `nonReentrant` function external, and make it call a
* `private` function that does the actual work.
*/
modifier nonReentrant() {
_enterNonReentrant();
_;
_exitNonReentrant();
}
function _enterNonReentrant() private {
// On the first call to nonReentrant, _status will be _NOT_ENTERED
_require(_status != _ENTERED, Errors.REENTRANCY);
// Any calls to nonReentrant after this point will fail
_status = _ENTERED;
}
function _exitNonReentrant() private {
// By storing the original value once again, a refund is triggered (see
// https://eips.ethereum.org/EIPS/eip-2200)
_status = _NOT_ENTERED;
}
}
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
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);
}
/**
* @title SafeERC20
* @dev Wrappers around ERC20 operations that throw on failure (when the token
* contract returns false). Tokens that return no value (and instead revert or
* throw on failure) are also supported, non-reverting calls are assumed to be
* successful.
* To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
* which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
*/
library SafeERC20 {
function safeTransfer(
IERC20 token,
address to,
uint256 value
) internal {
_callOptionalReturn(address(token), abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(
IERC20 token,
address from,
address to,
uint256 value
) internal {
_callOptionalReturn(address(token), abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
*
* WARNING: `token` is assumed to be a contract: calls to EOAs will *not* revert.
*/
function _callOptionalReturn(address token, bytes memory data) private {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves.
(bool success, bytes memory returndata) = token.call(data);
// If the low-level call didn't succeed we return whatever was returned from it.
assembly {
if eq(success, 0) {
returndatacopy(0, 0, returndatasize())
revert(0, returndatasize())
}
}
// Finally we check the returndata size is either zero or true - note that this check will always pass for EOAs
_require(returndata.length == 0 || abi.decode(returndata, (bool)), Errors.SAFE_ERC20_CALL_FAILED);
}
}
abstract contract AMPLRebaser {
event Rebase(uint256 old_supply, uint256 new_supply);
//
// Check last AMPL total supply from AMPL contract.
//
uint256 public last_ampl_supply;
uint256 public last_rebase_call;
IERC20 immutable public ampl_token;
constructor(IERC20 _ampl_token) {
ampl_token = _ampl_token;
last_ampl_supply = _ampl_token.totalSupply();
last_rebase_call = block.timestamp;
}
function rebase() external {
uint256 new_supply = ampl_token.totalSupply();
// require timestamp to exceed 24 hours in order to execute function OR if ampl supply changed
if(new_supply == last_ampl_supply)
require(block.timestamp - 24 hours > last_rebase_call, "AMPLRebaser: rebase can only be called once every 24 hours");
last_rebase_call = block.timestamp;
_rebase(new_supply);
emit Rebase(last_ampl_supply, new_supply);
last_ampl_supply = new_supply;
}
function _rebase(uint256 new_supply) internal virtual;
modifier _rebaseSynced() {
require(last_ampl_supply == ampl_token.totalSupply(), "AMPLRebaser: Operation unavailable mid-rebase");
_;
}
}
library DepositsLinkedList {
using Math for uint256;
struct Deposit {
uint208 amount;
uint48 timestamp;
}
struct Node {
Deposit deposit;
uint next;
}
struct List {
mapping(uint => Node) nodes;
uint head;
uint tail;
uint length;
uint nodeIdCounter;
}
uint private constant NULL = 0; // Represent the 'null' pointer
function initialize(List storage list) internal {
list.nodeIdCounter = 1; // Initialize node ID counter
}
function insertEnd(List storage list, Deposit memory _deposit) internal {
uint newNodeId = list.nodeIdCounter++; // Use and increment the counter for unique IDs
list.nodes[newNodeId] = Node({deposit: _deposit, next: NULL});
if (list.head == NULL) {
list.head = list.tail = newNodeId;
} else {
list.nodes[list.tail].next = newNodeId;
list.tail = newNodeId;
}
list.length++;
}
function popHead(List storage list) internal {
require(list.head != NULL, "List is empty, cannot pop head.");
uint oldHead = list.head;
list.head = list.nodes[oldHead].next;
delete list.nodes[oldHead];
list.length--;
if (list.head == NULL) {
list.tail = NULL; // Reset the tail if the list is empty
}
}
function sumExpiredDeposits(List storage list, uint256 lock_duration) internal view returns (uint256 sum) {
uint current = list.head;
while (current != NULL) {
Node memory currentNode = list.nodes[current];
if (lock_duration == 0 || ((block.timestamp.sub(currentNode.deposit.timestamp)) > lock_duration)) {
sum = sum.add(currentNode.deposit.amount);
} else {
break;
}
current = currentNode.next;
}
return sum;
}
function modifyDepositAmount(List storage list, uint nodeID, uint256 newAmount) internal {
require(newAmount <= type(uint208).max, "Invalid amount: Amount exceeds maximum deposit amount.");
Node storage node = list.nodes[nodeID];
require(nodeID < list.nodeIdCounter, "Invalid ID: ID does not exist.");
require(node.deposit.amount != 0, "Invalid amount: Deposit does not exist.");
node.deposit.amount = uint208(newAmount);
}
function getDepositById(List storage list, uint id) internal view returns (Deposit memory) {
require(id != NULL, "Invalid ID: ID cannot be zero.");
Node memory node = list.nodes[id];
require(node.next != NULL || id == list.head, "Node does not exist.");
return node.deposit;
}
}
/**
* staking contract for ERC20 tokens or ETH
*/
contract Distribute is Ownable, ReentrancyGuard {
using Math for uint256;
using SafeERC20 for IERC20;
/**
@dev This value is used so when reward token distribution is computed
the difference in precision between staking and reward token doesnt interfere
with bond increase computation
This will be computed based on the difference between decimals
of the staking token and the reward token
If both tokens have the same amount of decimals then this value is 1
If reward token has less decimals then amounts will be multiplied by this value
to match the staking token precision
If staking token has less decimals then this value will also be 1
*/
uint256 public DECIMALS_ADJUSTMENT;
uint256 public constant INITIAL_BOND_VALUE = 1_000_000;
uint256 public bond_value = INITIAL_BOND_VALUE;
//just for info
uint256 public staker_count;
uint256 private _total_staked;
uint256 private _temp_pool;
// the amount of dust left to distribute after the bond value has been updated
uint256 public to_distribute;
mapping(address => uint256) private _bond_value_addr;
mapping(address => uint256) private _stakes;
mapping(address => uint256) private pending_rewards;
uint256 immutable staking_decimals;
/// @dev token to distribute
IERC20 immutable public reward_token;
/**
@dev Initialize the contract
@param _staking_decimals Number of decimals of the staking token
@param _reward_decimals Number of decimals of the reward token
@param _reward_token The token used for rewards. Set to 0 for ETH
*/
constructor(uint256 _staking_decimals, uint256 _reward_decimals, IERC20 _reward_token) {
require(address(_reward_token) != address(0), "Distribute: Invalid reward token");
reward_token = _reward_token;
// sanitize reward token decimals
(bool success, uint256 checked_decimals) = tryGetDecimals(address(_reward_token));
if(success) {
require(checked_decimals == _reward_decimals, "Distribute: Invalid reward decimals");
}
staking_decimals = _staking_decimals;
if(_staking_decimals > _reward_decimals) {
DECIMALS_ADJUSTMENT = 10**(_staking_decimals - _reward_decimals);
} else {
DECIMALS_ADJUSTMENT = 1;
}
}
/**
* @dev Attempts to call the `decimals()` function on an ERC-20 token contract.
* @param tokenAddress The address of the ERC-20 token contract.
* @return success Indicates if the call was successful.
* @return decimals The number of decimals the token uses, or 0 if the call failed.
*/
function tryGetDecimals(address tokenAddress) public view returns (bool success, uint8 decimals) {
bytes memory payload = abi.encodeWithSignature("decimals()");
// Low-level call to the token contract
bytes memory returnData;
(success, returnData) = tokenAddress.staticcall(payload);
// If call was successful and returned data is the expected length for uint8
if (success && returnData.length == 32) {
// Decode the return data
decimals = abi.decode(returnData, (uint8));
} else {
// Default to 0 decimals if call failed or returned unexpected data
return (false, 0);
}
}
/**
@dev Stakes a certain amount, this MUST transfer the given amount from the caller
@param account Address who will own the stake afterwards
@param amount Amount to stake
*/
function stakeFor(address account, uint256 amount) public onlyOwner nonReentrant {
require(account != address(0), "Distribute: Invalid account");
require(amount > 0, "Distribute: Amount must be greater than zero");
_total_staked = _total_staked.add(amount);
uint256 stake = _stakes[account];
if(stake == 0) {
staker_count++;
}
uint256 accumulated_reward = getReward(account);
if(accumulated_reward > 0) {
// set pending rewards to the current reward
pending_rewards[account] = accumulated_reward;
}
_stakes[account] = stake.add(amount);
// reset bond value for this account
_bond_value_addr[account] = bond_value;
}
/**
@dev unstakes a certain amount, if unstaking is currently not possible the function MUST revert
@param account From whom
@param amount Amount to remove from the stake
*/
function unstakeFrom(address payable account, uint256 amount) public onlyOwner nonReentrant {
require(account != address(0), "Distribute: Invalid account");
require(amount > 0, "Distribute: Amount must be greater than zero");
uint256 stake = _stakes[account];
require(amount <= stake, "Distribute: Dont have enough staked");
uint256 to_reward = _getReward(account, amount);
_total_staked -= amount;
stake -= amount;
_stakes[account] = stake;
if(stake == 0) {
staker_count--;
}
if(to_reward == 0) return;
// void pending rewards
pending_rewards[account] = 0;
//take into account dust error during payment too
if(address(reward_token) != address(0)) {
reward_token.safeTransfer(account, to_reward);
}
else {
Address.sendValue(account, to_reward);
}
}
/**
@dev Withdraws rewards (basically unstake then restake)
@param account From whom
@param amount Amount to remove from the stake
*/
function withdrawFrom(address payable account, uint256 amount) external onlyOwner {
unstakeFrom(account, amount);
stakeFor(account, amount);
}
/**
@dev Called contracts to distribute dividends
Updates the bond value
@param amount Amount of token to distribute
@param from Address from which to take the token
*/
function distribute(uint256 amount, address from) external payable onlyOwner nonReentrant {
if(address(reward_token) != address(0)) {
if(amount == 0) return;
reward_token.safeTransferFrom(from, address(this), amount);
require(msg.value == 0, "Distribute: Illegal distribution");
} else {
amount = msg.value;
}
// bond precision is always based on 1 unit of staked token
uint256 total_bonds = _total_staked / 10**staking_decimals;
if(total_bonds == 0) {
// not enough staked to compute bonds account, put into temp pool
_temp_pool = _temp_pool.add(amount);
return;
}
// if a temp pool existed, add it to the current distribution
if(_temp_pool > 0) {
amount = amount.add(_temp_pool);
_temp_pool = 0;
}
uint256 temp_to_distribute = to_distribute + amount;
// bond value is always computed on decimals adjusted rewards
uint256 bond_increase = temp_to_distribute * DECIMALS_ADJUSTMENT / total_bonds;
// adjust back for distributed total
uint256 distributed_total = total_bonds.mul(bond_increase) / DECIMALS_ADJUSTMENT;
bond_value = bond_value.add(bond_increase);
//collect the dust because of the PRECISION used for bonds
//it will be reinjected into the next distribution
to_distribute = temp_to_distribute - distributed_total;
}
/**
@dev Returns the current total staked for an address
@param account address owning the stake
@return the total staked for this account
*/
function totalStakedFor(address account) external view returns (uint256) {
return _stakes[account];
}
/**
@return current staked token
*/
function totalStaked() external view returns (uint256) {
return _total_staked;
}
/**
@dev Returns how much the user can withdraw currently
@param account Address of the user to check reward for
@return the amount account will perceive if he unstakes now
*/
function getReward(address account) public view returns (uint256) {
return _getReward(account,_stakes[account]);
}
/**
@dev returns the total amount of stored rewards
*/
function getTotalReward() external view returns (uint256) {
if(address(reward_token) != address(0)) {
return reward_token.balanceOf(address(this));
} else {
return address(this).balance;
}
}
/**
@dev Returns how much the user can withdraw currently
@param account Address of the user to check reward for
@param amount Number of stakes
@return reward the amount account will perceive if he unstakes now
*/
function _getReward(address account, uint256 amount) internal view returns (uint256 reward) {
// we apply decimals adjustement as bond value is computed on decimals adjusted rewards
uint256 accountBonds = amount.divDown(10**staking_decimals);
reward = accountBonds.mul(bond_value.sub(_bond_value_addr[account])).divDown(DECIMALS_ADJUSTMENT);
// adding pending rewards
reward = reward.add(pending_rewards[account]);
}
}
interface IStakingDoubleERC20 {
function forward() external;
}
interface ITrader {
event Sale_EEFI(uint256 ampl_amount, uint256 eefi_amount);
event Sale_OHM(uint256 ampl_amount, uint256 ohm_amount);
function sellAMPLForOHM(uint256 amount, uint256 minimalExpectedAmount) external returns (uint256);
function sellAMPLForEEFI(uint256 amount, uint256 minimalExpectedAmount) external returns (uint256);
}
/**
* Helper inspired by waampl https://github.com/ampleforth/ampleforth-contracts/blob/master/contracts/waampl.sol
* The goal is to wrap AMPL into non rebasing user shares
*/
abstract contract Wrapper {
using Math for uint256;
/// @dev The maximum waampl supply.
uint256 public constant MAX_WAAMPL_SUPPLY = 10_000_000e12; // 10 M at 12 decimals
IERC20 immutable public ampl;
constructor(IERC20 _ampl) {
require(address(_ampl) != address(0), "Wrapper: Invalid ampl token address");
ampl = _ampl;
}
/// @dev Converts AMPLs to waampl amount.
function _ampleTowaample(uint256 amples)
internal
view
returns (uint208)
{
uint256 waamples = amples.mul(MAX_WAAMPL_SUPPLY).divDown(ampl.totalSupply());
// maximum value is 10_000_000e12 and always fits into uint208
require(waamples <= type(uint208).max, "Wrapper: waampl supply overflow");
return uint208(waamples);
}
}
interface IEEFIToken {
function mint(address account, uint256 amount) external;
function burn(uint256 amount) external;
}
contract TokenStorage is Ownable {
using SafeERC20 for IERC20;
function claim(address token) external onlyOwner() {
IERC20(token).safeTransfer(msg.sender, IERC20(token).balanceOf(address(this)));
}
}
contract ElasticVault is AMPLRebaser, Wrapper, Ownable, ReentrancyGuard {
using SafeERC20 for IERC20;
using Math for uint256;
using DepositsLinkedList for DepositsLinkedList.List;
TokenStorage public token_storage;
IStakingDoubleERC20 public staking_pool;
ITrader public trader;
ITrader pending_trader;
address public authorized_trader;
address public pending_authorized_trader;
IERC20 public eefi_token;
Distribute immutable public rewards_eefi;
Distribute immutable public rewards_ohm;
address payable public treasury;
uint256 public last_positive = block.timestamp;
uint256 public rebase_caller_reward = 0; // The amount of EEFI to be minted to the rebase caller as a reward
IERC20 public constant ohm_token = IERC20(0x64aa3364F17a4D01c6f1751Fd97C2BD3D7e7f1D5);
uint256 public trader_change_request_time;
uint256 public authorized_trader_change_request_time;
bool emergencyWithdrawalEnabled;
/*
Parameter Definitions: //Parameters updated from v1 vault
- EEFI Deposit Rate: Depositors receive reward of .0001 EEFI * Amount of AMPL user deposited into vault
- EEFI Negative Rebase Rate: When AMPL supply declines mint EEFI at rate of .000001 EEFI * total AMPL deposited into vault
- EEFI Equilibrium Rebase Rate: When AMPL supply is does not change (is at equilibrium) mint EEFI at a rate of .00001 EEFI * total AMPL deposited into vault
- Deposit FEE_10000: .65% of EEFI minted to user upon initial deposit is delivered to Treasury
- Lock Time: AMPL deposited into vault is locked for 90 days; lock time applies to each new AMPL deposit
- Trade Posiitve EEFI_100: Upon positive rebase 45% of new AMPL supply (based on total AMPL in vault) is sold and used to buy EEFI
- Trade Positive OHM_100: Upon positive rebase 22% of the new AMPL supply (based on total AMPL in vault) is sold for OHM
- Trade Positive Treasury_100: Upon positive rebase 3% of new AMPL supply (based on total AMPL in vault) is sent to Treasury
- Trade Positive Rewards_100: Upon positive rebase, send 55% of OHM rewards to users staking AMPL in vault
- Trade Positive LP Staking_100: Upon positive rebase, send 35% of OHM rewards to users staking LP tokens (EEFI/OHM)
- Trade Neutral/Negative Rewards: Upon neutral/negative rebase, send 55% of EEFI rewards to users staking AMPL in vault
- Trade Neutral/Negative LP Staking: Upon neutral/negative rebase, send 35% of EEFI rewards to users staking LP tokens (EEFI/OHM)
- Minting Decay: If AMPL does not experience a positive rebase (increase in AMPL supply) for 20 days, do not mint EEFI, distribute rewards to stakers
- Treasury EEFI_100: Amount of EEFI distributed to DAO Treasury after EEFI buy and burn; 10% of purchased EEFI distributed to Treasury
- Max Rebase Reward: Immutable maximum amount of EEFI that can be minted to rebase caller
- Trader Change Cooldown: Cooldown period for updates to authorized trader address
*/
uint256 constant public EEFI_DEPOSIT_RATE = 0.0001e8;
uint256 constant public EEFI_NEGATIVE_REBASE_RATE = 0.000001e12;
uint256 constant public EEFI_EQULIBRIUM_REBASE_RATE = 0.00001e10;
uint256 constant public DEPOSIT_FEE_10000 = 0.0065e4;
uint256 constant public LOCK_TIME = 90 days;
uint256 constant public TRADE_POSITIVE_EEFI_100 = 45;
uint256 constant public TRADE_POSITIVE_OHM_100 = 22;
uint256 constant public TRADE_POSITIVE_TREASURY_100 = 3;
uint256 constant public TRADE_POSITIVE_OHM_REWARDS_100 = 55;
uint256 constant public TRADE_NEUTRAL_NEG_EEFI_REWARDS_100 = 55;
uint256 constant public TRADE_POSITIVE_LPSTAKING_100 = 35;
uint256 constant public TRADE_NEUTRAL_NEG_LPSTAKING_100 = 35;
uint256 constant public TREASURY_EEFI_100 = 10;
uint256 constant public MINTING_DECAY = 20 days;
uint256 constant public MAX_REBASE_REWARD = 2 ether; // 2 EEFI is the maximum reward for a rebase caller
uint256 constant public CHANGE_COOLDOWN = 1 days;
/*
Event Definitions:
- Burn: EEFI burned (EEFI purchased using AMPL is burned)
- Claimed: Rewards claimed by address
- Deposit: AMPL deposited by address
- Withdrawal: AMPL withdrawn by address
- StakeChanged: AMPL staked in contract; calculated as shares of total AMPL deposited
- RebaseRewardChanged: Amount of reward distributed to rebase caller changed; Reward amount cannot exceed MAX_REBASE_REWARD
- TraderChangeRequest: Initates 1-day cooldown period to change authorized trader
- TraderChanged: Authorized trader contract changed
- AuthorizedTraderChanged: EOA authorized to conduct trading operations changed
- EmergencyWithdrawal: Emergency withdrawal mode enabled (allows depositors to withdraw deposits before timelock expires)
*/
event Burn(uint256 amount);
event Claimed(address indexed account, uint256 ohm, uint256 eefi);
event Deposit(address indexed account, uint256 amount, uint256 length);
event Withdrawal(address indexed account, uint256 amount, uint256 length);
event StakeChanged(uint256 total, uint256 timestamp);
event RebaseRewardChanged(uint256 rebaseCallerReward);
event TraderChangeRequest(address oldTrader, address newTrader);
event AuthorizedTraderChangeRequest(address oldTrader, address newTrader);
event TraderChanged(address trader);
event AuthorizedTraderChanged(address trader);
event EmergencyWithdrawal(bool enabled);
mapping(address => DepositsLinkedList.List) private _deposits;
// Contract can mint new EEFI, and distribute OHM and EEFI rewards
constructor(IERC20 _eefi_token, IERC20 ampl_token)
AMPLRebaser(ampl_token)
Wrapper(ampl_token)
Ownable() {
require(address(_eefi_token) != address(0), "ElasticVault: Invalid eefi token");
require(address(ampl_token) != address(0), "ElasticVault: Invalid ampl token");
eefi_token = _eefi_token;
// we're staking wampl which is 12 digits, reward eefi is 18 digits
rewards_eefi = new Distribute(12, 18, IERC20(eefi_token));
rewards_ohm = new Distribute(12, 9, IERC20(ohm_token));
token_storage = new TokenStorage();
}
/**
* @param account User address
* @return total amount of shares owned by...
// [truncated — 66480 bytes total]
Read Contract
CHANGE_COOLDOWN 0xcbc4762c → uint256
DEPOSIT_FEE_10000 0x8516d41d → uint256
EEFI_DEPOSIT_RATE 0xd60a2baf → uint256
EEFI_EQULIBRIUM_REBASE_RATE 0x502b8c70 → uint256
EEFI_NEGATIVE_REBASE_RATE 0x6e827ff6 → uint256
LOCK_TIME 0x413d9c3a → uint256
MAX_REBASE_REWARD 0x3776dea6 → uint256
MAX_WAAMPL_SUPPLY 0x19d96d67 → uint256
MINTING_DECAY 0x9b4e1e6c → uint256
TRADE_NEUTRAL_NEG_EEFI_REWARDS_100 0x64471220 → uint256
TRADE_NEUTRAL_NEG_LPSTAKING_100 0xc369bb2a → uint256
TRADE_POSITIVE_EEFI_100 0xb7059836 → uint256
TRADE_POSITIVE_LPSTAKING_100 0xce05bdf9 → uint256
TRADE_POSITIVE_OHM_100 0x1c994d07 → uint256
TRADE_POSITIVE_OHM_REWARDS_100 0x9be420b1 → uint256
TRADE_POSITIVE_TREASURY_100 0x6a53452d → uint256
TREASURY_EEFI_100 0xb5fc714f → uint256
ampl 0x42546ce4 → address
ampl_token 0x8e71e1b4 → address
authorized_trader 0xa7a42c8c → address
authorized_trader_change_request_time 0x828c1e0a → uint256
balanceOf 0x70a08231 → uint256
eefi_token 0xd4c30dd8 → address
firstDeposit 0xac4b3c04 → uint256, uint256
getReward 0xc00007b0 → uint256, uint256
last_ampl_supply 0x51c4c59c → uint256
last_positive 0x21258602 → uint256
last_rebase_call 0xfbfdd874 → uint256
ohm_token 0xa5e54dd0 → address
owner 0x8da5cb5b → address
pending_authorized_trader 0xeffa2a7d → address
rebase_caller_reward 0xd021f22a → uint256
rewards_eefi 0x807f05d9 → address
rewards_ohm 0xa20924e8 → address
staking_pool 0x8ee624a9 → address
token_storage 0x500ad5de → address
totalClaimableBy 0x4298c73f → uint256
totalReward 0x750142e6 → uint256, uint256
totalStaked 0x817b1cd2 → uint256
totalStakedFor 0x4b341aed → uint256
trader 0x1758078b → address
trader_change_request_time 0x95bdb59f → uint256
treasury 0x61d027b3 → address
Write Contract 14 functions
These functions modify contract state and require a wallet transaction to execute.
claim 0x4e71d92d
No parameters
initialize 0xc0c53b8b
address _staking_pool
address _treasury
address _trader
makeDeposit 0x059a500c
uint256 amount
rebase 0xaf14052c
No parameters
renounceOwnership 0x715018a6
No parameters
sell 0xd79875eb
uint256 minimalExpectedEEFI
uint256 minimalExpectedOHM
returns: uint256, uint256
setAuthorizedTrader 0x65731f01
No parameters
setAuthorizedTraderRequest 0x4bbdb6f9
address _authorized_trader
setEmergencyWithdrawal 0x2a276796
bool _emergencyWithdrawalEnabled
setRebaseReward 0xaccab550
uint256 new_rebase_reward
setTrader 0x8dca98b3
No parameters
setTraderRequest 0xce02ee08
address _trader
transferOwnership 0xf2fde38b
address newOwner
withdraw 0x2e1a7d4d
uint256 amount
returns: uint256
Recent Transactions
Transaction index is loading. Only unfinalized transactions are shown while the index starts up.