Compare commits

...

105 Commits
v0.1 ... master

Author SHA1 Message Date
Peter Lai
cede0cc410
Merge pull request #264 from srdante/patch-1
Fix Test Caching
2022-03-06 20:41:49 -08:00
Dante
1d2581e78e
Update php.yml 2022-03-07 00:34:50 -03:00
Peter Lai
2eafffbbe5
Merge pull request #257 from cryptounifier/php-80-test-passing
Support PHP 8.0
2022-03-06 19:26:51 -08:00
Dante
cc7d01e7eb
Add Dockerfile-80 2022-02-26 03:44:06 -03:00
srdante
1d47285b8b Fix preg match & testings 2022-02-26 03:15:44 -03:00
srdante
7c46919e7d Bump composer dependencies 2022-02-26 02:48:12 -03:00
srdante
91f4d2c40b Set setUp() return type 2022-02-26 02:43:32 -03:00
Dante
2ae840f538
Update TestCase.php 2022-02-26 02:40:01 -03:00
Dante
217851c62d
Update php.yml 2022-02-26 02:33:59 -03:00
Dante
7eb8a46366
Update php.yml 2022-02-26 02:28:17 -03:00
Dante
dabb7262b5
Update composer.json 2022-02-26 02:27:56 -03:00
sc0Vu
bd40f1946a
Update package to web3p/web3.php 2022-01-15 00:26:41 +08:00
Peter Lai
fb94ea92c5
Merge pull request #205 from harensarma/patch-2
Removed space from versions of guzzlehttp
2021-08-14 20:28:10 -07:00
harensarma
199cb1d175
Removed space from versions of guzzlehttp
Removed space from versions of guzzlehttp, else second version was not working in laravel
2021-04-18 21:07:09 +05:30
Peter Lai
1b093c8ad4
Merge pull request #198 from web3p/github-ci
Add github ci
2021-02-25 15:24:56 +08:00
sc0Vu
a96e8865a8 Update README.md 2021-02-25 15:23:05 +08:00
Peter Lai
d1582e3a7c Add github ci 2021-02-25 15:22:49 +08:00
sc0vu
9ed1b0d2dd
Add contribution in README.md 2021-02-22 13:14:08 +08:00
Peter Lai
17472b55fd
Merge pull request #191 from serderovsh/patch-1
Update composer.json
2020-11-14 17:15:11 +00:00
Peter Lai
8ba41a37b7
Merge branch 'master' into patch-1 2020-11-14 17:14:48 +00:00
Peter Lai
c0c57219d7
Merge pull request #195 from web3p/minor-update
Minor update
2020-11-14 17:12:21 +00:00
sc0vu
d4f747606f
Update codecov/README.md 2020-11-14 23:33:04 +08:00
sc0vu
292b579337
Test for php ~7.4 2020-11-14 22:31:10 +08:00
sc0vu
dc66eb1064
Update guzzlehttp to 7 2020-11-14 22:28:58 +08:00
Shamsudin
4e0548a3ea
Update composer.json 2020-10-13 07:45:01 +03:00
Peter Lai
d69ce96a64
Merge pull request #158 from sc0Vu/fix-150
Add toChecksumAddress in Utils.php
2019-10-03 22:41:19 +08:00
sc0vu
d0289506dc
Add toChecksumAddress in Utils.php
See #150 (https://github.com/ethereum/EIPs/blob/master/EIPS/eip-55.md)
2019-10-03 00:01:35 +08:00
Peter Lai
0659ec04dc
Merge pull request #157 from sc0Vu/add-defaultblock
Add defaultblock in Contract.call
2019-10-02 10:42:08 +08:00
sc0vu
0e303787ba
Add defaultblock in contract call 2019-10-02 10:26:41 +08:00
Peter Lai
fff321de37
Merge pull request #156 from sc0Vu/add-dockerfile
Add dockerfile
2019-09-08 23:41:37 +08:00
sc0vu
5e284d0037
Update .travis.yml 2019-09-08 23:12:58 +08:00
sc0vu
3943f00b38
Update php dockerfiles
Change:
* Add Dockerfile-73
* Install gmp library automatically
2019-09-08 23:11:27 +08:00
Peter Lai
8a34582d23
Merge pull request #154 from tomcyr/master
Fix call Eth NewBlockFilter
2019-07-30 11:48:05 +08:00
Tomasz Cyrankowski
d67367a2f2 Eth NewBlockFilter method hasn't any arguments so it shouldn't have any validators 2019-07-25 10:52:57 +02:00
sc0Vu
b6fc44d9e3
Update composer.json
Require ext-mbstring
2019-03-12 22:50:29 +08:00
sc0vu
10b000d3e5 Update docker/php/composer-setup.php 2019-02-26 18:50:22 +08:00
Peter Lai
08a237df37
Merge pull request #144 from sc0Vu/fix-143
Fix #143
2019-02-24 17:01:47 +08:00
sc0Vu
4779c7e710
Fix #143
Fix wrong $argsLen in contract.
2019-02-24 16:57:26 +08:00
Peter Lai
13f6d3338d
Merge pull request #142 from sc0Vu/fix-134
Fix #134
2019-02-11 22:15:57 +08:00
sc0Vu
2b5295533f
Add support for same function name but with different parameters
See issue #134
2019-02-11 22:11:29 +08:00
sc0Vu
cd1bcf3a17 Add test contracts 2019-02-11 22:08:09 +08:00
sc0Vu
d35ecef5f7
Fix bytes and dynamic bytes format 2019-02-11 00:10:14 +08:00
Peter Lai
b0615caf03
Merge pull request #138 from sc0Vu/fix-dynamic-bytes
Fix DynamicBytes.php
2019-01-17 01:08:50 +08:00
Peter Lai
5e4e5e35fd
Merge pull request #135 from 1099511627776/bclounge
Close connection after stream read
2019-01-17 01:08:34 +08:00
sc0Vu
fcb959dec7
Fix DynamicBytes.php
Call inputFormat failed when toBn return array of data
See: https://github.com/sc0Vu/web3.php/pull/97
2019-01-17 00:57:08 +08:00
1099511627776
b0b1841642 Fix indent 2019-01-13 12:24:06 +02:00
Peter Lai
fbf2ba4f97
Merge pull request #137 from sc0Vu/fix-112
Fix #112
2019-01-13 15:09:26 +08:00
sc0Vu
83188b171e
Fix #112
Remove depth for jsonToArray function
2019-01-13 14:07:31 +08:00
Peter Lai
c5fe156d97
Merge pull request #136 from sc0Vu/fix-94
Remove support for int and float in toWei
2019-01-12 23:59:47 +08:00
sc0Vu
734dbebb76
Remove support for int and float in toWei 2019-01-06 23:26:21 +08:00
sc0Vu
9dd5056790
Remove comments 2018-12-21 00:19:56 +08:00
Peter Lai
53b5165062
Merge pull request #104 from sinabs/master
getTransactionReceipt result maybe null
2018-12-21 00:14:15 +08:00
Peter Lai
08956500b0
Merge pull request #127 from sc0Vu/fix-125
Fix #125
2018-12-21 00:09:15 +08:00
sc0Vu
0ef6000be3 Fix #125 2018-12-21 00:03:38 +08:00
sc0Vu
51abbc2f5e Clean code 2018-12-14 12:12:04 +08:00
Peter Lai
d579535485
Merge pull request #131 from sc0Vu/add-test
Add test for personal_lockAccount
2018-12-13 18:35:48 +09:00
sc0Vu
f3259100ec Add test for personal_lockAccount
Change:
* Add test for personal_lockAccount
2018-12-13 17:27:52 +08:00
Peter Lai
28627c5ecf
Merge pull request #100 from soraclub/master
Add method: personal_lockAccount
2018-12-13 18:18:54 +09:00
Peter Lai
46708ab7b1
Merge pull request #130 from sc0Vu/update-docker
Update docker
2018-12-13 12:53:41 +09:00
sc0Vu
60bbb2c17a Update docker
Change:
* Add default configuration files.
* Update composer-setup.php.
* Remove composer-setup.php after intallation.
2018-12-13 11:18:48 +08:00
Peter Lai
77f20738aa
Merge pull request #108 from sunshuzhou/patch-1
Updata README.md
2018-12-12 19:22:27 +09:00
Peter Lai
fe5dc8ed63
Merge pull request #128 from sc0Vu/remove-get-compilers
Remove eth_getCompilers
2018-12-12 19:17:15 +09:00
sc0Vu
0d415ce360 Remove eth_getCompilers
Remove deprecated api: eth_getCompilers

See: https://github.com/ethereum/EIPs/issues/209
2018-12-12 16:46:58 +08:00
Peter Lai
01a70408ac
Merge pull request #106 from Arul-/patch-3
Fix for PHP Notice:  Undefined index: inputs in src/Contract.php:394
2018-12-12 00:09:06 +09:00
Peter Lai
e7e04c542b
Merge pull request #105 from amateescu/patch-1
Don't send an empty "to" address when estimating gas for the constructor
2018-12-12 00:05:51 +09:00
1099511627776
29ceef6552 exclude .idea from tracked files 2018-11-13 18:30:21 +02:00
1099511627776
54e18048e3 Rename result to stream 2018-11-02 13:40:24 +02:00
1099511627776
70ac87df82 Close connection after stream read 2018-10-30 19:29:03 +02:00
Sun Shuzhou
c81e4ee4e5
Updata README.md
Fix typo
2018-09-19 11:45:46 +08:00
Arul
0f1b9183b0
Fix for PHP Notice: Undefined index: inputs in src/Contract.php on line 394
When creating an instance of the contract with the new method, `$constructor['inputs']` is not available all the time, causing the above notice

This commit fixes it by checking it first
2018-09-04 21:04:39 +08:00
Andrei Mateescu
908d841bce
Don't send an empty "to" address when estimating gas for the constructor
Sending a empty "to" address when estimating the gas usage of the contract constructor returns the following error (from geth at least):

> invalid argument 0: hex string has length 0, want 40 for common.Address in
2018-08-23 08:31:18 +03:00
sinabs
6d59d3d372 Test transaction result maybe null 2018-08-08 14:59:25 +08:00
sinabs
1a9784da30 add code comments 2018-08-08 14:22:02 +08:00
sinabs
e5729b757c getTransactionReceipt result is NULL, Transaction status Pending
but RequestManage class throw 'Something wrong happened'
2018-08-08 12:52:09 +08:00
guanzi
16d341455d add method: personal_lockAccount 2018-08-01 19:32:44 +08:00
Kuan-Cheng,Lai
1fb7762eb7
Merge pull request #76 from sciku1/master
Added getToAddress
2018-06-24 22:45:20 +08:00
Kuan-Cheng,Lai
87dbe235a4
Merge pull request #87 from sc0Vu/fix-85
decodeParameters not work
2018-06-24 22:42:35 +08:00
sc0Vu
234b1e695f Fix failed test on travis
* Change gas limt to 6000000
* Change ganache container compose file
2018-06-24 22:30:34 +08:00
sc0Vu
07f730d8ce Fix decode dynamic bytes
* Fix outputFormat function in dynamic bytes
* Add test test for issue 85 #85
2018-06-24 21:59:08 +08:00
sc0Vu
d5c21afec0 Fix decode bytes
* Return non zero bytes.
  Decode bytes1 data: 0x6300000000000000000000000000000000000000000000000000000000000000
  Will return 0x63
2018-06-24 21:16:12 +08:00
sc0Vu
ecafb1c5ec Fix BytesTypeTest and DynamicBytesTypeTest 2018-06-24 18:30:45 +08:00
sc0Vu
ee9855fcbf
Change isType regex 2018-06-18 23:24:29 +08:00
Kuan-Cheng,Lai
d0559b8c91
Merge pull request #91 from Arul-/patch-2
Fix: method validation in both send and call
2018-06-14 10:18:32 +08:00
Arul
3d8a567054
Fix: method validation in both send and call
Making sure method is a string and it exists under functions

I was trying to call a nonexistent method. Since I passed a string it went through without checking if it exists as a function or not

This commit fixes that by changing `&&` to `||` (or)
2018-06-07 18:02:08 +08:00
Kuan-Cheng,Lai
be7403a00b
Merge pull request #89 from a10000005588/fix-personal-api-test
Change $e to $err
2018-06-04 11:23:28 +08:00
will
80c8c783ae Change $e to $err 2018-06-04 11:10:31 +08:00
sc0Vu
9521b2d167 Check output name is empty 2018-05-28 15:21:18 +08:00
Kuan-Cheng,Lai
5d04cec73f
Merge pull request #86 from sc0Vu/fix-83
Fix #83
2018-05-27 23:01:46 +08:00
sc0Vu
4fd5d1ab67 Fix #83
Return $fractionLength when number is with decimal point.
2018-05-27 22:48:02 +08:00
sc0Vu
4eaebd48aa Add test for issue 71 2018-05-18 12:03:27 +08:00
Kuan-Cheng,Lai
831d2ebdbe
Merge pull request #77 from lsmelan/lsmelan-patch-1
Fix: issue #71
2018-05-18 12:01:26 +08:00
Luiz Smelán
5a54ce5903
Fix: issue #71
- Not to limit the size of the string to 64 characters
2018-05-02 17:14:53 +01:00
Timothy Lopez
85dc060017 Added getToAddress 2018-04-27 18:20:06 -06:00
sc0Vu
02be3f815d Add ganache container
Add ganache container
Update 'README.md'
2018-04-25 10:25:09 +08:00
sc0Vu
233379966b Merge branch 'fix-62' 2018-04-25 10:09:04 +08:00
sc0Vu
a87010fc4c Add test
Add test for toHex(48), thanks @refear99
2018-04-25 10:07:06 +08:00
sc0Vu
fa95650437 Update composer.json 2018-04-25 10:06:30 +08:00
sc0Vu
6100a046c5 Update 'README.md' 2018-04-14 15:01:45 +08:00
Kuan-Cheng,Lai
00817ee6b9
Merge pull request #60 from pash7ka/dev-patch-new-contract
Remove `to` parameter when publishing contract transaction
2018-04-08 22:41:32 +08:00
Pavel Rubin
f97d4c9397 Remove to parameter when publishing contract transaction 2018-04-03 07:04:47 +03:00
Kuan-Cheng,Lai
4df0835f2b
Merge pull request #58 from sc0Vu/specify-package-version
Specify package version
2018-03-31 10:34:15 +08:00
sc0Vu
a2a68dd516 Fix undefined pow function in phpseclib 2.0 2018-03-31 10:28:45 +08:00
sc0Vu
04e9c14f5e Specify package version
1. Remove dev minimum stability
2. kornrunner/keccak ~1.0
3. phpseclib/phpseclib ~2.0
2018-03-31 10:26:29 +08:00
sc0Vu
5ede0c03e6 return to exit 2018-03-19 10:13:00 +08:00
sc0Vu
209baec4b7 Return test script status. 2018-03-19 10:08:58 +08:00
88 changed files with 5543 additions and 399 deletions

49
.github/workflows/php.yml vendored Normal file
View File

@ -0,0 +1,49 @@
name: PHP
on: ["push", "pull_request", "workflow_dispatch"]
jobs:
build_and_test:
name: Build and test web3.php with ${{ matrix.php-version }}
strategy:
matrix:
php-version: ["7.3", "7.4", "8.0"]
runs-on: ubuntu-latest
steps:
- name: Setup PHP
uses: shivammathur/setup-php@v2
with:
php-version: ${{ matrix.php-version }}
- name: PHP version
run: |
php --version
- name: Install nodejs and ganache-cli
run: |
sudo apt-get update -y
sudo apt-get install -y nodejs
sudo npm install -g ganache-cli
- uses: actions/checkout@v2
- name: Validate composer.json and composer.lock
run: composer validate
- name: Cache Composer packages
id: composer-cache
uses: actions/cache@v2
with:
path: vendor
key: ${{ runner.os }}-php-${{ matrix.php-version }}-${{ hashFiles('**/composer.lock') }}
restore-keys: |
${{ runner.os }}-php-${{ matrix.php-version }}-
- name: Install dependencies
if: steps.composer-cache.outputs.cache-hit != 'true'
run: composer install --prefer-dist --no-progress --no-suggest
- name: Run test suite
run: sudo scripts/test.sh

2
.gitignore vendored
View File

@ -1,6 +1,8 @@
composer.phar composer.phar
/vendor/ /vendor/
.phpintel/ .phpintel/
/.idea/
# Commit your application's lock file http://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file # Commit your application's lock file http://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file
# You may choose to ignore a library lock file http://getcomposer.org/doc/02-libraries.md#lock-file # You may choose to ignore a library lock file http://getcomposer.org/doc/02-libraries.md#lock-file

View File

@ -5,6 +5,8 @@ language: php
php: php:
- 7.1 - 7.1
- 7.2 - 7.2
- 7.3
- 7.4
before_install: before_install:
- nvm install 8 - nvm install 8
@ -24,4 +26,4 @@ script:
- scripts/test.sh - scripts/test.sh
after_success: after_success:
- bash <(curl -s https://codecov.io/bash) -t 8436848f-9a99-4b49-be60-629d7ff9a62c - bash <(curl -s https://codecov.io/bash) -t 76a0abaf-bbc4-4829-b5b8-474f10a6b9e9

View File

@ -1,9 +1,10 @@
# web3.php # web3.php
[![Build Status](https://travis-ci.org/sc0Vu/web3.php.svg?branch=master)](https://travis-ci.org/sc0Vu/web3.php) [![PHP](https://github.com/web3p/web3.php/actions/workflows/php.yml/badge.svg)](https://github.com/web3p/web3.php/actions/workflows/php.yml)
[![codecov](https://codecov.io/gh/sc0Vu/web3.php/branch/master/graph/badge.svg)](https://codecov.io/gh/sc0Vu/web3.php) [![Build Status](https://travis-ci.org/web3p/web3.php.svg?branch=master)](https://travis-ci.org/web3p/web3.php)
[![codecov](https://codecov.io/gh/web3p/web3.php/branch/master/graph/badge.svg)](https://codecov.io/gh/web3p/web3.php)
[![Join the chat at https://gitter.im/web3-php/web3.php](https://img.shields.io/badge/gitter-join%20chat-brightgreen.svg)](https://gitter.im/web3-php/web3.php) [![Join the chat at https://gitter.im/web3-php/web3.php](https://img.shields.io/badge/gitter-join%20chat-brightgreen.svg)](https://gitter.im/web3-php/web3.php)
[![Licensed under the MIT License](https://img.shields.io/badge/License-MIT-blue.svg)](https://github.com/sc0Vu/web3.php/blob/master/LICENSE) [![Licensed under the MIT License](https://img.shields.io/badge/License-MIT-blue.svg)](https://github.com/web3p/web3.php/blob/master/LICENSE)
A php interface for interacting with the Ethereum blockchain and ecosystem. A php interface for interacting with the Ethereum blockchain and ecosystem.
@ -17,13 +18,13 @@ Set minimum stability to dev
Then Then
``` ```
composer require sc0vu/web3.php dev-master composer require web3p/web3.php dev-master
``` ```
Or you can add this line in composer.json Or you can add this line in composer.json
``` ```
"sc0vu/web3.php": "dev-master" "web3p/web3.php": "dev-master"
``` ```
@ -55,7 +56,7 @@ $web3->clientVersion(function ($err, $version) {
// do something // do something
return; return;
} }
if (isset($client)) { if (isset($version)) {
echo 'Client version: ' . $version; echo 'Client version: ' . $version;
} }
}); });
@ -216,7 +217,7 @@ If you are using docker as development machain, you can try [ethdock](https://gi
1. Clone the repo and install packages. 1. Clone the repo and install packages.
``` ```
git clone https://github.com/sc0Vu/web3.php.git && cd web3.php && composer install git clone https://github.com/web3p/web3.php.git && cd web3.php && composer install
``` ```
2. Run test script. 2. Run test script.
@ -228,12 +229,12 @@ vendor/bin/phpunit
1. Clone the repo and run docker container. 1. Clone the repo and run docker container.
``` ```
git clone https://github.com/sc0Vu/web3.php.git git clone https://github.com/web3p/web3.php.git
``` ```
2. Copy web3.php to web3.php/docker/app directory and start container. 2. Copy web3.php to web3.php/docker/app directory and start container.
``` ```
cp files docker/app && docker-compose up -d php cp files docker/app && docker-compose up -d php ganache
``` ```
3. Enter php container and install packages. 3. Enter php container and install packages.
@ -241,14 +242,54 @@ cp files docker/app && docker-compose up -d php
docker-compose exec php ash docker-compose exec php ash
``` ```
4. Run test script 4. Change testHost in `TestCase.php`
```
/**
* testHost
*
* @var string
*/
protected $testHost = 'http://ganache:8545';
```
5. Run test script
``` ```
vendor/bin/phpunit vendor/bin/phpunit
``` ```
###### Install packages
Enter container first
```
docker-compose exec php ash
```
1. gmp
```
apk add gmp-dev
docker-php-ext-install gmp
```
2. bcmath
```
docker-php-ext-install bcmath
```
###### Remove extension
Move the extension config from `/usr/local/etc/php/conf.d/`
```
mv /usr/local/etc/php/conf.d/extension-config-name to/directory
```
# API # API
Todo. Todo.
# Contribution
Thank you to all the people who already contributed to web3.php!
<a href="https://github.com/web3p/web3.php/graphs/contributors">
<img src="https://contrib.rocks/image?repo=web3p/web3.php" />
</a>
# License # License
MIT MIT

View File

@ -1,5 +1,5 @@
{ {
"name": "sc0vu/web3.php", "name": "web3p/web3.php",
"description": "Ethereum web3 interface.", "description": "Ethereum web3 interface.",
"type": "library", "type": "library",
"license": "MIT", "license": "MIT",
@ -9,15 +9,15 @@
"email": "alk03073135@gmail.com" "email": "alk03073135@gmail.com"
} }
], ],
"minimum-stability": "dev",
"require": { "require": {
"guzzlehttp/guzzle": "~6.0", "guzzlehttp/guzzle": "^6.3|^7.0",
"PHP": "^7.1", "PHP": "^7.2|^8.0",
"kornrunner/keccak": "dev-master", "kornrunner/keccak": "~1.0",
"phpseclib/phpseclib": "dev-master" "phpseclib/phpseclib": "~2.0.30",
"ext-mbstring": "*"
}, },
"require-dev": { "require-dev": {
"phpunit/phpunit": "~6.0" "phpunit/phpunit": "~8.0|~9.0"
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {

View File

@ -9,3 +9,10 @@ services:
volumes: volumes:
- ./app:/app - ./app:/app
tty: true tty: true
ganache:
build:
context: ./ganache
dockerfile: Dockerfile
ports:
- "8545"

View File

@ -0,0 +1,9 @@
FROM node:9.11.1-alpine
MAINTAINER Peter Lai <alk03073135@gmail.com>
RUN npm install -g ganache-cli
EXPOSE 8545
CMD ganache-cli -g 0 -l 6000000 --hostname=0.0.0.0

View File

@ -1,17 +1,23 @@
FROM php:7.1.14-alpine FROM php:7.1-alpine
MAINTAINER Peter Lai <alk03073135@gmail.com> MAINTAINER Peter Lai <alk03073135@gmail.com>
ADD composer-setup.php composer-setup.php COPY composer-setup.php composer-setup.php
# COPY php.ini-production $PHP_INI_DIR/php.ini
RUN apk update && \ RUN apk update && \
apk add git apk add git
# Install gmp
Run apk add gmp-dev && \
docker-php-ext-install gmp
# Install nodejs # Install nodejs
# Run apk add --update nodejs nodejs-npm # Run apk add --update nodejs nodejs-npm
# Install composer # Install composer
RUN php composer-setup.php && \ RUN php composer-setup.php && \
php composer-setup.php --install-dir=/usr/bin --filename=composer php composer-setup.php --install-dir=/usr/bin --filename=composer && \
php -r "unlink('composer-setup.php');"
WORKDIR /app WORKDIR /app

View File

@ -1,17 +1,23 @@
FROM php:7.2.2-alpine FROM php:7.2-alpine
MAINTAINER Peter Lai <alk03073135@gmail.com> MAINTAINER Peter Lai <alk03073135@gmail.com>
ADD composer-setup.php composer-setup.php COPY composer-setup.php composer-setup.php
# COPY php.ini-production $PHP_INI_DIR/php.ini
RUN apk update && \ RUN apk update && \
apk add git apk add git
# Install gmp
Run apk add gmp-dev && \
docker-php-ext-install gmp
# Install nodejs # Install nodejs
# Run apk add --update nodejs nodejs-npm # Run apk add --update nodejs nodejs-npm
# Install composer # Install composer
RUN php composer-setup.php && \ RUN php composer-setup.php && \
php composer-setup.php --install-dir=/usr/bin --filename=composer php composer-setup.php --install-dir=/usr/bin --filename=composer && \
php -r "unlink('composer-setup.php');"
WORKDIR /app WORKDIR /app

23
docker/php/Dockerfile-73 Normal file
View File

@ -0,0 +1,23 @@
FROM php:7.3-alpine
MAINTAINER Peter Lai <alk03073135@gmail.com>
COPY composer-setup.php composer-setup.php
# COPY php.ini-production $PHP_INI_DIR/php.ini
RUN apk update && \
apk add git
# Install gmp
Run apk add gmp-dev && \
docker-php-ext-install gmp
# Install nodejs
# Run apk add --update nodejs nodejs-npm
# Install composer
RUN php composer-setup.php && \
php composer-setup.php --install-dir=/usr/bin --filename=composer && \
php -r "unlink('composer-setup.php');"
WORKDIR /app

23
docker/php/Dockerfile-74 Normal file
View File

@ -0,0 +1,23 @@
FROM php:7.4-alpine
MAINTAINER Peter Lai <alk03073135@gmail.com>
COPY composer-setup.php composer-setup.php
# COPY php.ini-production $PHP_INI_DIR/php.ini
RUN apk update && \
apk add git
# Install gmp
Run apk add gmp-dev && \
docker-php-ext-install gmp
# Install nodejs
# Run apk add --update nodejs nodejs-npm
# Install composer
RUN php composer-setup.php && \
php composer-setup.php --install-dir=/usr/bin --filename=composer && \
php -r "unlink('composer-setup.php');"
WORKDIR /app

23
docker/php/Dockerfile-80 Normal file
View File

@ -0,0 +1,23 @@
FROM php:8.0-alpine
MAINTAINER Peter Lai <alk03073135@gmail.com>
COPY composer-setup.php composer-setup.php
# COPY php.ini-production $PHP_INI_DIR/php.ini
RUN apk update && \
apk add git
# Install gmp
Run apk add gmp-dev && \
docker-php-ext-install gmp
# Install nodejs
# Run apk add --update nodejs nodejs-npm
# Install composer
RUN php composer-setup.php && \
php composer-setup.php --install-dir=/usr/bin --filename=composer && \
php -r "unlink('composer-setup.php');"
WORKDIR /app

View File

@ -1,5 +1,8 @@
<?php <?php
$sig = file_get_contents('https://composer.github.io/installer.sig');
$sig = trim($sig);
copy('https://getcomposer.org/installer', 'composer-setup.php'); copy('https://getcomposer.org/installer', 'composer-setup.php');
if (hash_file('SHA384', 'composer-setup.php') === '544e09ee996cdf60ece3804abc52599c22b1f40f4323403c44d44fdfdd586475ca9813a858088ffbc1f233e9b180f061') { echo 'Installer verified'; } else { echo 'Installer corrupt'; unlink('composer-setup.php'); } echo PHP_EOL; if (hash_file('sha384', 'composer-setup.php') === $sig) { echo 'Installer verified'; } else { echo 'Installer corrupt'; unlink('composer-setup.php'); } echo PHP_EOL;

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -38,3 +38,16 @@ $web3->eth->getBalance($newAccount, function ($err, $balance) {
} }
echo 'Balance: ' . $balance->toString() . PHP_EOL; echo 'Balance: ' . $balance->toString() . PHP_EOL;
}); });
// remember to lock account after transaction
$personal->lockAccount($newAccount, function ($err, $locked) {
if ($err !== null) {
echo 'Error: ' . $err->getMessage();
return;
}
if ($locked) {
echo 'New account is locked!' . PHP_EOL;
} else {
echo 'New account isn\'t locked' . PHP_EOL;
}
});

View File

@ -7,8 +7,7 @@
convertNoticesToExceptions="true" convertNoticesToExceptions="true"
convertWarningsToExceptions="true" convertWarningsToExceptions="true"
processIsolation="false" processIsolation="false"
stopOnFailure="false" stopOnFailure="false">
syntaxCheck="false">
<testsuite name="Web3.php unit test"> <testsuite name="Web3.php unit test">
<directory suffix="Test.php">./test/unit</directory> <directory suffix="Test.php">./test/unit</directory>

View File

@ -1,12 +1,15 @@
#!/usr/bin/env bash #!/usr/bin/env bash
ganache-cli -g 0 -l 0 > /dev/null & ganache-cli -g 0 -l 6000000 > /dev/null &
ganachecli_pid=$! ganachecli_pid=$!
echo "Start ganache-cli pid: $ganachecli_pid and sleep 3 seconds" echo "Start ganache-cli pid: $ganachecli_pid and sleep 3 seconds"
sleep 3 sleep 3
vendor/bin/phpunit --coverage-clover=coverage.xml vendor/bin/phpunit --coverage-clover=coverage.xml
ret=$?
kill -9 $ganachecli_pid kill -9 $ganachecli_pid
echo "Kill ganache-cli" echo "Kill ganache-cli"
exit $ret

View File

@ -22,13 +22,16 @@ use Web3\Contracts\Ethabi;
use Web3\Contracts\Types\Address; use Web3\Contracts\Types\Address;
use Web3\Contracts\Types\Boolean; use Web3\Contracts\Types\Boolean;
use Web3\Contracts\Types\Bytes; use Web3\Contracts\Types\Bytes;
use Web3\Contracts\Types\DynamicBytes;
use Web3\Contracts\Types\Integer; use Web3\Contracts\Types\Integer;
use Web3\Contracts\Types\Str; use Web3\Contracts\Types\Str;
use Web3\Contracts\Types\Uinteger; use Web3\Contracts\Types\Uinteger;
use Web3\Validators\AddressValidator; use Web3\Validators\AddressValidator;
use Web3\Validators\HexValidator; use Web3\Validators\HexValidator;
use Web3\Formatters\AddressFormatter;
use Web3\Validators\StringValidator; use Web3\Validators\StringValidator;
use Web3\Validators\TagValidator;
use Web3\Validators\QuantityValidator;
use Web3\Formatters\AddressFormatter;
class Contract class Contract
{ {
@ -95,14 +98,22 @@ class Contract
*/ */
protected $ethabi; protected $ethabi;
/**
* defaultBlock
*
* @var mixed
*/
protected $defaultBlock;
/** /**
* construct * construct
* *
* @param string|\Web3\Providers\Provider $provider * @param string|\Web3\Providers\Provider $provider
* @param string|\stdClass|array $abi * @param string|\stdClass|array $abi
* @param mixed $defaultBlock
* @return void * @return void
*/ */
public function __construct($provider, $abi) public function __construct($provider, $abi, $defaultBlock = 'latest')
{ {
if (is_string($provider) && (filter_var($provider, FILTER_VALIDATE_URL) !== false)) { if (is_string($provider) && (filter_var($provider, FILTER_VALIDATE_URL) !== false)) {
// check the uri schema // check the uri schema
@ -114,12 +125,21 @@ class Contract
} else if ($provider instanceof Provider) { } else if ($provider instanceof Provider) {
$this->provider = $provider; $this->provider = $provider;
} }
$abi = Utils::jsonToArray($abi, 5);
foreach ($abi as $item) { $abiArray = [];
if (is_string($abi)) {
$abiArray = json_decode($abi, true);
if (JSON_ERROR_NONE !== json_last_error()) {
throw new InvalidArgumentException('abi decode error: ' . json_last_error_msg());
}
} else {
$abiArray = Utils::jsonToArray($abi);
}
foreach ($abiArray as $item) {
if (isset($item['type'])) { if (isset($item['type'])) {
if ($item['type'] === 'function') { if ($item['type'] === 'function') {
$this->functions[$item['name']] = $item; $this->functions[] = $item;
} elseif ($item['type'] === 'constructor') { } elseif ($item['type'] === 'constructor') {
$this->constructor = $item; $this->constructor = $item;
} elseif ($item['type'] === 'event') { } elseif ($item['type'] === 'event') {
@ -127,12 +147,18 @@ class Contract
} }
} }
} }
$this->abi = $abi; if (TagValidator::validate($defaultBlock) || QuantityValidator::validate($defaultBlock)) {
$this->defaultBlock = $defaultBlock;
} else {
$this->$defaultBlock = 'latest';
}
$this->abi = $abiArray;
$this->eth = new Eth($this->provider); $this->eth = new Eth($this->provider);
$this->ethabi = new Ethabi([ $this->ethabi = new Ethabi([
'address' => new Address, 'address' => new Address,
'bool' => new Boolean, 'bool' => new Boolean,
'bytes' => new Bytes, 'bytes' => new Bytes,
'dynamicBytes' => new DynamicBytes,
'int' => new Integer, 'int' => new Integer,
'string' => new Str, 'string' => new Str,
'uint' => new Uinteger, 'uint' => new Uinteger,
@ -213,6 +239,32 @@ class Contract
return $this; return $this;
} }
/**
* getDefaultBlock
*
* @return string
*/
public function getDefaultBlock()
{
return $this->defaultBlock;
}
/**
* setDefaultBlock
*
* @param mixed $defaultBlock
* @return $this
*/
public function setDefaultBlock($defaultBlock)
{
if (TagValidator::validate($defaultBlock) || QuantityValidator::validate($defaultBlock)) {
$this->defaultBlock = $defaultBlock;
} else {
$this->$defaultBlock = 'latest';
}
return $this;
}
/** /**
* getFunctions * getFunctions
* *
@ -233,6 +285,14 @@ class Contract
return $this->events; return $this->events;
} }
/**
* @return string
*/
public function getToAddress()
{
return $this->toAddress;
}
/** /**
* getConstructor * getConstructor
* *
@ -349,12 +409,21 @@ class Contract
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 abi is valid.');
} }
$abi = Utils::jsonToArray($abi, 5); $abiArray = [];
if (is_string($abi)) {
$abiArray = json_decode($abi, true);
foreach ($abi as $item) { if (JSON_ERROR_NONE !== json_last_error()) {
throw new InvalidArgumentException('abi decode error: ' . json_last_error_msg());
}
} else {
$abiArray = Utils::jsonToArray($abi);
}
foreach ($abiArray as $item) {
if (isset($item['type'])) { if (isset($item['type'])) {
if ($item['type'] === 'function') { if ($item['type'] === 'function') {
$this->functions[$item['name']] = $item; $this->functions[] = $item;
} elseif ($item['type'] === 'constructor') { } elseif ($item['type'] === 'constructor') {
$this->constructor = $item; $this->constructor = $item;
} elseif ($item['type'] === 'event') { } elseif ($item['type'] === 'event') {
@ -362,7 +431,7 @@ class Contract
} }
} }
} }
$this->abi = $abi; $this->abi = $abiArray;
return $this; return $this;
} }
@ -381,7 +450,8 @@ class Contract
$arguments = func_get_args(); $arguments = func_get_args();
$callback = array_pop($arguments); $callback = array_pop($arguments);
if (count($arguments) < count($constructor['inputs'])) { $input_count = isset($constructor['inputs']) ? count($constructor['inputs']) : 0;
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 put all constructor params and callback.');
} }
if (is_callable($callback) !== true) { if (is_callable($callback) !== true) {
@ -390,14 +460,13 @@ class Contract
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, count($constructor['inputs'])); $params = array_splice($arguments, 0, $input_count);
$data = $this->ethabi->encodeParameters($constructor, $params); $data = $this->ethabi->encodeParameters($constructor, $params);
$transaction = []; $transaction = [];
if (count($arguments) > 0) { if (count($arguments) > 0) {
$transaction = $arguments[0]; $transaction = $arguments[0];
} }
$transaction['to'] = '';
$transaction['data'] = '0x' . $this->bytecode . Utils::stripZero($data); $transaction['data'] = '0x' . $this->bytecode . Utils::stripZero($data);
$this->eth->sendTransaction($transaction, function ($err, $transaction) use ($callback){ $this->eth->sendTransaction($transaction, function ($err, $transaction) use ($callback){
@ -423,26 +492,75 @@ class Contract
$method = array_splice($arguments, 0, 1)[0]; $method = array_splice($arguments, 0, 1)[0];
$callback = array_pop($arguments); $callback = array_pop($arguments);
if (!is_string($method) && !isset($this->functions[$method])) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is existed.'); throw new InvalidArgumentException('Please make sure the method is string.');
} }
$function = $this->functions[$method];
if (count($arguments) < count($function['inputs'])) { $functions = [];
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); foreach ($this->functions as $function) {
if ($function["name"] === $method) {
$functions[] = $function;
}
};
if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.');
} }
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 param must be callback function.');
} }
$params = array_splice($arguments, 0, count($function['inputs']));
// check the last one in arguments is transaction object
$argsLen = count($arguments);
$transaction = [];
$hasTransaction = false;
if ($argsLen > 0) {
$transaction = $arguments[$argsLen - 1];
}
if (
isset($transaction["from"]) ||
isset($transaction["to"]) ||
isset($transaction["gas"]) ||
isset($transaction["gasPrice"]) ||
isset($transaction["value"]) ||
isset($transaction["data"]) ||
isset($transaction["nonce"])
) {
$hasTransaction = true;
} else {
$transaction = [];
}
$params = [];
$data = "";
$functionName = "";
foreach ($functions as $function) {
if ($hasTransaction) {
if ($argsLen - 1 !== count($function['inputs'])) {
continue;
} else {
$paramsLen = $argsLen - 1;
}
} else {
if ($argsLen !== count($function['inputs'])) {
continue;
} else {
$paramsLen = $argsLen;
}
}
try {
$params = array_splice($arguments, 0, $paramsLen);
$data = $this->ethabi->encodeParameters($function, $params); $data = $this->ethabi->encodeParameters($function, $params);
$functionName = Utils::jsonMethodToString($function); $functionName = Utils::jsonMethodToString($function);
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName); } catch (InvalidArgumentException $e) {
$transaction = []; continue;
if (count($arguments) > 0) {
$transaction = $arguments[0];
} }
break;
}
if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.');
}
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName);
$transaction['to'] = $this->toAddress; $transaction['to'] = $this->toAddress;
$transaction['data'] = $functionSignature . Utils::stripZero($data); $transaction['data'] = $functionSignature . Utils::stripZero($data);
@ -469,30 +587,78 @@ class Contract
$method = array_splice($arguments, 0, 1)[0]; $method = array_splice($arguments, 0, 1)[0];
$callback = array_pop($arguments); $callback = array_pop($arguments);
if (!is_string($method) && !isset($this->functions[$method])) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is existed.'); throw new InvalidArgumentException('Please make sure the method is string.');
} }
$function = $this->functions[$method];
if (count($arguments) < count($function['inputs'])) { $functions = [];
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); foreach ($this->functions as $function) {
if ($function["name"] === $method) {
$functions[] = $function;
}
};
if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.');
} }
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 param must be callback function.');
} }
$params = array_splice($arguments, 0, count($function['inputs']));
// check the arguments
$argsLen = count($arguments);
$transaction = [];
$defaultBlock = $this->defaultBlock;
$params = [];
$data = "";
$functionName = "";
foreach ($functions as $function) {
try {
$paramsLen = count($function['inputs']);
$params = array_slice($arguments, 0, $paramsLen);
$data = $this->ethabi->encodeParameters($function, $params); $data = $this->ethabi->encodeParameters($function, $params);
$functionName = Utils::jsonMethodToString($function); $functionName = Utils::jsonMethodToString($function);
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName); } catch (InvalidArgumentException $e) {
$transaction = []; continue;
if (count($arguments) > 0) {
$transaction = $arguments[0];
} }
break;
}
if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.');
}
// remove arguments
array_splice($arguments, 0, $paramsLen);
$argsLen -= $paramsLen;
if ($argsLen > 1) {
$defaultBlock = $arguments[$argsLen - 1];
$transaction = $arguments[$argsLen - 2];
} else if ($argsLen > 0) {
if (is_array($arguments[$argsLen - 1])) {
$transaction = $arguments[$argsLen - 1];
} else {
$defaultBlock = $arguments[$argsLen - 1];
}
}
if (!TagValidator::validate($defaultBlock) && !QuantityValidator::validate($defaultBlock)) {
$defaultBlock = $this->defaultBlock;
}
if (
!is_array($transaction) &&
!isset($transaction["from"]) &&
!isset($transaction["to"]) &&
!isset($transaction["gas"]) &&
!isset($transaction["gasPrice"]) &&
!isset($transaction["value"]) &&
!isset($transaction["data"]) &&
!isset($transaction["nonce"])
) {
$transaction = [];
}
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName);
$transaction['to'] = $this->toAddress; $transaction['to'] = $this->toAddress;
$transaction['data'] = $functionSignature . Utils::stripZero($data); $transaction['data'] = $functionSignature . Utils::stripZero($data);
$this->eth->call($transaction, function ($err, $transaction) use ($callback, $function){ $this->eth->call($transaction, $defaultBlock, function ($err, $transaction) use ($callback, $function){
if ($err !== null) { if ($err !== null) {
return call_user_func($callback, $err, null); return call_user_func($callback, $err, null);
} }
@ -535,31 +701,79 @@ class Contract
if (count($arguments) > 0) { if (count($arguments) > 0) {
$transaction = $arguments[0]; $transaction = $arguments[0];
} }
$transaction['to'] = '';
$transaction['data'] = '0x' . $this->bytecode . Utils::stripZero($data); $transaction['data'] = '0x' . $this->bytecode . Utils::stripZero($data);
} else { } else {
$method = array_splice($arguments, 0, 1)[0]; $method = array_splice($arguments, 0, 1)[0];
if (!is_string($method) && !isset($this->functions[$method])) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is existed.'); throw new InvalidArgumentException('Please make sure the method is string.');
} }
$function = $this->functions[$method];
if (count($arguments) < count($function['inputs'])) { $functions = [];
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); foreach ($this->functions as $function) {
if ($function["name"] === $method) {
$functions[] = $function;
}
};
if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.');
} }
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 param must be callback function.');
} }
$params = array_splice($arguments, 0, count($function['inputs']));
// check the last one in arguments is transaction object
$argsLen = count($arguments);
$transaction = [];
$hasTransaction = false;
if ($argsLen > 0) {
$transaction = $arguments[$argsLen - 1];
}
if (
isset($transaction["from"]) ||
isset($transaction["to"]) ||
isset($transaction["gas"]) ||
isset($transaction["gasPrice"]) ||
isset($transaction["value"]) ||
isset($transaction["data"]) ||
isset($transaction["nonce"])
) {
$hasTransaction = true;
} else {
$transaction = [];
}
$params = [];
$data = "";
$functionName = "";
foreach ($functions as $function) {
if ($hasTransaction) {
if ($argsLen - 1 !== count($function['inputs'])) {
continue;
} else {
$paramsLen = $argsLen - 1;
}
} else {
if ($argsLen !== count($function['inputs'])) {
continue;
} else {
$paramsLen = $argsLen;
}
}
try {
$params = array_splice($arguments, 0, $paramsLen);
$data = $this->ethabi->encodeParameters($function, $params); $data = $this->ethabi->encodeParameters($function, $params);
$functionName = Utils::jsonMethodToString($function); $functionName = Utils::jsonMethodToString($function);
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName); } catch (InvalidArgumentException $e) {
$transaction = []; continue;
if (count($arguments) > 0) {
$transaction = $arguments[0];
} }
break;
}
if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.');
}
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName);
$transaction['to'] = $this->toAddress; $transaction['to'] = $this->toAddress;
$transaction['data'] = $functionSignature . Utils::stripZero($data); $transaction['data'] = $functionSignature . Utils::stripZero($data);
} }
@ -605,17 +819,38 @@ class Contract
} else { } else {
$method = array_splice($arguments, 0, 1)[0]; $method = array_splice($arguments, 0, 1)[0];
if (!is_string($method) && !isset($this->functions[$method])) { if (!is_string($method)) {
throw new InvalidArgumentException('Please make sure the method is existed.'); throw new InvalidArgumentException('Please make sure the method is string.');
} }
$function = $this->functions[$method];
if (count($arguments) < count($function['inputs'])) { $functions = [];
throw new InvalidArgumentException('Please make sure you have put all function params and callback.'); foreach ($this->functions as $function) {
if ($function["name"] === $method) {
$functions[] = $function;
} }
$params = array_splice($arguments, 0, count($function['inputs'])); };
if (count($functions) < 1) {
throw new InvalidArgumentException('Please make sure the method exists.');
}
$params = $arguments;
$data = "";
$functionName = "";
foreach ($functions as $function) {
if (count($arguments) !== count($function['inputs'])) {
continue;
}
try {
$data = $this->ethabi->encodeParameters($function, $params); $data = $this->ethabi->encodeParameters($function, $params);
$functionName = Utils::jsonMethodToString($function); $functionName = Utils::jsonMethodToString($function);
} catch (InvalidArgumentException $e) {
continue;
}
break;
}
if (empty($data) || empty($functionName)) {
throw new InvalidArgumentException('Please make sure you have put all function params and callback.');
}
$functionSignature = $this->ethabi->encodeFunctionSignature($functionName); $functionSignature = $this->ethabi->encodeFunctionSignature($functionName);
$functionData = Utils::stripZero($functionSignature) . Utils::stripZero($data); $functionData = Utils::stripZero($functionSignature) . Utils::stripZero($data);
} }

View File

@ -100,6 +100,7 @@ class Ethabi
/** /**
* encodeEventSignature * encodeEventSignature
* TODO: Fix same event name with different params
* *
* @param string|stdClass|array $functionName * @param string|stdClass|array $functionName
* @return string * @return string
@ -155,21 +156,6 @@ class Ethabi
} }
$typesLength = count($types); $typesLength = count($types);
$solidityTypes = $this->getSolidityTypes($types); $solidityTypes = $this->getSolidityTypes($types);
foreach ($types as $key => $type) {
$match = [];
if (preg_match('/^([a-zA-Z]+)/', $type, $match) === 1) {
if (isset($this->types[$match[0]])) {
$className = $this->types[$match[0]];
if (call_user_func([$this->types[$match[0]], 'isType'], $type) === false) {
throw new InvalidArgumentException('Unsupport solidity parameter type: ' . $type);
}
$solidityTypes[$key] = $className;
}
}
}
$encodes = array_fill(0, $typesLength, ''); $encodes = array_fill(0, $typesLength, '');
foreach ($solidityTypes as $key => $type) { foreach ($solidityTypes as $key => $type) {
@ -249,7 +235,7 @@ class Ethabi
$param = mb_strtolower(Utils::stripZero($param)); $param = mb_strtolower(Utils::stripZero($param));
for ($i=0; $i<$typesLength; $i++) { for ($i=0; $i<$typesLength; $i++) {
if (isset($outputTypes['outputs'][$i]['name'])) { if (isset($outputTypes['outputs'][$i]['name']) && empty($outputTypes['outputs'][$i]['name']) === false) {
$result[$outputTypes['outputs'][$i]['name']] = $solidityTypes[$i]->decode($param, $offsets[$i], $types[$i]); $result[$outputTypes['outputs'][$i]['name']] = $solidityTypes[$i]->decode($param, $offsets[$i], $types[$i]);
} else { } else {
$result[$i] = $solidityTypes[$i]->decode($param, $offsets[$i], $types[$i]); $result[$i] = $solidityTypes[$i]->decode($param, $offsets[$i], $types[$i]);
@ -280,8 +266,13 @@ class Ethabi
$className = $this->types[$match[0]]; $className = $this->types[$match[0]];
if (call_user_func([$this->types[$match[0]], 'isType'], $type) === false) { if (call_user_func([$this->types[$match[0]], 'isType'], $type) === false) {
// check dynamic bytes
if ($match[0] === 'bytes') {
$className = $this->types['dynamicBytes'];
} else {
throw new InvalidArgumentException('Unsupport solidity parameter type: ' . $type); throw new InvalidArgumentException('Unsupport solidity parameter type: ' . $type);
} }
}
$solidityTypes[$key] = $className; $solidityTypes[$key] = $className;
} }
} }

View File

@ -37,7 +37,7 @@ class Address extends SolidityType implements IType
*/ */
public function isType($name) public function isType($name)
{ {
return (preg_match('/address(\[([0-9]*)\])*/', $name) === 1); return (preg_match('/^address(\[([0-9]*)\])*$/', $name) === 1);
} }
/** /**

View File

@ -35,7 +35,7 @@ class Boolean extends SolidityType implements IType
*/ */
public function isType($name) public function isType($name)
{ {
return (preg_match('/bool(\[([0-9]*)\])*/', $name) === 1); return (preg_match('/^bool(\[([0-9]*)\])*$/', $name) === 1);
} }
/** /**

View File

@ -36,7 +36,7 @@ class Bytes extends SolidityType implements IType
*/ */
public function isType($name) public function isType($name)
{ {
return (preg_match('/bytes([0-9]{1,})?(\[([0-9]*)\])*/', $name) === 1); return (preg_match('/^bytes([0-9]{1,})(\[([0-9]*)\])*$/', $name) === 1);
} }
/** /**
@ -63,9 +63,10 @@ class Bytes extends SolidityType implements IType
} }
$value = Utils::stripZero($value); $value = Utils::stripZero($value);
// if (mb_strlen($value) % 2 !== 0) { if (mb_strlen($value) % 2 !== 0) {
$value = "0" . $value;
// throw new InvalidArgumentException('The value to inputFormat has invalid length. Value: ' . $value); // throw new InvalidArgumentException('The value to inputFormat has invalid length. Value: ' . $value);
// } }
if (mb_strlen($value) > 64) { if (mb_strlen($value) > 64) {
throw new InvalidArgumentException('The value to inputFormat is too long.'); throw new InvalidArgumentException('The value to inputFormat is too long.');
@ -90,6 +91,11 @@ class Bytes extends SolidityType implements IType
if (empty($checkZero)) { if (empty($checkZero)) {
return '0'; return '0';
} }
if (preg_match('/^bytes([0-9]*)/', $name, $match) === 1) {
$size = intval($match[1]);
$length = 2 * $size;
$value = mb_substr($value, 0, $length);
}
return '0x' . $value; return '0x' . $value;
} }
} }

View File

@ -36,7 +36,7 @@ class DynamicBytes extends SolidityType implements IType
*/ */
public function isType($name) public function isType($name)
{ {
return (preg_match('/bytes(\[([0-9]*)\])*/', $name) === 1); return (preg_match('/^bytes(\[([0-9]*)\])*$/', $name) === 1);
} }
/** /**
@ -63,12 +63,17 @@ class DynamicBytes extends SolidityType implements IType
} }
$value = Utils::stripZero($value); $value = Utils::stripZero($value);
// if (mb_strlen($value) % 2 !== 0) { if (mb_strlen($value) % 2 !== 0) {
$value = "0" . $value;
// throw new InvalidArgumentException('The value to inputFormat has invalid length.'); // throw new InvalidArgumentException('The value to inputFormat has invalid length.');
// } }
$bn = Utils::toBn(mb_strlen($value) / 2); $bn = Utils::toBn(floor(mb_strlen($value) / 2));
$bnHex = $bn->toHex(true); $bnHex = $bn->toHex(true);
$padded = mb_substr($bnHex, 0, 1); $padded = mb_substr($bnHex, 0, 1);
if ($padded !== '0' && $padded !== 'f') {
$padded = '0';
}
$l = floor((mb_strlen($value) + 63) / 64); $l = floor((mb_strlen($value) + 63) / 64);
$padding = (($l * 64 - mb_strlen($value) + 1) >= 0) ? $l * 64 - mb_strlen($value) : 0; $padding = (($l * 64 - mb_strlen($value) + 1) >= 0) ? $l * 64 - mb_strlen($value) : 0;
@ -89,6 +94,9 @@ class DynamicBytes extends SolidityType implements IType
if (empty($checkZero)) { if (empty($checkZero)) {
return '0'; return '0';
} }
return '0x' . $value; $size = intval(Utils::toBn('0x' . mb_substr($value, 0, 64))->toString());
$length = 2 * $size;
return '0x' . mb_substr($value, 64, $length);
} }
} }

View File

@ -37,7 +37,7 @@ class Integer extends SolidityType implements IType
*/ */
public function isType($name) public function isType($name)
{ {
return (preg_match('/int([0-9]{1,})?(\[([0-9]*)\])*/', $name) === 1); return (preg_match('/^int([0-9]{1,})?(\[([0-9]*)\])*$/', $name) === 1);
} }
/** /**

View File

@ -37,7 +37,7 @@ class Str extends SolidityType implements IType
*/ */
public function isType($name) public function isType($name)
{ {
return (preg_match('/string(\[([0-9]*)\])*/', $name) === 1); return (preg_match('/^string(\[([0-9]*)\])*$/', $name) === 1);
} }
/** /**
@ -77,7 +77,7 @@ class Str extends SolidityType implements IType
public function outputFormat($value, $name) public function outputFormat($value, $name)
{ {
$strLen = mb_substr($value, 0, 64); $strLen = mb_substr($value, 0, 64);
$strValue = mb_substr($value, 64, 64); $strValue = mb_substr($value, 64);
$match = []; $match = [];
if (preg_match('/^[0]+([a-f0-9]+)$/', $strLen, $match) === 1) { if (preg_match('/^[0]+([a-f0-9]+)$/', $strLen, $match) === 1) {

View File

@ -37,7 +37,7 @@ class Uinteger extends SolidityType implements IType
*/ */
public function isType($name) public function isType($name)
{ {
return (preg_match('/uint([0-9]{1,})?(\[([0-9]*)\])*/', $name) === 1); return (preg_match('/^uint([0-9]{1,})?(\[([0-9]*)\])*$/', $name) === 1);
} }
/** /**

View File

@ -38,7 +38,7 @@ class Eth
* @var array * @var array
*/ */
private $allowedMethods = [ private $allowedMethods = [
'eth_protocolVersion', 'eth_syncing', 'eth_coinbase', 'eth_mining', 'eth_hashrate', 'eth_gasPrice', 'eth_accounts', 'eth_blockNumber', 'eth_getBalance', 'eth_getStorageAt', 'eth_getTransactionCount', 'eth_getBlockTransactionCountByHash', 'eth_getBlockTransactionCountByNumber', 'eth_getUncleCountByBlockHash', 'eth_getUncleCountByBlockNumber', 'eth_getUncleByBlockHashAndIndex', 'eth_getUncleByBlockNumberAndIndex', 'eth_getCode', 'eth_sign', 'eth_sendTransaction', 'eth_sendRawTransaction', 'eth_call', 'eth_estimateGas', 'eth_getBlockByHash', 'eth_getBlockByNumber', 'eth_getTransactionByHash', 'eth_getTransactionByBlockHashAndIndex', 'eth_getTransactionByBlockNumberAndIndex', 'eth_getTransactionReceipt', 'eth_getCompilers', 'eth_compileSolidity', 'eth_compileLLL', 'eth_compileSerpent', 'eth_getWork', 'eth_newFilter', 'eth_newBlockFilter', 'eth_newPendingTransactionFilter', 'eth_uninstallFilter', 'eth_getFilterChanges', 'eth_getFilterLogs', 'eth_getLogs', 'eth_submitWork', 'eth_submitHashrate' 'eth_protocolVersion', 'eth_syncing', 'eth_coinbase', 'eth_mining', 'eth_hashrate', 'eth_gasPrice', 'eth_accounts', 'eth_blockNumber', 'eth_getBalance', 'eth_getStorageAt', 'eth_getTransactionCount', 'eth_getBlockTransactionCountByHash', 'eth_getBlockTransactionCountByNumber', 'eth_getUncleCountByBlockHash', 'eth_getUncleCountByBlockNumber', 'eth_getUncleByBlockHashAndIndex', 'eth_getUncleByBlockNumberAndIndex', 'eth_getCode', 'eth_sign', 'eth_sendTransaction', 'eth_sendRawTransaction', 'eth_call', 'eth_estimateGas', 'eth_getBlockByHash', 'eth_getBlockByNumber', 'eth_getTransactionByHash', 'eth_getTransactionByBlockHashAndIndex', 'eth_getTransactionByBlockNumberAndIndex', 'eth_getTransactionReceipt', 'eth_compileSolidity', 'eth_compileLLL', 'eth_compileSerpent', 'eth_getWork', 'eth_newFilter', 'eth_newBlockFilter', 'eth_newPendingTransactionFilter', 'eth_uninstallFilter', 'eth_getFilterChanges', 'eth_getFilterLogs', 'eth_getLogs', 'eth_submitWork', 'eth_submitHashrate'
]; ];
/** /**

View File

@ -23,18 +23,14 @@ class NewBlockFilter extends EthMethod
* *
* @var array * @var array
*/ */
protected $validators = [ protected $validators = [];
QuantityValidator::class
];
/** /**
* inputFormatters * inputFormatters
* *
* @var array * @var array
*/ */
protected $inputFormatters = [ protected $inputFormatters = [];
QuantityFormatter::class
];
/** /**
* outputFormatters * outputFormatters

View File

@ -9,26 +9,31 @@
* @license MIT * @license MIT
*/ */
namespace Web3\Methods\Eth; namespace Web3\Methods\Personal;
use InvalidArgumentException;
use Web3\Methods\EthMethod; use Web3\Methods\EthMethod;
use Web3\Validators\AddressValidator;
use Web3\Formatters\AddressFormatter;
class GetCompilers extends EthMethod class LockAccount extends EthMethod
{ {
/** /**
* validators * validators
* *
* @var array * @var array
*/ */
protected $validators = []; protected $validators = [
AddressValidator::class
];
/** /**
* inputFormatters * inputFormatters
* *
* @var array * @var array
*/ */
protected $inputFormatters = []; protected $inputFormatters = [
AddressFormatter::class
];
/** /**
* outputFormatters * outputFormatters
@ -56,3 +61,4 @@ class GetCompilers extends EthMethod
// parent::__construct($method, $arguments); // parent::__construct($method, $arguments);
// } // }
} }

View File

@ -38,7 +38,7 @@ class Personal
* @var array * @var array
*/ */
private $allowedMethods = [ private $allowedMethods = [
'personal_listAccounts', 'personal_newAccount', 'personal_unlockAccount', 'personal_sendTransaction' 'personal_listAccounts', 'personal_newAccount', 'personal_unlockAccount', 'personal_lockAccount', 'personal_sendTransaction'
]; ];
/** /**

View File

@ -12,6 +12,7 @@
namespace Web3\RequestManagers; namespace Web3\RequestManagers;
use InvalidArgumentException; use InvalidArgumentException;
use Psr\Http\Message\StreamInterface;
use RuntimeException as RPCException; use RuntimeException as RPCException;
use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ResponseInterface;
use GuzzleHttp\Exception\RequestException; use GuzzleHttp\Exception\RequestException;
@ -53,22 +54,7 @@ class HttpRequestManager extends RequestManager implements IRequestManager
if (!is_string($payload)) { if (!is_string($payload)) {
throw new \InvalidArgumentException('Payload must be string.'); throw new \InvalidArgumentException('Payload must be string.');
} }
// $promise = $this->client->postAsync($this->host, [
// 'headers' => [
// 'content-type' => 'application/json'
// ],
// 'body' => $payload
// ]);
// $promise->then(
// function (ResponseInterface $res) use ($callback) {
// var_dump($res->body());
// call_user_func($callback, null, $res);
// },
// function (RequestException $err) use ($callback) {
// var_dump($err->getMessage());
// call_user_func($callback, $err, null);
// }
// );
try { try {
$res = $this->client->post($this->host, [ $res = $this->client->post($this->host, [
'headers' => [ 'headers' => [
@ -78,7 +64,12 @@ class HttpRequestManager extends RequestManager implements IRequestManager
'timeout' => $this->timeout, 'timeout' => $this->timeout,
'connect_timeout' => $this->timeout 'connect_timeout' => $this->timeout
]); ]);
$json = json_decode($res->getBody()); /**
* @var StreamInterface $stream ;
*/
$stream = $res->getBody();
$json = json_decode($stream);
$stream->close();
if (JSON_ERROR_NONE !== json_last_error()) { if (JSON_ERROR_NONE !== json_last_error()) {
call_user_func($callback, new InvalidArgumentException('json_decode error: ' . json_last_error_msg()), null); call_user_func($callback, new InvalidArgumentException('json_decode error: ' . json_last_error_msg()), null);
@ -89,7 +80,7 @@ class HttpRequestManager extends RequestManager implements IRequestManager
$errors = []; $errors = [];
foreach ($json as $result) { foreach ($json as $result) {
if (isset($result->result)) { if (property_exists($result,'result')) {
$results[] = $result->result; $results[] = $result->result;
} else { } else {
if (isset($json->error)) { if (isset($json->error)) {
@ -105,7 +96,7 @@ class HttpRequestManager extends RequestManager implements IRequestManager
} else { } else {
call_user_func($callback, null, $results); call_user_func($callback, null, $results);
} }
} elseif (isset($json->result)) { } elseif (property_exists($json,'result')) {
call_user_func($callback, null, $json->result); call_user_func($callback, null, $json->result);
} else { } else {
if (isset($json->error)) { if (isset($json->error)) {

View File

@ -212,6 +212,31 @@ class Utils
return true; return true;
} }
/**
* toChecksumAddress
*
* @param string $value
* @return string
*/
public static function toChecksumAddress($value)
{
if (!is_string($value)) {
throw new InvalidArgumentException('The value to toChecksumAddress function must be string.');
}
$value = self::stripZero(strtolower($value));
$hash = self::stripZero(self::sha3($value));
$ret = '0x';
for ($i = 0; $i < 40; $i++) {
if (intval($hash[$i], 16) >= 8) {
$ret .= strtoupper($value[$i]);
} else {
$ret .= $value[$i];
}
}
return $ret;
}
/** /**
* isHex * isHex
* *
@ -266,12 +291,15 @@ class Utils
* $wei = Utils::toWei('1', 'kwei'); * $wei = Utils::toWei('1', 'kwei');
* $wei->toString(); // 1000 * $wei->toString(); // 1000
* *
* @param BigNumber|string|int $number * @param BigNumber|string $number
* @param string $unit * @param string $unit
* @return \phpseclib\Math\BigInteger * @return \phpseclib\Math\BigInteger
*/ */
public static function toWei($number, $unit) public static function toWei($number, $unit)
{ {
if (!is_string($number) && !($number instanceof BigNumber)) {
throw new InvalidArgumentException('toWei number must be string or bignumber.');
}
$bn = self::toBn($number); $bn = self::toBn($number);
if (!is_string($unit)) { if (!is_string($unit)) {
@ -284,15 +312,33 @@ class Utils
if (is_array($bn)) { if (is_array($bn)) {
// fraction number // fraction number
list($whole, $fraction, $negative1) = $bn; list($whole, $fraction, $fractionLength, $negative1) = $bn;
$fractionLength = strlen($fraction->toString());
if ($fractionLength > strlen(self::UNITS[$unit])) { if ($fractionLength > strlen(self::UNITS[$unit])) {
throw new InvalidArgumentException('toWei fraction part is out of limit.'); throw new InvalidArgumentException('toWei fraction part is out of limit.');
} }
$whole = $whole->multiply($bnt); $whole = $whole->multiply($bnt);
$base = (new BigNumber(10))->pow(new BigNumber($fractionLength));
// There is no pow function in phpseclib 2.0, only can see in dev-master
// Maybe implement own biginteger in the future
// See 2.0 BigInteger: https://github.com/phpseclib/phpseclib/blob/2.0/phpseclib/Math/BigInteger.php
// See dev-master BigInteger: https://github.com/phpseclib/phpseclib/blob/master/phpseclib/Math/BigInteger.php#L700
// $base = (new BigNumber(10))->pow(new BigNumber($fractionLength));
// So we switch phpseclib special global param, change in the future
switch (MATH_BIGINTEGER_MODE) {
case $whole::MODE_GMP:
static $two;
$powerBase = gmp_pow(gmp_init(10), (int) $fractionLength);
break;
case $whole::MODE_BCMATH:
$powerBase = bcpow('10', (string) $fractionLength, 0);
break;
default:
$powerBase = pow(10, (int) $fractionLength);
break;
}
$base = new BigNumber($powerBase);
$fraction = $fraction->multiply($bnt)->divide($base)[0]; $fraction = $fraction->multiply($bnt)->divide($base)[0];
if ($negative1 !== false) { if ($negative1 !== false) {
@ -401,53 +447,35 @@ class Utils
/** /**
* jsonToArray * jsonToArray
* *
* @param stdClass|array|string $json * @param stdClass|array $json
* @param int $depth
* @return array * @return array
*/ */
public static function jsonToArray($json, $depth=1) public static function jsonToArray($json)
{ {
if (!is_int($depth) || $depth <= 0) {
throw new InvalidArgumentException('jsonToArray depth must be int and depth must bigger than 0.');
}
if ($json instanceof stdClass) { if ($json instanceof stdClass) {
$json = (array) $json; $json = (array) $json;
$typeName = []; $typeName = [];
if ($depth > 1) {
foreach ($json as $key => $param) { foreach ($json as $key => $param) {
if (is_array($param)) { if (is_array($param)) {
foreach ($param as $subKey => $subParam) { foreach ($param as $subKey => $subParam) {
$json[$key][$subKey] = self::jsonToArray($subParam, $depth-1); $json[$key][$subKey] = self::jsonToArray($subParam);
} }
} elseif ($param instanceof stdClass) { } elseif ($param instanceof stdClass) {
$json[$key] = self::jsonToArray($param, $depth-1); $json[$key] = self::jsonToArray($param);
} }
} }
}
return $json;
} elseif (is_array($json)) { } elseif (is_array($json)) {
if ($depth > 1) {
foreach ($json as $key => $param) { foreach ($json as $key => $param) {
if (is_array($param)) { if (is_array($param)) {
foreach ($param as $subKey => $subParam) { foreach ($param as $subKey => $subParam) {
$json[$key][$subKey] = self::jsonToArray($subParam, $depth-1); $json[$key][$subKey] = self::jsonToArray($subParam);
} }
} elseif ($param instanceof stdClass) { } elseif ($param instanceof stdClass) {
$json[$key] = self::jsonToArray($param, $depth-1); $json[$key] = self::jsonToArray($param);
} }
} }
} }
} elseif (is_string($json)) {
$json = json_decode($json, true);
if (JSON_ERROR_NONE !== json_last_error()) {
throw new InvalidArgumentException('json_decode error: ' . json_last_error_msg());
}
return $json;
} else {
throw new InvalidArgumentException('The json param to jsonToArray must be array or stdClass or string.');
}
return $json; return $json;
} }
@ -484,6 +512,7 @@ class Utils
return [ return [
new BigNumber($whole), new BigNumber($whole),
new BigNumber($fraction), new BigNumber($fraction),
strlen($comps[1]),
isset($negative1) ? $negative1 : false isset($negative1) ? $negative1 : false
]; ];
} else { } else {
@ -500,7 +529,7 @@ class Utils
$number = str_replace('-', '', $number, $count); $number = str_replace('-', '', $number, $count);
$negative1 = new BigNumber(-1); $negative1 = new BigNumber(-1);
} }
if (self::isZeroPrefixed($number) || preg_match('/[a-f]+/', $number) === 1) { if (self::isZeroPrefixed($number) || preg_match('/^[0-9a-f]+$/i', $number) === 1) {
$number = self::stripZero($number); $number = self::stripZero($number);
$bn = new BigNumber($number, 16); $bn = new BigNumber($number, 16);
} elseif (empty($number)) { } elseif (empty($number)) {

View File

@ -37,10 +37,8 @@ class TestCase extends BaseTestCase
/** /**
* setUp * setUp
*
* @return void
*/ */
public function setUp() public function setUp(): void
{ {
$web3 = new Web3($this->testHost); $web3 = new Web3($this->testHost);
$this->web3 = $web3; $this->web3 = $web3;
@ -55,8 +53,6 @@ class TestCase extends BaseTestCase
/** /**
* tearDown * tearDown
*
* @return void
*/ */
public function tearDown() {} public function tearDown(): void {}
} }

View File

@ -0,0 +1,15 @@
pragma solidity ^0.5.1;
contract FIX125 {
bytes public data;
event SetData(
bytes indexed _data
);
function setData(bytes memory _data) public {
data = _data;
emit SetData(_data);
}
}

View File

@ -0,0 +1,26 @@
pragma solidity ^0.5.1;
contract FIX134 {
bytes public data;
uint256 public number;
event Say(
uint256 indexed _number
);
event Say(
uint256 indexed _number,
bytes indexed _data
);
function say(uint256 _number) public {
number = _number;
emit Say(_number);
}
function say(uint256 _number, bytes memory _data) public {
data = _data;
number = _number;
emit Say(_number, _data);
}
}

View File

@ -19,7 +19,7 @@ class AddressFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new AddressFormatter; $this->formatter = new AddressFormatter;

View File

@ -47,7 +47,7 @@ class AddressTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->solidityType = new Address; $this->solidityType = new Address;

View File

@ -19,7 +19,7 @@ class AddressValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new AddressValidator; $this->validator = new AddressValidator;

View File

@ -20,7 +20,7 @@ class BigNumberFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new BigNumberFormatter; $this->formatter = new BigNumberFormatter;

View File

@ -19,7 +19,7 @@ class BlockHashValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new BlockHashValidator; $this->validator = new BlockHashValidator;

View File

@ -20,7 +20,7 @@ class BooleanFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new BooleanFormatter; $this->formatter = new BooleanFormatter;

View File

@ -47,7 +47,7 @@ class BooleanTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->solidityType = new Boolean; $this->solidityType = new Boolean;

View File

@ -19,7 +19,7 @@ class BooleanValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new BooleanValidator; $this->validator = new BooleanValidator;

View File

@ -16,22 +16,22 @@ class BytesTypeTest extends TestCase
protected $testTypes = [ protected $testTypes = [
[ [
'value' => 'bytes', 'value' => 'bytes',
'result' => true 'result' => false
], [ ], [
'value' => 'bytes[]', 'value' => 'bytes[]',
'result' => true 'result' => false
], [ ], [
'value' => 'bytes[4]', 'value' => 'bytes[4]',
'result' => true 'result' => false
], [ ], [
'value' => 'bytes[][]', 'value' => 'bytes[][]',
'result' => true 'result' => false
], [ ], [
'value' => 'bytes[3][]', 'value' => 'bytes[3][]',
'result' => true 'result' => false
], [ ], [
'value' => 'bytes[][6][]', 'value' => 'bytes[][6][]',
'result' => true 'result' => false
], [ ], [
'value' => 'bytes32', 'value' => 'bytes32',
'result' => true 'result' => true
@ -53,7 +53,7 @@ class BytesTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->solidityType = new Bytes; $this->solidityType = new Bytes;

View File

@ -19,7 +19,7 @@ class CallValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new CallValidator; $this->validator = new CallValidator;

File diff suppressed because one or more lines are too long

View File

@ -34,10 +34,10 @@ class DynamicBytesTypeTest extends TestCase
'result' => true 'result' => true
], [ ], [
'value' => 'bytes32', 'value' => 'bytes32',
'result' => true 'result' => false
], [ ], [
'value' => 'bytes8[4]', 'value' => 'bytes8[4]',
'result' => true 'result' => false
], ],
]; ];
@ -53,7 +53,7 @@ class DynamicBytesTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->solidityType = new DynamicBytes; $this->solidityType = new DynamicBytes;

View File

@ -21,7 +21,7 @@ class EthApiTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
@ -439,7 +439,7 @@ class EthApiTest extends TestCase
if ($err !== null) { if ($err !== null) {
return $this->assertTrue($err !== null); return $this->assertTrue($err !== null);
} }
// weired behavior, see https://github.com/sc0Vu/web3.php/issues/16 // weird behavior, see https://github.com/web3p/web3.php/issues/16
$this->assertTrue($block !== null); $this->assertTrue($block !== null);
}); });
} }
@ -457,7 +457,7 @@ class EthApiTest extends TestCase
if ($err !== null) { if ($err !== null) {
return $this->assertTrue($err !== null); return $this->assertTrue($err !== null);
} }
$this->assertTrue($transaction !== null); $this->assertTrue($transaction == null);
}); });
} }
@ -474,7 +474,7 @@ class EthApiTest extends TestCase
if ($err !== null) { if ($err !== null) {
return $this->assertTrue($err !== null); return $this->assertTrue($err !== null);
} }
$this->assertTrue($transaction !== null); $this->assertTrue($transaction == null);
}); });
} }
@ -508,7 +508,7 @@ class EthApiTest extends TestCase
if ($err !== null) { if ($err !== null) {
return $this->assertTrue($err !== null); return $this->assertTrue($err !== null);
} }
$this->assertTrue($transaction !== null); $this->assertTrue($transaction == null);
}); });
} }
@ -546,24 +546,6 @@ class EthApiTest extends TestCase
}); });
} }
/**
* testGetCompilers
*
* @return void
*/
public function testGetCompilers()
{
$eth = $this->eth;
$eth->getCompilers(function ($err, $compilers) {
if ($err !== null) {
return $this->assertTrue($err !== null);
}
$this->assertTrue(is_array($compilers));
$this->assertEquals($compilers[0], 'solidity');
});
}
/** /**
* testCompileSolidity * testCompileSolidity
* *
@ -647,7 +629,7 @@ class EthApiTest extends TestCase
{ {
$eth = $this->eth; $eth = $this->eth;
$eth->newBlockFilter('0x01', function ($err, $filter) { $eth->newBlockFilter(function ($err, $filter) {
if ($err !== null) { if ($err !== null) {
// infura banned us to new block filter // infura banned us to new block filter
return $this->assertTrue($err !== null); return $this->assertTrue($err !== null);

View File

@ -20,7 +20,7 @@ class EthBatchTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -23,7 +23,7 @@ class EthTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -9,6 +9,7 @@ use Web3\Contracts\Ethabi;
use Web3\Contracts\Types\Address; use Web3\Contracts\Types\Address;
use Web3\Contracts\Types\Boolean; use Web3\Contracts\Types\Boolean;
use Web3\Contracts\Types\Bytes; use Web3\Contracts\Types\Bytes;
use Web3\Contracts\Types\DynamicBytes;
use Web3\Contracts\Types\Integer; use Web3\Contracts\Types\Integer;
use Web3\Contracts\Types\Str; use Web3\Contracts\Types\Str;
use Web3\Contracts\Types\Uinteger; use Web3\Contracts\Types\Uinteger;
@ -162,7 +163,7 @@ class EthabiTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
// Error: Using $this when not in object context // Error: Using $this when not in object context
@ -179,9 +180,10 @@ class EthabiTest extends TestCase
'address' => new Address, 'address' => new Address,
'bool' => new Boolean, 'bool' => new Boolean,
'bytes' => new Bytes, 'bytes' => new Bytes,
'dynamicBytes' => new DynamicBytes,
'int' => new Integer, 'int' => new Integer,
'string' => new Str, 'string' => new Str,
'uint' => new Uinteger, 'uint' => new Uinteger
]); ]);
} }
@ -293,4 +295,23 @@ class EthabiTest extends TestCase
} }
} }
} }
/**
* testIssue71
* test 33 bytes and 128 bytes string, see: https://github.com/web3p/web3.php/issues/71
* string generated from: https://www.lipsum.com/
*
* @return void
*/
public function testIssue71()
{
$abi = $this->abi;
$specialString = 'Lorem ipsum dolor sit amet metus.';
$encodedString = $abi->encodeParameter('string', $specialString);
$this->assertEquals($specialString, $abi->decodeParameter('string', $encodedString));
$specialString = 'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Fusce pulvinar quam felis, suscipit posuere neque aliquam in cras amet.';
$encodedString = $abi->encodeParameter('string', $specialString);
$this->assertEquals($specialString, $abi->decodeParameter('string', $encodedString));
}
} }

View File

@ -19,7 +19,7 @@ class FilterValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new FilterValidator; $this->validator = new FilterValidator;

View File

@ -19,7 +19,7 @@ class HexFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new HexFormatter; $this->formatter = new HexFormatter;

View File

@ -19,7 +19,7 @@ class HexValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new HexValidator; $this->validator = new HexValidator;

View File

@ -17,7 +17,7 @@ class HttpProviderTest extends TestCase
*/ */
public function testSend() public function testSend()
{ {
$requestManager = new HttpRequestManager('http://localhost:8545'); $requestManager = new HttpRequestManager($this->testHost);
$provider = new HttpProvider($requestManager); $provider = new HttpProvider($requestManager);
$method = new ClientVersion('web3_clientVersion', []); $method = new ClientVersion('web3_clientVersion', []);
@ -36,7 +36,7 @@ class HttpProviderTest extends TestCase
*/ */
public function testBatch() public function testBatch()
{ {
$requestManager = new HttpRequestManager('http://localhost:8545'); $requestManager = new HttpRequestManager($this->testHost);
$provider = new HttpProvider($requestManager); $provider = new HttpProvider($requestManager);
$method = new ClientVersion('web3_clientVersion', []); $method = new ClientVersion('web3_clientVersion', []);
$callback = function ($err, $data) { $callback = function ($err, $data) {

View File

@ -19,7 +19,7 @@ class IdentityValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new IdentityValidator; $this->validator = new IdentityValidator;

View File

@ -19,7 +19,7 @@ class IntegerFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new IntegerFormatter; $this->formatter = new IntegerFormatter;
@ -45,5 +45,11 @@ class IntegerFormatterTest extends TestCase
$hex = $formatter->format('1', 20); $hex = $formatter->format('1', 20);
$this->assertEquals($hex, implode('', array_fill(0, 19, '0')) . '1'); $this->assertEquals($hex, implode('', array_fill(0, 19, '0')) . '1');
$hex = $formatter->format(48);
$this->assertEquals($hex, implode('', array_fill(0, 62, '0')) . '30');
$hex = $formatter->format('48');
$this->assertEquals($hex, implode('', array_fill(0, 62, '0')) . '30');
} }
} }

View File

@ -53,7 +53,7 @@ class IntegerTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->solidityType = new Integer; $this->solidityType = new Integer;

View File

@ -21,7 +21,7 @@ class NetApiTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -20,7 +20,7 @@ class NetBatchTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -23,7 +23,7 @@ class NetTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -19,7 +19,7 @@ class NonceValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new NonceValidator; $this->validator = new NonceValidator;

View File

@ -19,7 +19,7 @@ class NumberFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new NumberFormatter; $this->formatter = new NumberFormatter;

View File

@ -19,7 +19,7 @@ class OptionalQuantityFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new OptionalQuantityFormatter; $this->formatter = new OptionalQuantityFormatter;
@ -55,6 +55,8 @@ class OptionalQuantityFormatterTest extends TestCase
$this->assertEquals('latest', $formatter->format('latest')); $this->assertEquals('latest', $formatter->format('latest'));
$this->assertEquals('earliest', $formatter->format('earliest')); $this->assertEquals('earliest', $formatter->format('earliest'));
$this->assertEquals('pending', $formatter->format('pending')); $this->assertEquals('pending', $formatter->format('pending'));
$this->assertEquals('0x0', $formatter->format('hello'));
$this->expectExceptionMessage('toBn number must be valid hex string.');
$formatter->format('hello');
} }
} }

View File

@ -27,7 +27,7 @@ class PersonalApiTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
@ -63,7 +63,7 @@ class PersonalApiTest extends TestCase
$personal->newAccount('123456', function ($err, $account) { $personal->newAccount('123456', function ($err, $account) {
if ($err !== null) { if ($err !== null) {
return $this->fail($e->getMessage()); return $this->fail($err->getMessage());
} }
$this->assertTrue(is_string($account)); $this->assertTrue(is_string($account));
}); });
@ -81,7 +81,7 @@ class PersonalApiTest extends TestCase
// create account // create account
$personal->newAccount('123456', function ($err, $account) { $personal->newAccount('123456', function ($err, $account) {
if ($err !== null) { if ($err !== null) {
return $this->fail($e->getMessage()); return $this->fail($err->getMessage());
} }
$this->newAccount = $account; $this->newAccount = $account;
$this->assertTrue(is_string($account)); $this->assertTrue(is_string($account));
@ -107,7 +107,7 @@ class PersonalApiTest extends TestCase
// create account // create account
$personal->newAccount('123456', function ($err, $account) { $personal->newAccount('123456', function ($err, $account) {
if ($err !== null) { if ($err !== null) {
return $this->fail($e->getMessage()); return $this->fail($err->getMessage());
} }
$this->newAccount = $account; $this->newAccount = $account;
$this->assertTrue(is_string($account)); $this->assertTrue(is_string($account));
@ -121,6 +121,39 @@ class PersonalApiTest extends TestCase
}); });
} }
/**
* testLockAccount
*
* @return void
*/
public function testLockAccount()
{
$personal = $this->personal;
// create account
$personal->newAccount('123456', function ($err, $account) {
if ($err !== null) {
return $this->fail($err->getMessage());
}
$this->newAccount = $account;
$this->assertTrue(is_string($account));
});
$personal->unlockAccount($this->newAccount, '123456', function ($err, $unlocked) {
if ($err !== null) {
return $this->fail($err->getMessage());
}
$this->assertTrue($unlocked);
});
$personal->lockAccount($this->newAccount, function ($err, $locked) {
if ($err !== null) {
return $this->fail($err->getMessage());
}
$this->assertTrue($locked);
});
}
/** /**
* testSendTransaction * testSendTransaction
* *
@ -133,7 +166,7 @@ class PersonalApiTest extends TestCase
// create account // create account
$personal->newAccount('123456', function ($err, $account) { $personal->newAccount('123456', function ($err, $account) {
if ($err !== null) { if ($err !== null) {
return $this->fail($e->getMessage()); return $this->fail($err->getMessage());
} }
$this->newAccount = $account; $this->newAccount = $account;
$this->assertTrue(is_string($account)); $this->assertTrue(is_string($account));

View File

@ -19,7 +19,7 @@ class PersonalBatchTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -23,7 +23,7 @@ class PersonalTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -19,7 +19,7 @@ class PostFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new PostFormatter; $this->formatter = new PostFormatter;

View File

@ -19,7 +19,7 @@ class PostValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new PostValidator; $this->validator = new PostValidator;

View File

@ -19,7 +19,7 @@ class QuantityFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new QuantityFormatter; $this->formatter = new QuantityFormatter;

View File

@ -19,7 +19,7 @@ class QuantityValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new QuantityValidator; $this->validator = new QuantityValidator;

View File

@ -21,7 +21,7 @@ class ShhApiTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -19,7 +19,7 @@ class ShhBatchTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -19,7 +19,7 @@ class ShhFilterValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new ShhFilterValidator; $this->validator = new ShhFilterValidator;

View File

@ -23,7 +23,7 @@ class ShhTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();

View File

@ -20,7 +20,7 @@ class SolidityTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->type = new SolidityType(); $this->type = new SolidityType();

View File

@ -47,7 +47,7 @@ class StrTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->solidityType = new Str; $this->solidityType = new Str;

View File

@ -19,7 +19,7 @@ class StringFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new StringFormatter; $this->formatter = new StringFormatter;

View File

@ -19,7 +19,7 @@ class StringValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new StringValidator; $this->validator = new StringValidator;

View File

@ -19,7 +19,7 @@ class TagValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new TagValidator; $this->validator = new TagValidator;

View File

@ -19,7 +19,7 @@ class TransactionFormatterTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->formatter = new TransactionFormatter; $this->formatter = new TransactionFormatter;

View File

@ -19,7 +19,7 @@ class TransactionValidatorTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->validator = new TransactionValidator; $this->validator = new TransactionValidator;

View File

@ -53,7 +53,7 @@ class UintegerTypeTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
$this->solidityType = new Uinteger; $this->solidityType = new Uinteger;

View File

@ -7,6 +7,7 @@ use stdClass;
use Test\TestCase; use Test\TestCase;
use phpseclib\Math\BigInteger as BigNumber; use phpseclib\Math\BigInteger as BigNumber;
use Web3\Utils; use Web3\Utils;
use Web3\Contract;
class UtilsTest extends TestCase class UtilsTest extends TestCase
{ {
@ -52,12 +53,91 @@ class UtilsTest extends TestCase
} }
}'; }';
/**
* testIssue112Json
* see: https://github.com/web3p/web3.php/issues/112
*
* @var string
*/
protected $testIssue112Json = '[
{
"constant": true,
"inputs": [],
"name": "name",
"outputs": [
{
"name": "",
"type": "string"
}
],
"payable": false,
"stateMutability": "view",
"type": "function"
},
{
"constant": true,
"inputs": [],
"name": "decimals",
"outputs": [
{
"name": "",
"type": "uint256"
}
],
"payable": false,
"stateMutability": "view",
"type": "function"
},
{
"constant": true,
"inputs": [
{
"name": "tokenOwner",
"type": "address"
}
],
"name": "balanceOf",
"outputs": [
{
"name": "balance",
"type": "uint256"
}
],
"payable": false,
"stateMutability": "view",
"type": "function"
},
{
"constant": false,
"inputs": [
{
"name": "to",
"type": "address"
},
{
"name": "tokens",
"type": "uint256"
}
],
"name": "transfer",
"outputs": [
{
"name": "success",
"type": "bool"
}
],
"payable": false,
"stateMutability": "nonpayable",
"type": "function"
}
]';
/** /**
* setUp * setUp
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
} }
@ -87,6 +167,16 @@ class UtilsTest extends TestCase
$this->assertEquals('0x', Utils::toHex(0, true)); $this->assertEquals('0x', Utils::toHex(0, true));
$this->assertEquals('0x', Utils::toHex(new BigNumber(0), true)); $this->assertEquals('0x', Utils::toHex(new BigNumber(0), true));
$this->assertEquals('0x30', Utils::toHex(48, true));
$this->assertEquals('0x30', Utils::toHex('48', true));
$this->assertEquals('30', Utils::toHex(48));
$this->assertEquals('30', Utils::toHex('48'));
$this->assertEquals('0x30', Utils::toHex(new BigNumber(48), true));
$this->assertEquals('0x30', Utils::toHex(new BigNumber('48'), true));
$this->assertEquals('30', Utils::toHex(new BigNumber(48)));
$this->assertEquals('30', Utils::toHex(new BigNumber('48')));
$this->expectException(InvalidArgumentException::class); $this->expectException(InvalidArgumentException::class);
$hex = Utils::toHex(new stdClass); $hex = Utils::toHex(new stdClass);
} }
@ -201,6 +291,33 @@ class UtilsTest extends TestCase
$isAddressChecksum = Utils::isAddressChecksum(new stdClass); $isAddressChecksum = Utils::isAddressChecksum(new stdClass);
} }
/**
* testToChecksumAddress
*
* @return void
*/
public function testToChecksumAddress()
{
$checksumAddressTest = [
// All caps
'0x52908400098527886E0F7030069857D2E4169EE7',
'0x8617E340B3D01FA5F11F306F4090FD50E238070D',
// All Lower
'0xde709f2102306220921060314715629080e2fb77',
'0x27b1fdb04752bbc536007a920d24acb045561c26',
// Normal
'0x5aAeb6053F3E94C9b9A09f33669435E7Ef1BeAed',
'0xfB6916095ca1df60bB79Ce92cE3Ea74c37c5d359',
'0xdbF03B407c01E7cD3CBea99509d93f8DDDC8C6FB',
'0xD1220A0cf47c7B9Be7A2E6BA89F429762e7b9aDb'
];
for ($i=0; $i<count($checksumAddressTest); $i++) {
$checksumAddress = Utils::toChecksumAddress(strtolower($checksumAddressTest[$i]));
$this->assertEquals($checksumAddressTest[$i], $checksumAddress);
}
}
/** /**
* testStripZero * testStripZero
* *
@ -242,70 +359,58 @@ class UtilsTest extends TestCase
public function testToWei() public function testToWei()
{ {
$bn = Utils::toWei('0x1', 'wei'); $bn = Utils::toWei('0x1', 'wei');
$this->assertEquals($bn->toString(), '1'); $this->assertEquals('1', $bn->toString());
$bn = Utils::toWei('18', 'wei'); $bn = Utils::toWei('18', 'wei');
$this->assertEquals($bn->toString(), '18'); $this->assertEquals('18', $bn->toString());
$bn = Utils::toWei(1, 'wei');
$this->assertEquals($bn->toString(), '1');
$bn = Utils::toWei(0x11, 'wei');
$this->assertEquals($bn->toString(), '17');
$bn = Utils::toWei('1', 'ether'); $bn = Utils::toWei('1', 'ether');
$this->assertEquals($bn->toString(), '1000000000000000000'); $this->assertEquals('1000000000000000000', $bn->toString());
$bn = Utils::toWei('0x5218', 'wei'); $bn = Utils::toWei('0x5218', 'wei');
$this->assertEquals($bn->toString(), '21016'); $this->assertEquals('21016', $bn->toString());
$bn = Utils::toWei('0.000012', 'ether');
$this->assertEquals('12000000000000', $bn->toString());
$bn = Utils::toWei('0.1', 'ether'); $bn = Utils::toWei('0.1', 'ether');
$this->assertEquals($bn->toString(), '100000000000000000'); $this->assertEquals('100000000000000000', $bn->toString());
$bn = Utils::toWei('1.69', 'ether'); $bn = Utils::toWei('1.69', 'ether');
$this->assertEquals($bn->toString(), '1690000000000000000'); $this->assertEquals('1690000000000000000', $bn->toString());
$bn = Utils::toWei(0.1, 'ether'); $bn = Utils::toWei('0.01', 'ether');
$this->assertEquals($bn->toString(), '100000000000000000'); $this->assertEquals('10000000000000000', $bn->toString());
$bn = Utils::toWei(1.69, 'ether'); $bn = Utils::toWei('0.002', 'ether');
$this->assertEquals($bn->toString(), '1690000000000000000'); $this->assertEquals('2000000000000000', $bn->toString());
$bn = Utils::toWei('-0.1', 'ether'); $bn = Utils::toWei('-0.1', 'ether');
$this->assertEquals($bn->toString(), '-100000000000000000'); $this->assertEquals('-100000000000000000', $bn->toString());
$bn = Utils::toWei('-1.69', 'ether'); $bn = Utils::toWei('-1.69', 'ether');
$this->assertEquals($bn->toString(), '-1690000000000000000'); $this->assertEquals('-1690000000000000000', $bn->toString());
$bn = Utils::toWei(-0.1, 'ether');
$this->assertEquals($bn->toString(), '-100000000000000000');
$bn = Utils::toWei(-1.69, 'ether');
$this->assertEquals($bn->toString(), '-1690000000000000000');
$bn = Utils::toWei('', 'ether'); $bn = Utils::toWei('', 'ether');
$this->assertEquals($bn->toString(), '0'); $this->assertEquals('0', $bn->toString());
$bn = Utils::toWei(-1.697, 'kwei');
$this->assertEquals($bn->toString(), '-1697');
try { try {
$bn = Utils::toWei('0x5218', new stdClass); $bn = Utils::toWei('0x5218', new stdClass);
} catch (InvalidArgumentException $e) { } catch (InvalidArgumentException $e) {
$this->assertTrue($e !== null); $this->assertEquals('toWei unit must be string.', $e->getMessage());
} }
try { try {
$bn = Utils::toWei('0x5218', 'test'); $bn = Utils::toWei('0x5218', 'test');
} catch (InvalidArgumentException $e) { } catch (InvalidArgumentException $e) {
$this->assertTrue($e !== null); $this->assertEquals('toWei doesn\'t support test unit.', $e->getMessage());
} }
try { try {
// out of limit // out of limit
$bn = Utils::toWei(-1.6977, 'kwei'); $bn = Utils::toWei(-1.6977, 'kwei');
} catch (InvalidArgumentException $e) { } catch (InvalidArgumentException $e) {
$this->assertTrue($e !== null); $this->assertEquals('toWei number must be string or bignumber.', $e->getMessage());
} }
} }
@ -326,16 +431,6 @@ class UtilsTest extends TestCase
$this->assertEquals($bnq->toString(), '0'); $this->assertEquals($bnq->toString(), '0');
$this->assertEquals($bnr->toString(), '18'); $this->assertEquals($bnr->toString(), '18');
list($bnq, $bnr) = Utils::toEther(1, 'wei');
$this->assertEquals($bnq->toString(), '0');
$this->assertEquals($bnr->toString(), '1');
list($bnq, $bnr) = Utils::toEther(0x11, 'wei');
$this->assertEquals($bnq->toString(), '0');
$this->assertEquals($bnr->toString(), '17');
list($bnq, $bnr) = Utils::toEther('1', 'kether'); list($bnq, $bnr) = Utils::toEther('1', 'kether');
$this->assertEquals($bnq->toString(), '1000'); $this->assertEquals($bnq->toString(), '1000');
@ -430,39 +525,16 @@ class UtilsTest extends TestCase
*/ */
public function testJsonToArray() public function testJsonToArray()
{ {
$json = json_decode($this->testJsonMethodString); $decodedJson = json_decode($this->testJsonMethodString);
$jsonArrayDepth1 = Utils::jsonToArray($json); $jsonArray = Utils::jsonToArray($decodedJson);
$this->assertEquals($jsonArrayDepth1, (array) $json);
$jsonAssoc = json_decode($this->testJsonMethodString, true); $jsonAssoc = json_decode($this->testJsonMethodString, true);
$jsonArrayDepth2 = Utils::jsonToArray($json, 2); $jsonArray2 = Utils::jsonToArray($jsonAssoc);
$this->assertEquals($jsonAssoc, $jsonArray);
$this->assertEquals($jsonAssoc, $jsonArray2);
$this->assertEquals($jsonArrayDepth2, $jsonAssoc); $jsonAssoc = json_decode($this->testIssue112Json, true);
$jsonArray = Utils::jsonToArray($jsonAssoc);
$jsonArrayDepth2 = Utils::jsonToArray($jsonArrayDepth1, 2); $this->assertEquals($jsonAssoc, $jsonArray);
$this->assertEquals($jsonArrayDepth2, $jsonAssoc);
$jsonArray = Utils::jsonToArray($this->testJsonMethodString);
$this->assertEquals($jsonArray, $jsonAssoc);
try {
$jsonArray = Utils::jsonToArray($json, 0);
} catch (InvalidArgumentException $e) {
$this->assertTrue($e !== null);
}
try {
$jsonArray = Utils::jsonToArray(mb_substr($this->testJsonMethodString, 0, 50), 1);
} catch (InvalidArgumentException $e) {
$this->assertTrue($e !== null);
}
try {
$jsonArray = Utils::jsonToArray(0, 1);
} catch (InvalidArgumentException $e) {
$this->assertTrue($e !== null);
}
} }
/** /**
@ -531,39 +603,45 @@ class UtilsTest extends TestCase
$this->assertEquals($bn->toString(), '-1'); $this->assertEquals($bn->toString(), '-1');
$bn = Utils::toBn('-0.1'); $bn = Utils::toBn('-0.1');
$this->assertEquals(count($bn), 3); $this->assertEquals(count($bn), 4);
$this->assertEquals($bn[0]->toString(), '0'); $this->assertEquals($bn[0]->toString(), '0');
$this->assertEquals($bn[1]->toString(), '1'); $this->assertEquals($bn[1]->toString(), '1');
$this->assertEquals($bn[2]->toString(), '-1'); $this->assertEquals($bn[2], 1);
$this->assertEquals($bn[3]->toString(), '-1');
$bn = Utils::toBn(-0.1); $bn = Utils::toBn(-0.1);
$this->assertEquals(count($bn), 3); $this->assertEquals(count($bn), 4);
$this->assertEquals($bn[0]->toString(), '0'); $this->assertEquals($bn[0]->toString(), '0');
$this->assertEquals($bn[1]->toString(), '1'); $this->assertEquals($bn[1]->toString(), '1');
$this->assertEquals($bn[2]->toString(), '-1'); $this->assertEquals($bn[2], 1);
$this->assertEquals($bn[3]->toString(), '-1');
$bn = Utils::toBn('0.1'); $bn = Utils::toBn('0.1');
$this->assertEquals(count($bn), 3); $this->assertEquals(count($bn), 4);
$this->assertEquals($bn[0]->toString(), '0'); $this->assertEquals($bn[0]->toString(), '0');
$this->assertEquals($bn[1]->toString(), '1'); $this->assertEquals($bn[1]->toString(), '1');
$this->assertEquals($bn[2], false); $this->assertEquals($bn[2], 1);
$this->assertEquals($bn[3], false);
$bn = Utils::toBn('-1.69'); $bn = Utils::toBn('-1.69');
$this->assertEquals(count($bn), 3); $this->assertEquals(count($bn), 4);
$this->assertEquals($bn[0]->toString(), '1'); $this->assertEquals($bn[0]->toString(), '1');
$this->assertEquals($bn[1]->toString(), '69'); $this->assertEquals($bn[1]->toString(), '69');
$this->assertEquals($bn[2]->toString(), '-1'); $this->assertEquals($bn[2], 2);
$this->assertEquals($bn[3]->toString(), '-1');
$bn = Utils::toBn(-1.69); $bn = Utils::toBn(-1.69);
$this->assertEquals($bn[0]->toString(), '1'); $this->assertEquals($bn[0]->toString(), '1');
$this->assertEquals($bn[1]->toString(), '69'); $this->assertEquals($bn[1]->toString(), '69');
$this->assertEquals($bn[2]->toString(), '-1'); $this->assertEquals($bn[2], 2);
$this->assertEquals($bn[3]->toString(), '-1');
$bn = Utils::toBn('1.69'); $bn = Utils::toBn('1.69');
$this->assertEquals(count($bn), 3); $this->assertEquals(count($bn), 4);
$this->assertEquals($bn[0]->toString(), '1'); $this->assertEquals($bn[0]->toString(), '1');
$this->assertEquals($bn[1]->toString(), '69'); $this->assertEquals($bn[1]->toString(), '69');
$this->assertEquals($bn[2], false); $this->assertEquals($bn[2], 2);
$this->assertEquals($bn[3], false);
$bn = Utils::toBn(new BigNumber(1)); $bn = Utils::toBn(new BigNumber(1));
$this->assertEquals($bn->toString(), '1'); $this->assertEquals($bn->toString(), '1');

View File

@ -29,7 +29,7 @@ class Web3ApiTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
} }

View File

@ -28,7 +28,7 @@ class Web3BatchTest extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
} }

View File

@ -37,7 +37,7 @@ class Web3Test extends TestCase
* *
* @return void * @return void
*/ */
public function setUp() public function setUp(): void
{ {
parent::setUp(); parent::setUp();
} }