-
Notifications
You must be signed in to change notification settings - Fork 350
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
83 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,83 @@ | ||
import { | ||
allocate, | ||
entryPoint, | ||
execute, | ||
IPreContractCallJP, | ||
PreContractCallInput, | ||
sys, | ||
uint8ArrayToHex, | ||
UintData | ||
} from '@artela/aspect-libs'; | ||
import { Protobuf } from 'as-proto/assembly'; | ||
|
||
/** | ||
*/ | ||
class Aspect implements IPreContractCallJP { | ||
/** | ||
* | ||
* @param input | ||
*/ | ||
preContractCall(input: PreContractCallInput): void { | ||
// read the throttle config from the properties and decode | ||
const interval = sys.aspect.property.get<u64>('interval'); | ||
const limit = sys.aspect.property.get<u64>('limit'); | ||
|
||
// get the contract address, from address and build the storage prefix | ||
const contractAddress = uint8ArrayToHex(input.call!.to); | ||
const from = uint8ArrayToHex(input.call!.from); | ||
const storagePrefix = `${contractAddress}:${from}`; | ||
|
||
// load the current block timestamp | ||
const blockTimeBytes = sys.hostApi.runtimeContext.get( | ||
'block.header.timestamp' | ||
); | ||
const blockTime = Protobuf.decode<UintData>( | ||
blockTimeBytes, | ||
UintData.decode | ||
).data; | ||
|
||
// load last execution timestamp | ||
const lastExecState = sys.aspect.mutableState.get<u64>( | ||
storagePrefix + 'lastExecAt' | ||
); | ||
const lastExec = lastExecState.unwrap(); | ||
|
||
// check if the throttle interval has passed, revert if not | ||
if (lastExec > 0 && blockTime - lastExec < interval) { | ||
sys.revert('throttled'); | ||
} | ||
|
||
// check if the throttle limit has been reached, revert if so | ||
const execTimeState = sys.aspect.mutableState.get<u64>( | ||
storagePrefix + 'execTimes' | ||
); | ||
const execTimes = execTimeState.unwrap(); | ||
if (limit && execTimes >= limit) { | ||
sys.revert('execution limit exceeded'); | ||
} | ||
|
||
// update the throttle state | ||
execTimeState.set(execTimes + 1); | ||
lastExecState.set(blockTime); | ||
} | ||
|
||
/** | ||
* isOwner is the governance account implemented by the Aspect, when any of the governance operation | ||
* (including upgrade, config, destroy) is made, isOwner method will be invoked to check | ||
* against the initiator's account to make sure it has the permission. | ||
* | ||
* @param sender address of the transaction | ||
* @return true if check success, false if check fail | ||
*/ | ||
isOwner(sender: Uint8Array): bool { | ||
return false; | ||
} | ||
} | ||
|
||
// 2.register aspect Instance | ||
const aspect = new Aspect(); | ||
entryPoint.setAspect(aspect); | ||
|
||
// 3.must export it | ||
export { execute, allocate }; |