Skip to content

Commit

Permalink
Merge pull request #1353 from Tsuizen/task7
Browse files Browse the repository at this point in the history
task7: Tsuizen
  • Loading branch information
ourai authored Aug 2, 2024
2 parents 57a42b7 + 024a53e commit 373db5b
Showing 1 changed file with 83 additions and 0 deletions.
83 changes: 83 additions & 0 deletions members/Tsuizen/task7/index.ts
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 };

0 comments on commit 373db5b

Please sign in to comment.