This commit is contained in:
Nanolucas 2022-03-08 14:57:22 +01:00 committed by GitHub
commit 9d73fc91cf
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 140 additions and 40 deletions

View File

@ -186,6 +186,10 @@ $constructorData = $contract->bytecode($bytecode)->getData($params);
// get function data // get function data
$functionData = $contract->at($contractAddress)->getData($functionName, $params); $functionData = $contract->at($contractAddress)->getData($functionName, $params);
//get event log data
//$fromBlock and $toBlock are optional, default to 'latest' and accept block numbers integers
$events = $contract->getEventLogs($eventName, $fromBlock, $toBlock);
``` ```
# Assign value to outside scope(from callback scope to outside scope) # Assign value to outside scope(from callback scope to outside scope)

View File

@ -1,7 +1,7 @@
<?php <?php
/** /**
* This file is part of web3.php package. * This file is part of the web3.php package.
* *
* (c) Kuan-Cheng,Lai <alk03073135@gmail.com> * (c) Kuan-Cheng,Lai <alk03073135@gmail.com>
* *
@ -11,7 +11,8 @@
namespace Web3; namespace Web3;
use InvalidArgumentException; use \InvalidArgumentException;
use \RuntimeException;
use Web3\Providers\Provider; use Web3\Providers\Provider;
use Web3\Providers\HttpProvider; use Web3\Providers\HttpProvider;
use Web3\RequestManagers\RequestManager; use Web3\RequestManagers\RequestManager;
@ -175,7 +176,7 @@ class Contract
// public function __call($name, $arguments) // public function __call($name, $arguments)
// { // {
// if (empty($this->provider)) { // if (empty($this->provider)) {
// throw new \RuntimeException('Please set provider first.'); // throw new RuntimeException('Please set provider first.');
// } // }
// $class = explode('\\', get_class()); // $class = explode('\\', get_class());
// if (preg_match('/^[a-zA-Z0-9]+$/', $name) === 1) { // if (preg_match('/^[a-zA-Z0-9]+$/', $name) === 1) {
@ -267,6 +268,7 @@ class Contract
/** /**
* getFunctions * getFunctions
* get an array of all methods in the loaded contract
* *
* @return array * @return array
*/ */
@ -277,6 +279,7 @@ class Contract
/** /**
* getEvents * getEvents
* get an array of all events (and their inputs) in the loaded contract
* *
* @return array * @return array
*/ */
@ -368,6 +371,7 @@ class Contract
/** /**
* at * at
* set the address of the deployed contract to make calls to
* *
* @param string $address * @param string $address
* @return $this * @return $this
@ -375,7 +379,7 @@ class Contract
public function at($address) public function at($address)
{ {
if (AddressValidator::validate($address) === false) { if (AddressValidator::validate($address) === false) {
throw new InvalidArgumentException('Please make sure address is valid.'); throw new InvalidArgumentException('Please make sure the contract address is valid.');
} }
$this->toAddress = AddressFormatter::format($address); $this->toAddress = AddressFormatter::format($address);
@ -391,7 +395,7 @@ class Contract
public function bytecode($bytecode) public function bytecode($bytecode)
{ {
if (HexValidator::validate($bytecode) === false) { if (HexValidator::validate($bytecode) === false) {
throw new InvalidArgumentException('Please make sure bytecode is valid.'); throw new InvalidArgumentException('Please make sure the bytecode input is valid.');
} }
$this->bytecode = Utils::stripZero($bytecode); $this->bytecode = Utils::stripZero($bytecode);
@ -407,7 +411,7 @@ class Contract
public function abi($abi) public function abi($abi)
{ {
if (StringValidator::validate($abi) === false) { if (StringValidator::validate($abi) === false) {
throw new InvalidArgumentException('Please make sure abi is valid.'); throw new InvalidArgumentException('Please make sure the abi input is valid.');
} }
$abiArray = []; $abiArray = [];
if (is_string($abi)) { if (is_string($abi)) {
@ -438,7 +442,7 @@ class Contract
/** /**
* new * new
* Deploy a contruct with params. * Deploy a new contract, along with any relevant parameters for its constructor.
* *
* @param mixed * @param mixed
* @return void * @return void
@ -452,13 +456,13 @@ class Contract
$input_count = isset($constructor['inputs']) ? count($constructor['inputs']) : 0; $input_count = isset($constructor['inputs']) ? count($constructor['inputs']) : 0;
if (count($arguments) < $input_count) { if (count($arguments) < $input_count) {
throw new InvalidArgumentException('Please make sure you have put all constructor params and callback.'); throw new InvalidArgumentException('Please make sure you have included all constructor parameters and a callback function.');
} }
if (is_callable($callback) !== true) { if (is_callable($callback) !== true) {
throw new \InvalidArgumentException('The last param must be callback function.'); throw new InvalidArgumentException('The last parameter must be a callback function.');
} }
if (!isset($this->bytecode)) { if (!isset($this->bytecode)) {
throw new \InvalidArgumentException('Please call bytecode($bytecode) before new().'); throw new InvalidArgumentException('Please call bytecode($bytecode) before new().');
} }
$params = array_splice($arguments, 0, $input_count); $params = array_splice($arguments, 0, $input_count);
$data = $this->ethabi->encodeParameters($constructor, $params); $data = $this->ethabi->encodeParameters($constructor, $params);
@ -480,7 +484,8 @@ class Contract
/** /**
* send * send
* Send function method. * Send inputs to a specific method of the deployed contract
* (interacts with chain data and can alter it: costs gas)
* *
* @param mixed * @param mixed
* @return void * @return void
@ -493,7 +498,7 @@ class Contract
$callback = array_pop($arguments); $callback = array_pop($arguments);
if (!is_string($method)) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is string.'); throw new InvalidArgumentException('Please make sure the method name is supplied as a string as the first parameter.');
} }
$functions = []; $functions = [];
@ -503,10 +508,10 @@ class Contract
} }
}; };
if (count($functions) < 1) { if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.'); throw new InvalidArgumentException('Please make sure the named method exists in the contract.');
} }
if (is_callable($callback) !== true) { if (is_callable($callback) !== true) {
throw new \InvalidArgumentException('The last param must be callback function.'); throw new InvalidArgumentException('The last parameter must be a callback function.');
} }
// check the last one in arguments is transaction object // check the last one in arguments is transaction object
@ -558,7 +563,7 @@ class Contract
break; break;
} }
if (empty($data) || empty($functionName)) { if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); throw new InvalidArgumentException('Please make sure you have included all parameters of the method and a callback function.');
} }
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName); $functionSignature = $this->ethabi->encodeFunctionSignature($functionName);
$transaction['to'] = $this->toAddress; $transaction['to'] = $this->toAddress;
@ -575,7 +580,8 @@ class Contract
/** /**
* call * call
* Call function method. * Call a specific method of the deployed contract
* (read-only, cannot alter chain data: does not cost gas)
* *
* @param mixed * @param mixed
* @return void * @return void
@ -588,7 +594,7 @@ class Contract
$callback = array_pop($arguments); $callback = array_pop($arguments);
if (!is_string($method)) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is string.'); throw new InvalidArgumentException('Please make sure the method name is supplied as a string as the first parameter.');
} }
$functions = []; $functions = [];
@ -598,10 +604,10 @@ class Contract
} }
}; };
if (count($functions) < 1) { if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.'); throw new InvalidArgumentException('Please make sure the named method exists in the contract.');
} }
if (is_callable($callback) !== true) { if (is_callable($callback) !== true) {
throw new \InvalidArgumentException('The last param must be callback function.'); throw new InvalidArgumentException('The last parameter must be a callback function.');
} }
// check the arguments // check the arguments
@ -623,7 +629,7 @@ class Contract
break; break;
} }
if (empty($data) || empty($functionName)) { if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); throw new InvalidArgumentException('Please make sure you have included all parameters of the method and a callback function.');
} }
// remove arguments // remove arguments
array_splice($arguments, 0, $paramsLen); array_splice($arguments, 0, $paramsLen);
@ -686,13 +692,13 @@ class Contract
$constructor = $this->constructor; $constructor = $this->constructor;
if (count($arguments) < count($constructor['inputs'])) { if (count($arguments) < count($constructor['inputs'])) {
throw new InvalidArgumentException('Please make sure you have put all constructor params and callback.'); throw new InvalidArgumentException('Please make sure you have included all constructor parameters and a callback function.');
} }
if (is_callable($callback) !== true) { if (is_callable($callback) !== true) {
throw new \InvalidArgumentException('The last param must be callback function.'); throw new InvalidArgumentException('The last parameter must be a callback function.');
} }
if (!isset($this->bytecode)) { if (!isset($this->bytecode)) {
throw new \InvalidArgumentException('Please call bytecode($bytecode) before estimateGas().'); throw new InvalidArgumentException('Please call bytecode($bytecode) before estimateGas().');
} }
$params = array_splice($arguments, 0, count($constructor['inputs'])); $params = array_splice($arguments, 0, count($constructor['inputs']));
$data = $this->ethabi->encodeParameters($constructor, $params); $data = $this->ethabi->encodeParameters($constructor, $params);
@ -706,7 +712,7 @@ class Contract
$method = array_splice($arguments, 0, 1)[0]; $method = array_splice($arguments, 0, 1)[0];
if (!is_string($method)) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is string.'); throw new InvalidArgumentException('Please make sure the method name is supplied as a string as the first parameter.');
} }
$functions = []; $functions = [];
@ -716,10 +722,10 @@ class Contract
} }
}; };
if (count($functions) < 1) { if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.'); throw new InvalidArgumentException('Please make sure the named method exists in the contract.');
} }
if (is_callable($callback) !== true) { if (is_callable($callback) !== true) {
throw new \InvalidArgumentException('The last param must be callback function.'); throw new InvalidArgumentException('The last parameter must be a callback function.');
} }
// check the last one in arguments is transaction object // check the last one in arguments is transaction object
@ -771,7 +777,7 @@ class Contract
break; break;
} }
if (empty($data) || empty($functionName)) { if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); throw new InvalidArgumentException('Please make sure you have included all parameters of the method and a callback function.');
} }
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName); $functionSignature = $this->ethabi->encodeFunctionSignature($functionName);
$transaction['to'] = $this->toAddress; $transaction['to'] = $this->toAddress;
@ -789,9 +795,9 @@ class Contract
/** /**
* getData * getData
* Get the function method call data. * Get the contract method's call data.
* With this function, you can send signed contract function transaction. * With this function, you can send signed contract method transactions.
* 1. Get the funtion data with params. * 1. Get the method data with parameters.
* 2. Sign the data with user private key. * 2. Sign the data with user private key.
* 3. Call sendRawTransaction. * 3. Call sendRawTransaction.
* *
@ -808,10 +814,10 @@ class Contract
$constructor = $this->constructor; $constructor = $this->constructor;
if (count($arguments) < count($constructor['inputs'])) { if (count($arguments) < count($constructor['inputs'])) {
throw new InvalidArgumentException('Please make sure you have put all constructor params and callback.'); throw new InvalidArgumentException('Please make sure you have included all constructor parameters and a callback function.');
} }
if (!isset($this->bytecode)) { if (!isset($this->bytecode)) {
throw new \InvalidArgumentException('Please call bytecode($bytecode) before getData().'); throw new InvalidArgumentException('Please call bytecode($bytecode) before getData().');
} }
$params = array_splice($arguments, 0, count($constructor['inputs'])); $params = array_splice($arguments, 0, count($constructor['inputs']));
$data = $this->ethabi->encodeParameters($constructor, $params); $data = $this->ethabi->encodeParameters($constructor, $params);
@ -820,7 +826,7 @@ class Contract
$method = array_splice($arguments, 0, 1)[0]; $method = array_splice($arguments, 0, 1)[0];
if (!is_string($method)) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is string.'); throw new InvalidArgumentException('Please make sure the method name is supplied as a string as the first parameter.');
} }
$functions = []; $functions = [];
@ -830,7 +836,7 @@ class Contract
} }
}; };
if (count($functions) < 1) { if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.'); throw new InvalidArgumentException('Please make sure the named method exists in the contract.');
} }
$params = $arguments; $params = $arguments;
@ -849,7 +855,7 @@ class Contract
break; break;
} }
if (empty($data) || empty($functionName)) { if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); throw new InvalidArgumentException('Please make sure you have included all parameters of the method and a callback function.');
} }
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName); $functionSignature = $this->ethabi->encodeFunctionSignature($functionName);
$functionData = Utils::stripZero($functionSignature) . Utils::stripZero($data); $functionData = Utils::stripZero($functionSignature) . Utils::stripZero($data);
@ -857,4 +863,94 @@ class Contract
return $functionData; return $functionData;
} }
} }
/**
* getEventLogs
*
* @param string $eventName
* @param string|int $fromBlock
* @param string|int $toBlock
* @return array
*/
public function getEventLogs(string $eventName, $fromBlock = 'latest', $toBlock = 'latest')
{
//try to ensure block numbers are valid together
if ($fromBlock !== 'latest') {
if (!is_int($fromBlock) || $fromBlock < 1) {
throw new InvalidArgumentException('Please make sure fromBlock is a valid block number');
} else if ($toBlock !== 'latest' && $fromBlock > $toBlock) {
throw new InvalidArgumentException('Please make sure fromBlock is equal or less than toBlock');
}
}
if ($toBlock !== 'latest') {
if (!is_int($toBlock) || $toBlock < 1) {
throw new InvalidArgumentException('Please make sure toBlock is a valid block number');
} else if ($fromBlock === 'latest') {
throw new InvalidArgumentException('Please make sure toBlock is equal or greater than fromBlock');
}
}
$eventLogData = [];
//ensure the event actually exists before trying to filter for it
if (!array_key_exists($eventName, $this->events)) {
throw new InvalidArgumentException("'{$eventName}' does not exist in the ABI for this contract");
}
//indexed and non-indexed event parameters must be treated separately
//indexed parameters are stored in the 'topics' array
//non-indexed parameters are stored in the 'data' value
$eventParameterNames = [];
$eventParameterTypes = [];
$eventIndexedParameterNames = [];
$eventIndexedParameterTypes = [];
foreach ($this->events[$eventName]['inputs'] as $input) {
if ($input['indexed']) {
$eventIndexedParameterNames[] = $input['name'];
$eventIndexedParameterTypes[] = $input['type'];
} else {
$eventParameterNames[] = $input['name'];
$eventParameterTypes[] = $input['type'];
}
}
$numEventIndexedParameterNames = count($eventIndexedParameterNames);
//filter through log data to find any logs which match this event (topic) from
//this contract, between these specified blocks (defaulting to the latest block only)
$this->eth->getLogs([
'fromBlock' => (is_int($fromBlock)) ? '0x' . dechex($fromBlock) : $fromBlock,
'toBlock' => (is_int($toBlock)) ? '0x' . dechex($toBlock) : $toBlock,
'topics' => [$this->ethabi->encodeEventSignature($this->events[$eventName])],
'address' => $this->toAddress
],
function ($error, $result) use (&$eventLogData, $eventParameterTypes, $eventParameterNames, $eventIndexedParameterTypes, $eventIndexedParameterNames) {
if ($error !== null) {
throw new RuntimeException($error->getMessage());
}
foreach ($result as $object) {
//decode the data from the log into the expected formats, with its corresponding named key
$decodedData = array_combine($eventParameterNames, $this->ethabi->decodeParameters($eventParameterTypes, $object->data));
//decode the indexed parameter data
for ($i = 0; $i < $numEventIndexedParameterNames; $i++) {
//topics[0] is the event signature, so we start from $i + 1 for the indexed parameter data
$decodedData[$eventIndexedParameterNames[$i]] = $this->ethabi->decodeParameters([$eventIndexedParameterTypes[$i]], $object->topics[$i + 1])[0];
}
//include block metadata for context, along with event data
$eventLogData[] = [
'transactionHash' => $object->transactionHash,
'blockHash' => $object->blockHash,
'blockNumber' => hexdec($object->blockNumber),
'data' => $decodedData
];
}
});
return $eventLogData;
}
} }

View File

@ -102,15 +102,15 @@ class Ethabi
* encodeEventSignature * encodeEventSignature
* TODO: Fix same event name with different params * TODO: Fix same event name with different params
* *
* @param string|stdClass|array $functionName * @param string|stdClass|array $eventName
* @return string * @return string
*/ */
public function encodeEventSignature($functionName) public function encodeEventSignature($eventName)
{ {
if (!is_string($functionName)) { if (!is_string($eventName)) {
$functionName = Utils::jsonMethodToString($functionName); $eventName = Utils::jsonMethodToString($eventName);
} }
return Utils::sha3($functionName); return Utils::sha3($eventName);
} }
/** /**