Merge remote-tracking branch 'origin/master' into fix-tx-rpc
This commit is contained in:
commit
5e1fdc4b11
165
Cargo.lock
generated
165
Cargo.lock
generated
@ -24,10 +24,9 @@ dependencies = [
|
|||||||
"ethcore-stratum 1.4.0",
|
"ethcore-stratum 1.4.0",
|
||||||
"ethcore-util 1.5.0",
|
"ethcore-util 1.5.0",
|
||||||
"ethsync 1.5.0",
|
"ethsync 1.5.0",
|
||||||
"fdlimit 0.1.0",
|
"fdlimit 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"hyper 0.9.10 (registry+https://github.com/rust-lang/crates.io-index)",
|
"hyper 0.9.10 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"isatty 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
"isatty 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"json-ipc-server 0.2.4 (git+https://github.com/ethcore/json-ipc-server.git)",
|
|
||||||
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"num_cpus 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)",
|
"num_cpus 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
@ -175,6 +174,15 @@ dependencies = [
|
|||||||
"url 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"url 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "cookie"
|
||||||
|
version = "0.3.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
dependencies = [
|
||||||
|
"time 0.1.35 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"url 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crossbeam"
|
name = "crossbeam"
|
||||||
version = "0.2.9"
|
version = "0.2.9"
|
||||||
@ -296,7 +304,7 @@ dependencies = [
|
|||||||
"ethstore 0.1.0",
|
"ethstore 0.1.0",
|
||||||
"evmjit 1.4.0",
|
"evmjit 1.4.0",
|
||||||
"heapsize 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"heapsize 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"hyper 0.9.4 (git+https://github.com/ethcore/hyper)",
|
"hyper 0.10.0-a.0 (git+https://github.com/ethcore/hyper)",
|
||||||
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"linked-hash-map 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"linked-hash-map 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
@ -340,9 +348,9 @@ dependencies = [
|
|||||||
"ethcore-rpc 1.5.0",
|
"ethcore-rpc 1.5.0",
|
||||||
"ethcore-util 1.5.0",
|
"ethcore-util 1.5.0",
|
||||||
"fetch 0.1.0",
|
"fetch 0.1.0",
|
||||||
"hyper 0.9.4 (git+https://github.com/ethcore/hyper)",
|
"hyper 0.10.0-a.0 (git+https://github.com/ethcore/hyper)",
|
||||||
"jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
"jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"jsonrpc-http-server 6.1.1 (git+https://github.com/ethcore/jsonrpc-http-server.git)",
|
"jsonrpc-http-server 6.1.1 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"linked-hash-map 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"linked-hash-map 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"mime 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"mime 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
@ -503,9 +511,9 @@ dependencies = [
|
|||||||
"ethstore 0.1.0",
|
"ethstore 0.1.0",
|
||||||
"ethsync 1.5.0",
|
"ethsync 1.5.0",
|
||||||
"fetch 0.1.0",
|
"fetch 0.1.0",
|
||||||
"json-ipc-server 0.2.4 (git+https://github.com/ethcore/json-ipc-server.git)",
|
"jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
"jsonrpc-http-server 6.1.1 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"jsonrpc-http-server 6.1.1 (git+https://github.com/ethcore/jsonrpc-http-server.git)",
|
"jsonrpc-ipc-server 0.2.4 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"rlp 0.1.0",
|
"rlp 0.1.0",
|
||||||
"rustc-serialize 0.3.19 (registry+https://github.com/rust-lang/crates.io-index)",
|
"rustc-serialize 0.3.19 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
@ -526,7 +534,7 @@ dependencies = [
|
|||||||
"ethcore-io 1.5.0",
|
"ethcore-io 1.5.0",
|
||||||
"ethcore-rpc 1.5.0",
|
"ethcore-rpc 1.5.0",
|
||||||
"ethcore-util 1.5.0",
|
"ethcore-util 1.5.0",
|
||||||
"jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
"jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"parity-dapps-glue 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"parity-dapps-glue 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"parity-ui 1.4.0",
|
"parity-ui 1.4.0",
|
||||||
@ -545,8 +553,8 @@ dependencies = [
|
|||||||
"ethcore-ipc-codegen 1.4.0",
|
"ethcore-ipc-codegen 1.4.0",
|
||||||
"ethcore-ipc-nano 1.4.0",
|
"ethcore-ipc-nano 1.4.0",
|
||||||
"ethcore-util 1.5.0",
|
"ethcore-util 1.5.0",
|
||||||
"json-tcp-server 0.1.0 (git+https://github.com/ethcore/json-tcp-server)",
|
"jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
"jsonrpc-tcp-server 0.1.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"mio 0.5.1 (git+https://github.com/ethcore/mio?branch=v0.5.x)",
|
"mio 0.5.1 (git+https://github.com/ethcore/mio?branch=v0.5.x)",
|
||||||
@ -676,7 +684,8 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fdlimit"
|
name = "fdlimit"
|
||||||
version = "0.1.0"
|
version = "0.1.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"libc 0.2.16 (registry+https://github.com/rust-lang/crates.io-index)",
|
"libc 0.2.16 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
@ -686,7 +695,7 @@ name = "fetch"
|
|||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"https-fetch 0.1.0",
|
"https-fetch 0.1.0",
|
||||||
"hyper 0.9.4 (git+https://github.com/ethcore/hyper)",
|
"hyper 0.10.0-a.0 (git+https://github.com/ethcore/hyper)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"rand 0.3.14 (registry+https://github.com/rust-lang/crates.io-index)",
|
"rand 0.3.14 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
@ -748,27 +757,6 @@ dependencies = [
|
|||||||
"rustls 0.1.2 (git+https://github.com/ctz/rustls)",
|
"rustls 0.1.2 (git+https://github.com/ctz/rustls)",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "hyper"
|
|
||||||
version = "0.9.4"
|
|
||||||
source = "git+https://github.com/ethcore/hyper#9e346c1d4bc30cd4142dea9d8a0b117d30858ca4"
|
|
||||||
dependencies = [
|
|
||||||
"cookie 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"httparse 1.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"mime 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"rotor 0.6.3 (git+https://github.com/ethcore/rotor)",
|
|
||||||
"rustc-serialize 0.3.19 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"spmc 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"time 0.1.35 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"traitobject 0.0.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"typeable 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"unicase 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"url 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"vecio 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "hyper"
|
name = "hyper"
|
||||||
version = "0.9.10"
|
version = "0.9.10"
|
||||||
@ -789,6 +777,25 @@ dependencies = [
|
|||||||
"url 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"url 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "hyper"
|
||||||
|
version = "0.10.0-a.0"
|
||||||
|
source = "git+https://github.com/ethcore/hyper#7d4f7fa0baddcb2b0c523f7c05855d67de94fe88"
|
||||||
|
dependencies = [
|
||||||
|
"cookie 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"httparse 1.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"mime 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"rotor 0.6.3 (git+https://github.com/ethcore/rotor)",
|
||||||
|
"rustc-serialize 0.3.19 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"spmc 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"time 0.1.35 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"unicase 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"url 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"vecio 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "idna"
|
name = "idna"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
@ -831,39 +838,10 @@ name = "itoa"
|
|||||||
version = "0.1.1"
|
version = "0.1.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "json-ipc-server"
|
|
||||||
version = "0.2.4"
|
|
||||||
source = "git+https://github.com/ethcore/json-ipc-server.git#4642cd03ec1d23db89df80d22d5a88e7364ab885"
|
|
||||||
dependencies = [
|
|
||||||
"bytes 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"mio 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"miow 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"slab 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "json-tcp-server"
|
|
||||||
version = "0.1.0"
|
|
||||||
source = "git+https://github.com/ethcore/json-tcp-server#c2858522274ae56042472bb5d22845a1b85e5338"
|
|
||||||
dependencies = [
|
|
||||||
"bytes 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"mio 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"slab 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "jsonrpc-core"
|
name = "jsonrpc-core"
|
||||||
version = "3.0.2"
|
version = "4.0.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "git+https://github.com/ethcore/jsonrpc.git#20c7e55b84d7fd62732f062dc3058e1b71133e4a"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"parking_lot 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)",
|
"parking_lot 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
@ -875,14 +853,44 @@ dependencies = [
|
|||||||
[[package]]
|
[[package]]
|
||||||
name = "jsonrpc-http-server"
|
name = "jsonrpc-http-server"
|
||||||
version = "6.1.1"
|
version = "6.1.1"
|
||||||
source = "git+https://github.com/ethcore/jsonrpc-http-server.git#cd6d4cb37d672cc3057aecd0692876f9e85f3ba5"
|
source = "git+https://github.com/ethcore/jsonrpc.git#20c7e55b84d7fd62732f062dc3058e1b71133e4a"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"hyper 0.9.4 (git+https://github.com/ethcore/hyper)",
|
"hyper 0.10.0-a.0 (git+https://github.com/ethcore/hyper)",
|
||||||
"jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
"jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"unicase 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"unicase 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "jsonrpc-ipc-server"
|
||||||
|
version = "0.2.4"
|
||||||
|
source = "git+https://github.com/ethcore/jsonrpc.git#20c7e55b84d7fd62732f062dc3058e1b71133e4a"
|
||||||
|
dependencies = [
|
||||||
|
"bytes 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
|
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"mio 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"miow 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"slab 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "jsonrpc-tcp-server"
|
||||||
|
version = "0.1.0"
|
||||||
|
source = "git+https://github.com/ethcore/jsonrpc.git#20c7e55b84d7fd62732f062dc3058e1b71133e4a"
|
||||||
|
dependencies = [
|
||||||
|
"bytes 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)",
|
||||||
|
"lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"mio 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"serde_json 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"slab 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kernel32-sys"
|
name = "kernel32-sys"
|
||||||
version = "0.2.2"
|
version = "0.2.2"
|
||||||
@ -1263,7 +1271,7 @@ dependencies = [
|
|||||||
[[package]]
|
[[package]]
|
||||||
name = "parity-ui-precompiled"
|
name = "parity-ui-precompiled"
|
||||||
version = "1.4.0"
|
version = "1.4.0"
|
||||||
source = "git+https://github.com/ethcore/js-precompiled.git#d99369b5010cfa73c3a62eb8c031bf873a7ca709"
|
source = "git+https://github.com/ethcore/js-precompiled.git#3cf6c68b7d08be71d12ff142e255a42043e50c75"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"parity-dapps-glue 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"parity-dapps-glue 1.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
@ -1503,11 +1511,12 @@ dependencies = [
|
|||||||
[[package]]
|
[[package]]
|
||||||
name = "rotor"
|
name = "rotor"
|
||||||
version = "0.6.3"
|
version = "0.6.3"
|
||||||
source = "git+https://github.com/ethcore/rotor#e63d45137b2eb66d1e085a7c6321a5db8b187576"
|
source = "git+https://github.com/ethcore/rotor#c1a2dd0046c5ea2517a5b637fca8ee2e77021e82"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"mio 0.5.1 (git+https://github.com/ethcore/mio?branch=v0.5.x)",
|
"mio 0.6.1 (git+https://github.com/ethcore/mio)",
|
||||||
"quick-error 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"quick-error 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"slab 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"void 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
"void 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -2008,6 +2017,7 @@ dependencies = [
|
|||||||
"checksum clippy 0.0.103 (registry+https://github.com/rust-lang/crates.io-index)" = "5b4fabf979ddf6419a313c1c0ada4a5b95cfd2049c56e8418d622d27b4b6ff32"
|
"checksum clippy 0.0.103 (registry+https://github.com/rust-lang/crates.io-index)" = "5b4fabf979ddf6419a313c1c0ada4a5b95cfd2049c56e8418d622d27b4b6ff32"
|
||||||
"checksum clippy_lints 0.0.103 (registry+https://github.com/rust-lang/crates.io-index)" = "ce96ec05bfe018a0d5d43da115e54850ea2217981ff0f2e462780ab9d594651a"
|
"checksum clippy_lints 0.0.103 (registry+https://github.com/rust-lang/crates.io-index)" = "ce96ec05bfe018a0d5d43da115e54850ea2217981ff0f2e462780ab9d594651a"
|
||||||
"checksum cookie 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)" = "90266f45846f14a1e986c77d1e9c2626b8c342ed806fe60241ec38cc8697b245"
|
"checksum cookie 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)" = "90266f45846f14a1e986c77d1e9c2626b8c342ed806fe60241ec38cc8697b245"
|
||||||
|
"checksum cookie 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)" = "d53b80dde876f47f03cda35303e368a79b91c70b0d65ecba5fd5280944a08591"
|
||||||
"checksum crossbeam 0.2.9 (registry+https://github.com/rust-lang/crates.io-index)" = "fb974f835e90390c5f9dfac00f05b06dc117299f5ea4e85fbc7bb443af4911cc"
|
"checksum crossbeam 0.2.9 (registry+https://github.com/rust-lang/crates.io-index)" = "fb974f835e90390c5f9dfac00f05b06dc117299f5ea4e85fbc7bb443af4911cc"
|
||||||
"checksum ctrlc 1.1.1 (git+https://github.com/ethcore/rust-ctrlc.git)" = "<none>"
|
"checksum ctrlc 1.1.1 (git+https://github.com/ethcore/rust-ctrlc.git)" = "<none>"
|
||||||
"checksum daemonize 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "271ec51b7e0bee92f0d04601422c73eb76ececf197026711c97ad25038a010cf"
|
"checksum daemonize 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "271ec51b7e0bee92f0d04601422c73eb76ececf197026711c97ad25038a010cf"
|
||||||
@ -2018,6 +2028,7 @@ dependencies = [
|
|||||||
"checksum env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "aba65b63ffcc17ffacd6cf5aa843da7c5a25e3bd4bbe0b7def8b214e411250e5"
|
"checksum env_logger 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "aba65b63ffcc17ffacd6cf5aa843da7c5a25e3bd4bbe0b7def8b214e411250e5"
|
||||||
"checksum eth-secp256k1 0.5.4 (git+https://github.com/ethcore/rust-secp256k1)" = "<none>"
|
"checksum eth-secp256k1 0.5.4 (git+https://github.com/ethcore/rust-secp256k1)" = "<none>"
|
||||||
"checksum ethabi 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "f7b0c53453517f620847be51943db329276ae52f2e210cfc659e81182864be2f"
|
"checksum ethabi 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "f7b0c53453517f620847be51943db329276ae52f2e210cfc659e81182864be2f"
|
||||||
|
"checksum fdlimit 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "b1ee15a7050e5580b3712877157068ea713b245b080ff302ae2ca973cfcd9baa"
|
||||||
"checksum flate2 0.2.14 (registry+https://github.com/rust-lang/crates.io-index)" = "3eeb481e957304178d2e782f2da1257f1434dfecbae883bafb61ada2a9fea3bb"
|
"checksum flate2 0.2.14 (registry+https://github.com/rust-lang/crates.io-index)" = "3eeb481e957304178d2e782f2da1257f1434dfecbae883bafb61ada2a9fea3bb"
|
||||||
"checksum gcc 0.3.35 (registry+https://github.com/rust-lang/crates.io-index)" = "91ecd03771effb0c968fd6950b37e89476a578aaf1c70297d8e92b6516ec3312"
|
"checksum gcc 0.3.35 (registry+https://github.com/rust-lang/crates.io-index)" = "91ecd03771effb0c968fd6950b37e89476a578aaf1c70297d8e92b6516ec3312"
|
||||||
"checksum glob 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)" = "8be18de09a56b60ed0edf84bc9df007e30040691af7acd1c41874faac5895bfb"
|
"checksum glob 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)" = "8be18de09a56b60ed0edf84bc9df007e30040691af7acd1c41874faac5895bfb"
|
||||||
@ -2025,17 +2036,17 @@ dependencies = [
|
|||||||
"checksum heapsize 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)" = "abb306abb8d398e053cfb1b3e7b72c2f580be048b85745c52652954f8ad1439c"
|
"checksum heapsize 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)" = "abb306abb8d398e053cfb1b3e7b72c2f580be048b85745c52652954f8ad1439c"
|
||||||
"checksum hpack 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "3d2da7d3a34cf6406d9d700111b8eafafe9a251de41ae71d8052748259343b58"
|
"checksum hpack 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "3d2da7d3a34cf6406d9d700111b8eafafe9a251de41ae71d8052748259343b58"
|
||||||
"checksum httparse 1.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "46534074dbb80b070d60a5cb8ecadd8963a00a438ae1a95268850a7ef73b67ae"
|
"checksum httparse 1.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "46534074dbb80b070d60a5cb8ecadd8963a00a438ae1a95268850a7ef73b67ae"
|
||||||
|
"checksum hyper 0.10.0-a.0 (git+https://github.com/ethcore/hyper)" = "<none>"
|
||||||
"checksum hyper 0.9.10 (registry+https://github.com/rust-lang/crates.io-index)" = "eb27e8a3e8f17ac43ffa41bbda9cf5ad3f9f13ef66fa4873409d4902310275f7"
|
"checksum hyper 0.9.10 (registry+https://github.com/rust-lang/crates.io-index)" = "eb27e8a3e8f17ac43ffa41bbda9cf5ad3f9f13ef66fa4873409d4902310275f7"
|
||||||
"checksum hyper 0.9.4 (git+https://github.com/ethcore/hyper)" = "<none>"
|
|
||||||
"checksum idna 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "1053236e00ce4f668aeca4a769a09b3bf5a682d802abd6f3cb39374f6b162c11"
|
"checksum idna 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "1053236e00ce4f668aeca4a769a09b3bf5a682d802abd6f3cb39374f6b162c11"
|
||||||
"checksum igd 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)" = "c8c12b1795b8b168f577c45fa10379b3814dcb11b7ab702406001f0d63f40484"
|
"checksum igd 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)" = "c8c12b1795b8b168f577c45fa10379b3814dcb11b7ab702406001f0d63f40484"
|
||||||
"checksum isatty 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "7408a548dc0e406b7912d9f84c261cc533c1866e047644a811c133c56041ac0c"
|
"checksum isatty 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "7408a548dc0e406b7912d9f84c261cc533c1866e047644a811c133c56041ac0c"
|
||||||
"checksum itertools 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)" = "086e1fa5fe48840b1cfdef3a20c7e3115599f8d5c4c87ef32a794a7cdd184d76"
|
"checksum itertools 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)" = "086e1fa5fe48840b1cfdef3a20c7e3115599f8d5c4c87ef32a794a7cdd184d76"
|
||||||
"checksum itoa 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "ae3088ea4baeceb0284ee9eea42f591226e6beaecf65373e41b38d95a1b8e7a1"
|
"checksum itoa 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "ae3088ea4baeceb0284ee9eea42f591226e6beaecf65373e41b38d95a1b8e7a1"
|
||||||
"checksum json-ipc-server 0.2.4 (git+https://github.com/ethcore/json-ipc-server.git)" = "<none>"
|
"checksum jsonrpc-core 4.0.0 (git+https://github.com/ethcore/jsonrpc.git)" = "<none>"
|
||||||
"checksum json-tcp-server 0.1.0 (git+https://github.com/ethcore/json-tcp-server)" = "<none>"
|
"checksum jsonrpc-http-server 6.1.1 (git+https://github.com/ethcore/jsonrpc.git)" = "<none>"
|
||||||
"checksum jsonrpc-core 3.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "3c5094610b07f28f3edaf3947b732dadb31dbba4941d4d0c1c7a8350208f4414"
|
"checksum jsonrpc-ipc-server 0.2.4 (git+https://github.com/ethcore/jsonrpc.git)" = "<none>"
|
||||||
"checksum jsonrpc-http-server 6.1.1 (git+https://github.com/ethcore/jsonrpc-http-server.git)" = "<none>"
|
"checksum jsonrpc-tcp-server 0.1.0 (git+https://github.com/ethcore/jsonrpc.git)" = "<none>"
|
||||||
"checksum kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7507624b29483431c0ba2d82aece8ca6cdba9382bff4ddd0f7490560c056098d"
|
"checksum kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7507624b29483431c0ba2d82aece8ca6cdba9382bff4ddd0f7490560c056098d"
|
||||||
"checksum language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "a91d884b6667cd606bb5a69aa0c99ba811a115fc68915e7056ec08a46e93199a"
|
"checksum language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "a91d884b6667cd606bb5a69aa0c99ba811a115fc68915e7056ec08a46e93199a"
|
||||||
"checksum lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "49247ec2a285bb3dcb23cbd9c35193c025e7251bfce77c1d5da97e6362dffe7f"
|
"checksum lazy_static 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "49247ec2a285bb3dcb23cbd9c35193c025e7251bfce77c1d5da97e6362dffe7f"
|
||||||
|
@ -30,8 +30,7 @@ serde = "0.8.0"
|
|||||||
serde_json = "0.8.0"
|
serde_json = "0.8.0"
|
||||||
hyper = { version = "0.9", default-features = false }
|
hyper = { version = "0.9", default-features = false }
|
||||||
ctrlc = { git = "https://github.com/ethcore/rust-ctrlc.git" }
|
ctrlc = { git = "https://github.com/ethcore/rust-ctrlc.git" }
|
||||||
json-ipc-server = { git = "https://github.com/ethcore/json-ipc-server.git" }
|
fdlimit = "0.1"
|
||||||
fdlimit = { path = "util/fdlimit" }
|
|
||||||
ethcore = { path = "ethcore" }
|
ethcore = { path = "ethcore" }
|
||||||
ethcore-util = { path = "util" }
|
ethcore-util = { path = "util" }
|
||||||
ethsync = { path = "sync" }
|
ethsync = { path = "sync" }
|
||||||
|
@ -12,8 +12,8 @@ build = "build.rs"
|
|||||||
rand = "0.3.14"
|
rand = "0.3.14"
|
||||||
log = "0.3"
|
log = "0.3"
|
||||||
env_logger = "0.3"
|
env_logger = "0.3"
|
||||||
jsonrpc-core = "3.0"
|
jsonrpc-core = { git = "https://github.com/ethcore/jsonrpc.git" }
|
||||||
jsonrpc-http-server = { git = "https://github.com/ethcore/jsonrpc-http-server.git" }
|
jsonrpc-http-server = { git = "https://github.com/ethcore/jsonrpc.git" }
|
||||||
hyper = { default-features = false, git = "https://github.com/ethcore/hyper" }
|
hyper = { default-features = false, git = "https://github.com/ethcore/hyper" }
|
||||||
unicase = "1.3"
|
unicase = "1.3"
|
||||||
url = "1.0"
|
url = "1.0"
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
|
|
||||||
//! Simple Content Handler
|
//! Simple Content Handler
|
||||||
|
|
||||||
use std::io::Write;
|
|
||||||
use hyper::{header, server, Decoder, Encoder, Next};
|
use hyper::{header, server, Decoder, Encoder, Next};
|
||||||
use hyper::net::HttpStream;
|
use hyper::net::HttpStream;
|
||||||
use hyper::mime::Mime;
|
use hyper::mime::Mime;
|
||||||
|
@ -58,7 +58,7 @@ pub fn extract_url(req: &server::Request<net::HttpStream>) -> Option<Url> {
|
|||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
uri::RequestUri::AbsolutePath(ref path) => {
|
uri::RequestUri::AbsolutePath { ref path, .. } => {
|
||||||
// Attempt to prepend the Host header (mandatory in HTTP/1.1)
|
// Attempt to prepend the Host header (mandatory in HTTP/1.1)
|
||||||
let url_string = match req.headers().get::<header::Host>() {
|
let url_string = match req.headers().get::<header::Host>() {
|
||||||
Some(ref host) => {
|
Some(ref host) => {
|
||||||
|
@ -266,7 +266,11 @@ impl Server {
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
/// Returns address that this server is bound to.
|
/// Returns address that this server is bound to.
|
||||||
pub fn addr(&self) -> &SocketAddr {
|
pub fn addr(&self) -> &SocketAddr {
|
||||||
self.server.as_ref().expect("server is always Some at the start; it's consumed only when object is dropped; qed").addr()
|
self.server.as_ref()
|
||||||
|
.expect("server is always Some at the start; it's consumed only when object is dropped; qed")
|
||||||
|
.addrs()
|
||||||
|
.first()
|
||||||
|
.expect("You cannot start the server without binding to at least one address; qed")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,7 +14,6 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
use std::io::Write;
|
|
||||||
use time::{self, Duration};
|
use time::{self, Duration};
|
||||||
|
|
||||||
use hyper::header;
|
use hyper::header;
|
||||||
@ -126,7 +125,7 @@ impl<T: Dapp> PageHandler<T> {
|
|||||||
impl<T: Dapp> server::Handler<HttpStream> for PageHandler<T> {
|
impl<T: Dapp> server::Handler<HttpStream> for PageHandler<T> {
|
||||||
fn on_request(&mut self, req: server::Request<HttpStream>) -> Next {
|
fn on_request(&mut self, req: server::Request<HttpStream>) -> Next {
|
||||||
self.file = match *req.uri() {
|
self.file = match *req.uri() {
|
||||||
RequestUri::AbsolutePath(ref path) => {
|
RequestUri::AbsolutePath { ref path, .. } => {
|
||||||
self.app.file(&self.extract_path(path))
|
self.app.file(&self.extract_path(path))
|
||||||
},
|
},
|
||||||
RequestUri::AbsoluteUri(ref url) => {
|
RequestUri::AbsoluteUri(ref url) => {
|
||||||
|
@ -16,13 +16,14 @@
|
|||||||
|
|
||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc, Mutex};
|
||||||
use hyper;
|
use hyper;
|
||||||
use jsonrpc_core::IoHandler;
|
|
||||||
use jsonrpc_http_server::{ServerHandler, PanicHandler, AccessControlAllowOrigin};
|
use jsonrpc_core::{IoHandler, ResponseHandler, Request, Response};
|
||||||
|
use jsonrpc_http_server::{ServerHandler, PanicHandler, AccessControlAllowOrigin, RpcHandler};
|
||||||
use endpoint::{Endpoint, EndpointPath, Handler};
|
use endpoint::{Endpoint, EndpointPath, Handler};
|
||||||
|
|
||||||
pub fn rpc(handler: Arc<IoHandler>, panic_handler: Arc<Mutex<Option<Box<Fn() -> () + Send>>>>) -> Box<Endpoint> {
|
pub fn rpc(handler: Arc<IoHandler>, panic_handler: Arc<Mutex<Option<Box<Fn() -> () + Send>>>>) -> Box<Endpoint> {
|
||||||
Box::new(RpcEndpoint {
|
Box::new(RpcEndpoint {
|
||||||
handler: handler,
|
handler: Arc::new(RpcMiddleware::new(handler)),
|
||||||
panic_handler: panic_handler,
|
panic_handler: panic_handler,
|
||||||
cors_domain: None,
|
cors_domain: None,
|
||||||
// NOTE [ToDr] We don't need to do any hosts validation here. It's already done in router.
|
// NOTE [ToDr] We don't need to do any hosts validation here. It's already done in router.
|
||||||
@ -31,7 +32,7 @@ pub fn rpc(handler: Arc<IoHandler>, panic_handler: Arc<Mutex<Option<Box<Fn() ->
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct RpcEndpoint {
|
struct RpcEndpoint {
|
||||||
handler: Arc<IoHandler>,
|
handler: Arc<RpcMiddleware>,
|
||||||
panic_handler: Arc<Mutex<Option<Box<Fn() -> () + Send>>>>,
|
panic_handler: Arc<Mutex<Option<Box<Fn() -> () + Send>>>>,
|
||||||
cors_domain: Option<Vec<AccessControlAllowOrigin>>,
|
cors_domain: Option<Vec<AccessControlAllowOrigin>>,
|
||||||
allowed_hosts: Option<Vec<String>>,
|
allowed_hosts: Option<Vec<String>>,
|
||||||
@ -49,3 +50,86 @@ impl Endpoint for RpcEndpoint {
|
|||||||
))
|
))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const MIDDLEWARE_METHOD: &'static str = "eth_accounts";
|
||||||
|
|
||||||
|
struct RpcMiddleware {
|
||||||
|
handler: Arc<IoHandler>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RpcMiddleware {
|
||||||
|
fn new(handler: Arc<IoHandler>) -> Self {
|
||||||
|
RpcMiddleware {
|
||||||
|
handler: handler,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Appends additional parameter for specific calls.
|
||||||
|
fn augment_request(&self, request: &mut Request, meta: Option<Meta>) {
|
||||||
|
use jsonrpc_core::{Call, Params, to_value};
|
||||||
|
|
||||||
|
fn augment_call(call: &mut Call, meta: Option<&Meta>) {
|
||||||
|
match (call, meta) {
|
||||||
|
(&mut Call::MethodCall(ref mut method_call), Some(meta)) if &method_call.method == MIDDLEWARE_METHOD => {
|
||||||
|
let session = to_value(&meta.app_id);
|
||||||
|
|
||||||
|
let params = match method_call.params {
|
||||||
|
Some(Params::Array(ref vec)) if vec.len() == 0 => Some(Params::Array(vec![session])),
|
||||||
|
// invalid params otherwise
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
|
||||||
|
method_call.params = params;
|
||||||
|
},
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
match *request {
|
||||||
|
Request::Single(ref mut call) => augment_call(call, meta.as_ref()),
|
||||||
|
Request::Batch(ref mut vec) => {
|
||||||
|
for mut call in vec {
|
||||||
|
augment_call(call, meta.as_ref())
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct Meta {
|
||||||
|
app_id: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RpcHandler for RpcMiddleware {
|
||||||
|
type Metadata = Meta;
|
||||||
|
|
||||||
|
fn read_metadata(&self, request: &hyper::server::Request<hyper::net::HttpStream>) -> Option<Self::Metadata> {
|
||||||
|
request.headers().get::<hyper::header::Referer>()
|
||||||
|
.and_then(|referer| hyper::Url::parse(referer).ok())
|
||||||
|
.and_then(|url| {
|
||||||
|
url.path_segments()
|
||||||
|
.and_then(|mut split| split.next())
|
||||||
|
.map(|app_id| Meta {
|
||||||
|
app_id: app_id.to_owned(),
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn handle_request<H>(&self, request_str: &str, response_handler: H, meta: Option<Self::Metadata>) where
|
||||||
|
H: ResponseHandler<Option<String>, Option<String>> + 'static
|
||||||
|
{
|
||||||
|
let handler = IoHandler::convert_handler(response_handler);
|
||||||
|
let request = IoHandler::read_request(request_str);
|
||||||
|
trace!(target: "rpc", "Request metadata: {:?}", meta);
|
||||||
|
|
||||||
|
match request {
|
||||||
|
Ok(mut request) => {
|
||||||
|
self.augment_request(&mut request, meta);
|
||||||
|
self.handler.request_handler().handle_request(request, handler, None)
|
||||||
|
},
|
||||||
|
Err(error) => handler.send(Some(Response::from(error))),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ -21,8 +21,7 @@
|
|||||||
"genesis": {
|
"genesis": {
|
||||||
"seal": {
|
"seal": {
|
||||||
"generic": {
|
"generic": {
|
||||||
"fields": 2,
|
"rlp": "0xc28080"
|
||||||
"rlp": "0x200"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"difficulty": "0x20000",
|
"difficulty": "0x20000",
|
||||||
|
@ -12,7 +12,6 @@
|
|||||||
"genesis": {
|
"genesis": {
|
||||||
"seal": {
|
"seal": {
|
||||||
"generic": {
|
"generic": {
|
||||||
"fields": 0,
|
|
||||||
"rlp": "0x0"
|
"rlp": "0x0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -16,9 +16,12 @@
|
|||||||
|
|
||||||
//! Account management.
|
//! Account management.
|
||||||
|
|
||||||
use std::{fs, fmt};
|
mod stores;
|
||||||
|
|
||||||
|
use self::stores::{AddressBook, DappsSettingsStore};
|
||||||
|
|
||||||
|
use std::fmt;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::path::PathBuf;
|
|
||||||
use std::time::{Instant, Duration};
|
use std::time::{Instant, Duration};
|
||||||
use util::{Mutex, RwLock};
|
use util::{Mutex, RwLock};
|
||||||
use ethstore::{SecretStore, Error as SSError, SafeAccount, EthStore};
|
use ethstore::{SecretStore, Error as SSError, SafeAccount, EthStore};
|
||||||
@ -91,84 +94,16 @@ impl KeyDirectory for NullDir {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Disk-backed map from Address to String. Uses JSON.
|
/// Dapp identifier
|
||||||
struct AddressBook {
|
pub type DappId = String;
|
||||||
path: PathBuf,
|
|
||||||
cache: HashMap<Address, AccountMeta>,
|
|
||||||
transient: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AddressBook {
|
|
||||||
pub fn new(path: String) -> Self {
|
|
||||||
trace!(target: "addressbook", "new({})", path);
|
|
||||||
let mut path: PathBuf = path.into();
|
|
||||||
path.push("address_book.json");
|
|
||||||
trace!(target: "addressbook", "path={:?}", path);
|
|
||||||
let mut r = AddressBook {
|
|
||||||
path: path,
|
|
||||||
cache: HashMap::new(),
|
|
||||||
transient: false,
|
|
||||||
};
|
|
||||||
r.revert();
|
|
||||||
r
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn transient() -> Self {
|
|
||||||
let mut book = AddressBook::new(Default::default());
|
|
||||||
book.transient = true;
|
|
||||||
book
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get(&self) -> HashMap<Address, AccountMeta> {
|
|
||||||
self.cache.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_name(&mut self, a: Address, name: String) {
|
|
||||||
let mut x = self.cache.get(&a)
|
|
||||||
.cloned()
|
|
||||||
.unwrap_or_else(|| AccountMeta {name: Default::default(), meta: "{}".to_owned(), uuid: None});
|
|
||||||
x.name = name;
|
|
||||||
self.cache.insert(a, x);
|
|
||||||
self.save();
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_meta(&mut self, a: Address, meta: String) {
|
|
||||||
let mut x = self.cache.get(&a)
|
|
||||||
.cloned()
|
|
||||||
.unwrap_or_else(|| AccountMeta {name: "Anonymous".to_owned(), meta: Default::default(), uuid: None});
|
|
||||||
x.meta = meta;
|
|
||||||
self.cache.insert(a, x);
|
|
||||||
self.save();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn revert(&mut self) {
|
|
||||||
if self.transient { return; }
|
|
||||||
trace!(target: "addressbook", "revert");
|
|
||||||
let _ = fs::File::open(self.path.clone())
|
|
||||||
.map_err(|e| trace!(target: "addressbook", "Couldn't open address book: {}", e))
|
|
||||||
.and_then(|f| AccountMeta::read_address_map(&f)
|
|
||||||
.map_err(|e| warn!(target: "addressbook", "Couldn't read address book: {}", e))
|
|
||||||
.and_then(|m| { self.cache = m; Ok(()) })
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn save(&mut self) {
|
|
||||||
if self.transient { return; }
|
|
||||||
trace!(target: "addressbook", "save");
|
|
||||||
let _ = fs::File::create(self.path.clone())
|
|
||||||
.map_err(|e| warn!(target: "addressbook", "Couldn't open address book for writing: {}", e))
|
|
||||||
.and_then(|mut f| AccountMeta::write_address_map(&self.cache, &mut f)
|
|
||||||
.map_err(|e| warn!(target: "addressbook", "Couldn't write to address book: {}", e))
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Account management.
|
/// Account management.
|
||||||
/// Responsible for unlocking accounts.
|
/// Responsible for unlocking accounts.
|
||||||
pub struct AccountProvider {
|
pub struct AccountProvider {
|
||||||
unlocked: Mutex<HashMap<Address, AccountData>>,
|
unlocked: Mutex<HashMap<Address, AccountData>>,
|
||||||
sstore: Box<SecretStore>,
|
sstore: Box<SecretStore>,
|
||||||
address_book: Mutex<AddressBook>,
|
address_book: RwLock<AddressBook>,
|
||||||
|
dapps_settings: RwLock<DappsSettingsStore>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AccountProvider {
|
impl AccountProvider {
|
||||||
@ -176,7 +111,8 @@ impl AccountProvider {
|
|||||||
pub fn new(sstore: Box<SecretStore>) -> Self {
|
pub fn new(sstore: Box<SecretStore>) -> Self {
|
||||||
AccountProvider {
|
AccountProvider {
|
||||||
unlocked: Mutex::new(HashMap::new()),
|
unlocked: Mutex::new(HashMap::new()),
|
||||||
address_book: Mutex::new(AddressBook::new(sstore.local_path().into())),
|
address_book: RwLock::new(AddressBook::new(sstore.local_path().into())),
|
||||||
|
dapps_settings: RwLock::new(DappsSettingsStore::new(sstore.local_path().into())),
|
||||||
sstore: sstore,
|
sstore: sstore,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -185,7 +121,8 @@ impl AccountProvider {
|
|||||||
pub fn transient_provider() -> Self {
|
pub fn transient_provider() -> Self {
|
||||||
AccountProvider {
|
AccountProvider {
|
||||||
unlocked: Mutex::new(HashMap::new()),
|
unlocked: Mutex::new(HashMap::new()),
|
||||||
address_book: Mutex::new(AddressBook::transient()),
|
address_book: RwLock::new(AddressBook::transient()),
|
||||||
|
dapps_settings: RwLock::new(DappsSettingsStore::transient()),
|
||||||
sstore: Box::new(EthStore::open(Box::new(NullDir::default()))
|
sstore: Box::new(EthStore::open(Box::new(NullDir::default()))
|
||||||
.expect("NullDir load always succeeds; qed"))
|
.expect("NullDir load always succeeds; qed"))
|
||||||
}
|
}
|
||||||
@ -230,19 +167,31 @@ impl AccountProvider {
|
|||||||
Ok(accounts)
|
Ok(accounts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Gets addresses visile for dapp.
|
||||||
|
pub fn dapps_addresses(&self, dapp: DappId) -> Result<Vec<Address>, Error> {
|
||||||
|
let accounts = self.dapps_settings.read().get();
|
||||||
|
Ok(accounts.get(&dapp).map(|settings| settings.accounts.clone()).unwrap_or_else(Vec::new))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets addresses visile for dapp.
|
||||||
|
pub fn set_dapps_addresses(&self, dapp: DappId, addresses: Vec<Address>) -> Result<(), Error> {
|
||||||
|
self.dapps_settings.write().set_accounts(dapp, addresses);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns each address along with metadata.
|
/// Returns each address along with metadata.
|
||||||
pub fn addresses_info(&self) -> Result<HashMap<Address, AccountMeta>, Error> {
|
pub fn addresses_info(&self) -> Result<HashMap<Address, AccountMeta>, Error> {
|
||||||
Ok(self.address_book.lock().get())
|
Ok(self.address_book.read().get())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns each address along with metadata.
|
/// Returns each address along with metadata.
|
||||||
pub fn set_address_name(&self, account: Address, name: String) -> Result<(), Error> {
|
pub fn set_address_name(&self, account: Address, name: String) -> Result<(), Error> {
|
||||||
Ok(self.address_book.lock().set_name(account, name))
|
Ok(self.address_book.write().set_name(account, name))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns each address along with metadata.
|
/// Returns each address along with metadata.
|
||||||
pub fn set_address_meta(&self, account: Address, meta: String) -> Result<(), Error> {
|
pub fn set_address_meta(&self, account: Address, meta: String) -> Result<(), Error> {
|
||||||
Ok(self.address_book.lock().set_meta(account, meta))
|
Ok(self.address_book.write().set_meta(account, meta))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns each account along with name and meta.
|
/// Returns each account along with name and meta.
|
||||||
@ -379,23 +328,9 @@ impl AccountProvider {
|
|||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::{AccountProvider, AddressBook, Unlock};
|
use super::{AccountProvider, Unlock};
|
||||||
use std::collections::HashMap;
|
|
||||||
use std::time::Instant;
|
use std::time::Instant;
|
||||||
use ethjson::misc::AccountMeta;
|
|
||||||
use ethstore::ethkey::{Generator, Random};
|
use ethstore::ethkey::{Generator, Random};
|
||||||
use devtools::RandomTempPath;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn should_save_and_reload_address_book() {
|
|
||||||
let temp = RandomTempPath::create_dir();
|
|
||||||
let path = temp.as_str().to_owned();
|
|
||||||
let mut b = AddressBook::new(path.clone());
|
|
||||||
b.set_name(1.into(), "One".to_owned());
|
|
||||||
b.set_meta(1.into(), "{1:1}".to_owned());
|
|
||||||
let b = AddressBook::new(path);
|
|
||||||
assert_eq!(b.get(), hash_map![1.into() => AccountMeta{name: "One".to_owned(), meta: "{1:1}".to_owned(), uuid: None}]);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn unlock_account_temp() {
|
fn unlock_account_temp() {
|
||||||
@ -433,4 +368,17 @@ mod tests {
|
|||||||
ap.unlocked.lock().get_mut(&kp.address()).unwrap().unlock = Unlock::Timed(Instant::now());
|
ap.unlocked.lock().get_mut(&kp.address()).unwrap().unlock = Unlock::Timed(Instant::now());
|
||||||
assert!(ap.sign(kp.address(), None, Default::default()).is_err());
|
assert!(ap.sign(kp.address(), None, Default::default()).is_err());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn should_set_dapps_addresses() {
|
||||||
|
// given
|
||||||
|
let ap = AccountProvider::transient_provider();
|
||||||
|
let app = "app1".to_owned();
|
||||||
|
|
||||||
|
// when
|
||||||
|
ap.set_dapps_addresses(app.clone(), vec![1.into(), 2.into()]).unwrap();
|
||||||
|
|
||||||
|
// then
|
||||||
|
assert_eq!(ap.dapps_addresses(app.clone()).unwrap(), vec![1.into(), 2.into()]);
|
||||||
|
}
|
||||||
}
|
}
|
247
ethcore/src/account_provider/stores.rs
Normal file
247
ethcore/src/account_provider/stores.rs
Normal file
@ -0,0 +1,247 @@
|
|||||||
|
// Copyright 2015, 2016 Ethcore (UK) Ltd.
|
||||||
|
// This file is part of Parity.
|
||||||
|
|
||||||
|
// Parity is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
|
||||||
|
// Parity is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
//! Address Book and Dapps Settings Store
|
||||||
|
|
||||||
|
use std::{fs, fmt, hash, ops};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::path::PathBuf;
|
||||||
|
|
||||||
|
use ethstore::ethkey::Address;
|
||||||
|
use ethjson::misc::{AccountMeta, DappsSettings as JsonSettings};
|
||||||
|
use account_provider::DappId;
|
||||||
|
|
||||||
|
/// Disk-backed map from Address to String. Uses JSON.
|
||||||
|
pub struct AddressBook {
|
||||||
|
cache: DiskMap<Address, AccountMeta>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AddressBook {
|
||||||
|
/// Creates new address book at given directory.
|
||||||
|
pub fn new(path: String) -> Self {
|
||||||
|
let mut r = AddressBook {
|
||||||
|
cache: DiskMap::new(path, "address_book.json".into())
|
||||||
|
};
|
||||||
|
r.cache.revert(AccountMeta::read_address_map);
|
||||||
|
r
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates transient address book (no changes are saved to disk).
|
||||||
|
pub fn transient() -> Self {
|
||||||
|
AddressBook {
|
||||||
|
cache: DiskMap::transient()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the address book.
|
||||||
|
pub fn get(&self) -> HashMap<Address, AccountMeta> {
|
||||||
|
self.cache.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn save(&self) {
|
||||||
|
self.cache.save(AccountMeta::write_address_map)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets new name for given address.
|
||||||
|
pub fn set_name(&mut self, a: Address, name: String) {
|
||||||
|
{
|
||||||
|
let mut x = self.cache.entry(a)
|
||||||
|
.or_insert_with(|| AccountMeta {name: Default::default(), meta: "{}".to_owned(), uuid: None});
|
||||||
|
x.name = name;
|
||||||
|
}
|
||||||
|
self.save();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets new meta for given address.
|
||||||
|
pub fn set_meta(&mut self, a: Address, meta: String) {
|
||||||
|
{
|
||||||
|
let mut x = self.cache.entry(a)
|
||||||
|
.or_insert_with(|| AccountMeta {name: "Anonymous".to_owned(), meta: Default::default(), uuid: None});
|
||||||
|
x.meta = meta;
|
||||||
|
}
|
||||||
|
self.save();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Dapps user settings
|
||||||
|
#[derive(Debug, Default, Clone, Eq, PartialEq)]
|
||||||
|
pub struct DappsSettings {
|
||||||
|
/// A list of visible accounts
|
||||||
|
pub accounts: Vec<Address>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<JsonSettings> for DappsSettings {
|
||||||
|
fn from(s: JsonSettings) -> Self {
|
||||||
|
DappsSettings {
|
||||||
|
accounts: s.accounts.into_iter().map(Into::into).collect(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<DappsSettings> for JsonSettings {
|
||||||
|
fn from(s: DappsSettings) -> Self {
|
||||||
|
JsonSettings {
|
||||||
|
accounts: s.accounts.into_iter().map(Into::into).collect(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Disk-backed map from DappId to Settings. Uses JSON.
|
||||||
|
pub struct DappsSettingsStore {
|
||||||
|
cache: DiskMap<DappId, DappsSettings>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DappsSettingsStore {
|
||||||
|
/// Creates new store at given directory path.
|
||||||
|
pub fn new(path: String) -> Self {
|
||||||
|
let mut r = DappsSettingsStore {
|
||||||
|
cache: DiskMap::new(path, "dapps_accounts.json".into())
|
||||||
|
};
|
||||||
|
r.cache.revert(JsonSettings::read_dapps_settings);
|
||||||
|
r
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates transient store (no changes are saved to disk).
|
||||||
|
pub fn transient() -> Self {
|
||||||
|
DappsSettingsStore {
|
||||||
|
cache: DiskMap::transient()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get copy of the dapps settings
|
||||||
|
pub fn get(&self) -> HashMap<DappId, DappsSettings> {
|
||||||
|
self.cache.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn save(&self) {
|
||||||
|
self.cache.save(JsonSettings::write_dapps_settings)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_accounts(&mut self, id: DappId, accounts: Vec<Address>) {
|
||||||
|
{
|
||||||
|
let mut settings = self.cache.entry(id).or_insert_with(DappsSettings::default);
|
||||||
|
settings.accounts = accounts;
|
||||||
|
}
|
||||||
|
self.save();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Disk-serializable HashMap
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct DiskMap<K: hash::Hash + Eq, V> {
|
||||||
|
path: PathBuf,
|
||||||
|
cache: HashMap<K, V>,
|
||||||
|
transient: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K: hash::Hash + Eq, V> ops::Deref for DiskMap<K, V> {
|
||||||
|
type Target = HashMap<K, V>;
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.cache
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K: hash::Hash + Eq, V> ops::DerefMut for DiskMap<K, V> {
|
||||||
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||||
|
&mut self.cache
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K: hash::Hash + Eq, V> DiskMap<K, V> {
|
||||||
|
pub fn new(path: String, file_name: String) -> Self {
|
||||||
|
trace!(target: "diskmap", "new({})", path);
|
||||||
|
let mut path: PathBuf = path.into();
|
||||||
|
path.push(file_name);
|
||||||
|
trace!(target: "diskmap", "path={:?}", path);
|
||||||
|
DiskMap {
|
||||||
|
path: path,
|
||||||
|
cache: HashMap::new(),
|
||||||
|
transient: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn transient() -> Self {
|
||||||
|
let mut map = DiskMap::new(Default::default(), "diskmap.json".into());
|
||||||
|
map.transient = true;
|
||||||
|
map
|
||||||
|
}
|
||||||
|
|
||||||
|
fn revert<F, E>(&mut self, read: F) where
|
||||||
|
F: Fn(fs::File) -> Result<HashMap<K, V>, E>,
|
||||||
|
E: fmt::Display,
|
||||||
|
{
|
||||||
|
if self.transient { return; }
|
||||||
|
trace!(target: "diskmap", "revert {:?}", self.path);
|
||||||
|
let _ = fs::File::open(self.path.clone())
|
||||||
|
.map_err(|e| trace!(target: "diskmap", "Couldn't open disk map: {}", e))
|
||||||
|
.and_then(|f| read(f).map_err(|e| warn!(target: "diskmap", "Couldn't read disk map: {}", e)))
|
||||||
|
.and_then(|m| {
|
||||||
|
self.cache = m;
|
||||||
|
Ok(())
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn save<F, E>(&self, write: F) where
|
||||||
|
F: Fn(&HashMap<K, V>, &mut fs::File) -> Result<(), E>,
|
||||||
|
E: fmt::Display,
|
||||||
|
{
|
||||||
|
if self.transient { return; }
|
||||||
|
trace!(target: "diskmap", "save {:?}", self.path);
|
||||||
|
let _ = fs::File::create(self.path.clone())
|
||||||
|
.map_err(|e| warn!(target: "diskmap", "Couldn't open disk map for writing: {}", e))
|
||||||
|
.and_then(|mut f| {
|
||||||
|
write(&self.cache, &mut f).map_err(|e| warn!(target: "diskmap", "Couldn't write to disk map: {}", e))
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::{AddressBook, DappsSettingsStore, DappsSettings};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use ethjson::misc::AccountMeta;
|
||||||
|
use devtools::RandomTempPath;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn should_save_and_reload_address_book() {
|
||||||
|
let temp = RandomTempPath::create_dir();
|
||||||
|
let path = temp.as_str().to_owned();
|
||||||
|
let mut b = AddressBook::new(path.clone());
|
||||||
|
b.set_name(1.into(), "One".to_owned());
|
||||||
|
b.set_meta(1.into(), "{1:1}".to_owned());
|
||||||
|
let b = AddressBook::new(path);
|
||||||
|
assert_eq!(b.get(), hash_map![1.into() => AccountMeta{name: "One".to_owned(), meta: "{1:1}".to_owned(), uuid: None}]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn should_save_and_reload_dapps_settings() {
|
||||||
|
// given
|
||||||
|
let temp = RandomTempPath::create_dir();
|
||||||
|
let path = temp.as_str().to_owned();
|
||||||
|
let mut b = DappsSettingsStore::new(path.clone());
|
||||||
|
|
||||||
|
// when
|
||||||
|
b.set_accounts("dappOne".into(), vec![1.into(), 2.into()]);
|
||||||
|
|
||||||
|
// then
|
||||||
|
let b = DappsSettingsStore::new(path);
|
||||||
|
assert_eq!(b.get(), hash_map![
|
||||||
|
"dappOne".into() => DappsSettings {
|
||||||
|
accounts: vec![1.into(), 2.into()],
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
@ -34,6 +34,7 @@ use blockchain::update::ExtrasUpdate;
|
|||||||
use blockchain::{CacheSize, ImportRoute, Config};
|
use blockchain::{CacheSize, ImportRoute, Config};
|
||||||
use db::{self, Writable, Readable, CacheUpdatePolicy};
|
use db::{self, Writable, Readable, CacheUpdatePolicy};
|
||||||
use cache_manager::CacheManager;
|
use cache_manager::CacheManager;
|
||||||
|
use engines::Engine;
|
||||||
|
|
||||||
const LOG_BLOOMS_LEVELS: usize = 3;
|
const LOG_BLOOMS_LEVELS: usize = 3;
|
||||||
const LOG_BLOOMS_ELEMENTS_PER_INDEX: usize = 16;
|
const LOG_BLOOMS_ELEMENTS_PER_INDEX: usize = 16;
|
||||||
@ -198,6 +199,9 @@ pub struct BlockChain {
|
|||||||
pending_block_hashes: RwLock<HashMap<BlockNumber, H256>>,
|
pending_block_hashes: RwLock<HashMap<BlockNumber, H256>>,
|
||||||
pending_block_details: RwLock<HashMap<H256, BlockDetails>>,
|
pending_block_details: RwLock<HashMap<H256, BlockDetails>>,
|
||||||
pending_transaction_addresses: RwLock<HashMap<H256, Option<TransactionAddress>>>,
|
pending_transaction_addresses: RwLock<HashMap<H256, Option<TransactionAddress>>>,
|
||||||
|
|
||||||
|
// Used for block ordering.
|
||||||
|
engine: Arc<Engine>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BlockProvider for BlockChain {
|
impl BlockProvider for BlockChain {
|
||||||
@ -415,9 +419,8 @@ impl<'a> Iterator for AncestryIter<'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl BlockChain {
|
impl BlockChain {
|
||||||
#[cfg_attr(feature="dev", allow(useless_let_if_seq))]
|
/// Create new instance of blockchain from given Genesis and block picking rules of Engine.
|
||||||
/// Create new instance of blockchain from given Genesis
|
pub fn new(config: Config, genesis: &[u8], db: Arc<Database>, engine: Arc<Engine>) -> BlockChain {
|
||||||
pub fn new(config: Config, genesis: &[u8], db: Arc<Database>) -> BlockChain {
|
|
||||||
// 400 is the avarage size of the key
|
// 400 is the avarage size of the key
|
||||||
let cache_man = CacheManager::new(config.pref_cache_size, config.max_cache_size, 400);
|
let cache_man = CacheManager::new(config.pref_cache_size, config.max_cache_size, 400);
|
||||||
|
|
||||||
@ -442,6 +445,7 @@ impl BlockChain {
|
|||||||
pending_block_hashes: RwLock::new(HashMap::new()),
|
pending_block_hashes: RwLock::new(HashMap::new()),
|
||||||
pending_block_details: RwLock::new(HashMap::new()),
|
pending_block_details: RwLock::new(HashMap::new()),
|
||||||
pending_transaction_addresses: RwLock::new(HashMap::new()),
|
pending_transaction_addresses: RwLock::new(HashMap::new()),
|
||||||
|
engine: engine,
|
||||||
};
|
};
|
||||||
|
|
||||||
// load best block
|
// load best block
|
||||||
@ -858,13 +862,12 @@ impl BlockChain {
|
|||||||
let number = header.number();
|
let number = header.number();
|
||||||
let parent_hash = header.parent_hash();
|
let parent_hash = header.parent_hash();
|
||||||
let parent_details = self.block_details(&parent_hash).unwrap_or_else(|| panic!("Invalid parent hash: {:?}", parent_hash));
|
let parent_details = self.block_details(&parent_hash).unwrap_or_else(|| panic!("Invalid parent hash: {:?}", parent_hash));
|
||||||
let total_difficulty = parent_details.total_difficulty + header.difficulty();
|
let is_new_best = self.engine.is_new_best_block(self.best_block_total_difficulty(), HeaderView::new(&self.best_block_header()), &parent_details, header);
|
||||||
let is_new_best = total_difficulty > self.best_block_total_difficulty();
|
|
||||||
|
|
||||||
BlockInfo {
|
BlockInfo {
|
||||||
hash: hash,
|
hash: hash,
|
||||||
number: number,
|
number: number,
|
||||||
total_difficulty: total_difficulty,
|
total_difficulty: parent_details.total_difficulty + header.difficulty(),
|
||||||
location: if is_new_best {
|
location: if is_new_best {
|
||||||
// on new best block we need to make sure that all ancestors
|
// on new best block we need to make sure that all ancestors
|
||||||
// are moved to "canon chain"
|
// are moved to "canon chain"
|
||||||
@ -1319,11 +1322,16 @@ mod tests {
|
|||||||
use views::BlockView;
|
use views::BlockView;
|
||||||
use transaction::{Transaction, Action};
|
use transaction::{Transaction, Action};
|
||||||
use log_entry::{LogEntry, LocalizedLogEntry};
|
use log_entry::{LogEntry, LocalizedLogEntry};
|
||||||
|
use spec::Spec;
|
||||||
|
|
||||||
fn new_db(path: &str) -> Arc<Database> {
|
fn new_db(path: &str) -> Arc<Database> {
|
||||||
Arc::new(Database::open(&DatabaseConfig::with_columns(::db::NUM_COLUMNS), path).unwrap())
|
Arc::new(Database::open(&DatabaseConfig::with_columns(::db::NUM_COLUMNS), path).unwrap())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn new_chain(genesis: &[u8], db: Arc<Database>) -> BlockChain {
|
||||||
|
BlockChain::new(Config::default(), genesis, db, Spec::new_null().engine)
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn should_cache_best_block() {
|
fn should_cache_best_block() {
|
||||||
// given
|
// given
|
||||||
@ -1334,7 +1342,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
assert_eq!(bc.best_block_number(), 0);
|
assert_eq!(bc.best_block_number(), 0);
|
||||||
|
|
||||||
// when
|
// when
|
||||||
@ -1360,7 +1368,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
assert_eq!(bc.genesis_hash(), genesis_hash.clone());
|
assert_eq!(bc.genesis_hash(), genesis_hash.clone());
|
||||||
assert_eq!(bc.best_block_hash(), genesis_hash.clone());
|
assert_eq!(bc.best_block_hash(), genesis_hash.clone());
|
||||||
@ -1391,7 +1399,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
let mut block_hashes = vec![genesis_hash.clone()];
|
let mut block_hashes = vec![genesis_hash.clone()];
|
||||||
let mut batch = db.transaction();
|
let mut batch = db.transaction();
|
||||||
@ -1427,7 +1435,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
let mut batch =db.transaction();
|
let mut batch =db.transaction();
|
||||||
for b in &[&b1a, &b1b, &b2a, &b2b, &b3a, &b3b, &b4a, &b4b, &b5a, &b5b] {
|
for b in &[&b1a, &b1b, &b2a, &b2b, &b3a, &b3b, &b4a, &b4b, &b5a, &b5b] {
|
||||||
@ -1489,7 +1497,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
let mut batch = db.transaction();
|
let mut batch = db.transaction();
|
||||||
let _ = bc.insert_block(&mut batch, &b1a, vec![]);
|
let _ = bc.insert_block(&mut batch, &b1a, vec![]);
|
||||||
@ -1577,7 +1585,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
let mut batch = db.transaction();
|
let mut batch = db.transaction();
|
||||||
let _ = bc.insert_block(&mut batch, &b1a, vec![]);
|
let _ = bc.insert_block(&mut batch, &b1a, vec![]);
|
||||||
@ -1639,7 +1647,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
let mut batch = db.transaction();
|
let mut batch = db.transaction();
|
||||||
let ir1 = bc.insert_block(&mut batch, &b1, vec![]);
|
let ir1 = bc.insert_block(&mut batch, &b1, vec![]);
|
||||||
@ -1755,7 +1763,7 @@ mod tests {
|
|||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
{
|
{
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
assert_eq!(bc.best_block_hash(), genesis_hash);
|
assert_eq!(bc.best_block_hash(), genesis_hash);
|
||||||
let mut batch =db.transaction();
|
let mut batch =db.transaction();
|
||||||
bc.insert_block(&mut batch, &first, vec![]);
|
bc.insert_block(&mut batch, &first, vec![]);
|
||||||
@ -1766,7 +1774,7 @@ mod tests {
|
|||||||
|
|
||||||
{
|
{
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
assert_eq!(bc.best_block_hash(), first_hash);
|
assert_eq!(bc.best_block_hash(), first_hash);
|
||||||
}
|
}
|
||||||
@ -1821,7 +1829,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
let mut batch =db.transaction();
|
let mut batch =db.transaction();
|
||||||
bc.insert_block(&mut batch, &b1, vec![]);
|
bc.insert_block(&mut batch, &b1, vec![]);
|
||||||
db.write(batch).unwrap();
|
db.write(batch).unwrap();
|
||||||
@ -1881,7 +1889,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
insert_block(&db, &bc, &b1, vec![Receipt {
|
insert_block(&db, &bc, &b1, vec![Receipt {
|
||||||
state_root: H256::default(),
|
state_root: H256::default(),
|
||||||
gas_used: 10_000.into(),
|
gas_used: 10_000.into(),
|
||||||
@ -1985,7 +1993,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
let blocks_b1 = bc.blocks_with_bloom(&bloom_b1, 0, 5);
|
let blocks_b1 = bc.blocks_with_bloom(&bloom_b1, 0, 5);
|
||||||
let blocks_b2 = bc.blocks_with_bloom(&bloom_b2, 0, 5);
|
let blocks_b2 = bc.blocks_with_bloom(&bloom_b2, 0, 5);
|
||||||
@ -2042,7 +2050,7 @@ mod tests {
|
|||||||
|
|
||||||
{
|
{
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
let uncle = canon_chain.fork(1).generate(&mut finalizer.fork()).unwrap();
|
let uncle = canon_chain.fork(1).generate(&mut finalizer.fork()).unwrap();
|
||||||
|
|
||||||
let mut batch =db.transaction();
|
let mut batch =db.transaction();
|
||||||
@ -2061,7 +2069,7 @@ mod tests {
|
|||||||
|
|
||||||
// re-loading the blockchain should load the correct best block.
|
// re-loading the blockchain should load the correct best block.
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
assert_eq!(bc.best_block_number(), 5);
|
assert_eq!(bc.best_block_number(), 5);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2078,7 +2086,7 @@ mod tests {
|
|||||||
|
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(Config::default(), &genesis, db.clone());
|
let bc = new_chain(&genesis, db.clone());
|
||||||
|
|
||||||
let mut batch =db.transaction();
|
let mut batch =db.transaction();
|
||||||
bc.insert_block(&mut batch, &first, vec![]);
|
bc.insert_block(&mut batch, &first, vec![]);
|
||||||
|
@ -164,7 +164,7 @@ impl Client {
|
|||||||
let gb = spec.genesis_block();
|
let gb = spec.genesis_block();
|
||||||
|
|
||||||
let db = Arc::new(try!(Database::open(&db_config, &path.to_str().expect("DB path could not be converted to string.")).map_err(ClientError::Database)));
|
let db = Arc::new(try!(Database::open(&db_config, &path.to_str().expect("DB path could not be converted to string.")).map_err(ClientError::Database)));
|
||||||
let chain = Arc::new(BlockChain::new(config.blockchain.clone(), &gb, db.clone()));
|
let chain = Arc::new(BlockChain::new(config.blockchain.clone(), &gb, db.clone(), spec.engine.clone()));
|
||||||
let tracedb = RwLock::new(TraceDB::new(config.tracing.clone(), db.clone(), chain.clone()));
|
let tracedb = RwLock::new(TraceDB::new(config.tracing.clone(), db.clone(), chain.clone()));
|
||||||
|
|
||||||
let trie_spec = match config.fat_db {
|
let trie_spec = match config.fat_db {
|
||||||
@ -787,7 +787,7 @@ impl snapshot::DatabaseRestore for Client {
|
|||||||
|
|
||||||
let cache_size = state_db.cache_size();
|
let cache_size = state_db.cache_size();
|
||||||
*state_db = StateDB::new(journaldb::new(db.clone(), self.pruning, ::db::COL_STATE), cache_size);
|
*state_db = StateDB::new(journaldb::new(db.clone(), self.pruning, ::db::COL_STATE), cache_size);
|
||||||
*chain = Arc::new(BlockChain::new(self.config.blockchain.clone(), &[], db.clone()));
|
*chain = Arc::new(BlockChain::new(self.config.blockchain.clone(), &[], db.clone(), self.engine.clone()));
|
||||||
*tracedb = TraceDB::new(self.config.tracing.clone(), db.clone(), chain.clone());
|
*tracedb = TraceDB::new(self.config.tracing.clone(), db.clone(), chain.clone());
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -1272,7 +1272,7 @@ impl BlockChainClient for Client {
|
|||||||
self.miner.pending_transactions(self.chain.read().best_block_number())
|
self.miner.pending_transactions(self.chain.read().best_block_number())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn signing_network_id(&self) -> Option<u8> {
|
fn signing_network_id(&self) -> Option<u64> {
|
||||||
self.engine.signing_network_id(&self.latest_env_info())
|
self.engine.signing_network_id(&self.latest_env_info())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -662,7 +662,7 @@ impl BlockChainClient for TestBlockChainClient {
|
|||||||
self.miner.pending_transactions(self.chain_info().best_block_number)
|
self.miner.pending_transactions(self.chain_info().best_block_number)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn signing_network_id(&self) -> Option<u8> { None }
|
fn signing_network_id(&self) -> Option<u64> { None }
|
||||||
|
|
||||||
fn mode(&self) -> Mode { Mode::Active }
|
fn mode(&self) -> Mode { Mode::Active }
|
||||||
|
|
||||||
|
@ -237,7 +237,7 @@ pub trait BlockChainClient : Sync + Send {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Get the preferred network ID to sign on
|
/// Get the preferred network ID to sign on
|
||||||
fn signing_network_id(&self) -> Option<u8>;
|
fn signing_network_id(&self) -> Option<u64>;
|
||||||
|
|
||||||
/// Get the mode.
|
/// Get the mode.
|
||||||
fn mode(&self) -> Mode;
|
fn mode(&self) -> Mode;
|
||||||
|
@ -21,7 +21,7 @@ use std::sync::Weak;
|
|||||||
use std::time::{UNIX_EPOCH, Duration};
|
use std::time::{UNIX_EPOCH, Duration};
|
||||||
use util::*;
|
use util::*;
|
||||||
use ethkey::{verify_address, Signature};
|
use ethkey::{verify_address, Signature};
|
||||||
use rlp::{UntrustedRlp, View, encode};
|
use rlp::{Rlp, UntrustedRlp, View, encode};
|
||||||
use account_provider::AccountProvider;
|
use account_provider::AccountProvider;
|
||||||
use block::*;
|
use block::*;
|
||||||
use spec::CommonParams;
|
use spec::CommonParams;
|
||||||
@ -35,6 +35,8 @@ use service::ClientIoMessage;
|
|||||||
use transaction::SignedTransaction;
|
use transaction::SignedTransaction;
|
||||||
use env_info::EnvInfo;
|
use env_info::EnvInfo;
|
||||||
use builtin::Builtin;
|
use builtin::Builtin;
|
||||||
|
use blockchain::extras::BlockDetails;
|
||||||
|
use views::HeaderView;
|
||||||
|
|
||||||
/// `AuthorityRound` params.
|
/// `AuthorityRound` params.
|
||||||
#[derive(Debug, PartialEq)]
|
#[derive(Debug, PartialEq)]
|
||||||
@ -272,7 +274,6 @@ impl Engine for AuthorityRound {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn verify_block_family(&self, header: &Header, parent: &Header, _block: Option<&[u8]>) -> Result<(), Error> {
|
fn verify_block_family(&self, header: &Header, parent: &Header, _block: Option<&[u8]>) -> Result<(), Error> {
|
||||||
// Don't calculate difficulty for genesis blocks.
|
|
||||||
if header.number() == 0 {
|
if header.number() == 0 {
|
||||||
return Err(From::from(BlockError::RidiculousNumber(OutOfBounds { min: Some(1), max: None, found: header.number() })));
|
return Err(From::from(BlockError::RidiculousNumber(OutOfBounds { min: Some(1), max: None, found: header.number() })));
|
||||||
}
|
}
|
||||||
@ -284,10 +285,6 @@ impl Engine for AuthorityRound {
|
|||||||
try!(Err(BlockError::DoubleVote(header.author().clone())));
|
try!(Err(BlockError::DoubleVote(header.author().clone())));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check difficulty is correct given the two timestamps.
|
|
||||||
if header.difficulty() != parent.difficulty() {
|
|
||||||
return Err(From::from(BlockError::InvalidDifficulty(Mismatch { expected: *parent.difficulty(), found: *header.difficulty() })))
|
|
||||||
}
|
|
||||||
let gas_limit_divisor = self.our_params.gas_limit_bound_divisor;
|
let gas_limit_divisor = self.our_params.gas_limit_bound_divisor;
|
||||||
let min_gas = parent.gas_limit().clone() - parent.gas_limit().clone() / gas_limit_divisor;
|
let min_gas = parent.gas_limit().clone() - parent.gas_limit().clone() / gas_limit_divisor;
|
||||||
let max_gas = parent.gas_limit().clone() + parent.gas_limit().clone() / gas_limit_divisor;
|
let max_gas = parent.gas_limit().clone() + parent.gas_limit().clone() / gas_limit_divisor;
|
||||||
@ -310,6 +307,19 @@ impl Engine for AuthorityRound {
|
|||||||
let mut guard = self.message_channel.lock();
|
let mut guard = self.message_channel.lock();
|
||||||
*guard = Some(message_channel);
|
*guard = Some(message_channel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn is_new_best_block(&self, _best_total_difficulty: U256, best_header: HeaderView, _parent_details: &BlockDetails, new_header: &HeaderView) -> bool {
|
||||||
|
let new_number = new_header.number();
|
||||||
|
let best_number = best_header.number();
|
||||||
|
if new_number != best_number {
|
||||||
|
new_number > best_number
|
||||||
|
} else {
|
||||||
|
// Take the oldest step at given height.
|
||||||
|
let new_step: usize = Rlp::new(&new_header.seal()[0]).as_val();
|
||||||
|
let best_step: usize = Rlp::new(&best_header.seal()[0]).as_val();
|
||||||
|
new_step < best_step
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
@ -38,6 +38,9 @@ use io::IoChannel;
|
|||||||
use service::ClientIoMessage;
|
use service::ClientIoMessage;
|
||||||
use header::Header;
|
use header::Header;
|
||||||
use transaction::SignedTransaction;
|
use transaction::SignedTransaction;
|
||||||
|
use ethereum::ethash;
|
||||||
|
use blockchain::extras::BlockDetails;
|
||||||
|
use views::HeaderView;
|
||||||
|
|
||||||
/// A consensus mechanism for the chain. Generally either proof-of-work or proof-of-stake-based.
|
/// A consensus mechanism for the chain. Generally either proof-of-work or proof-of-stake-based.
|
||||||
/// Provides hooks into each of the major parts of block import.
|
/// Provides hooks into each of the major parts of block import.
|
||||||
@ -113,7 +116,7 @@ pub trait Engine : Sync + Send {
|
|||||||
fn verify_transaction(&self, _t: &SignedTransaction, _header: &Header) -> Result<(), Error> { Ok(()) }
|
fn verify_transaction(&self, _t: &SignedTransaction, _header: &Header) -> Result<(), Error> { Ok(()) }
|
||||||
|
|
||||||
/// The network ID that transactions should be signed with.
|
/// The network ID that transactions should be signed with.
|
||||||
fn signing_network_id(&self, _env_info: &EnvInfo) -> Option<u8> { None }
|
fn signing_network_id(&self, _env_info: &EnvInfo) -> Option<u64> { None }
|
||||||
|
|
||||||
/// Verify the seal of a block. This is an auxilliary method that actually just calls other `verify_` methods
|
/// Verify the seal of a block. This is an auxilliary method that actually just calls other `verify_` methods
|
||||||
/// to get the job done. By default it must pass `verify_basic` and `verify_block_unordered`. If more or fewer
|
/// to get the job done. By default it must pass `verify_basic` and `verify_block_unordered`. If more or fewer
|
||||||
@ -146,5 +149,9 @@ pub trait Engine : Sync + Send {
|
|||||||
|
|
||||||
/// Add a channel for communication with Client which can be used for sealing.
|
/// Add a channel for communication with Client which can be used for sealing.
|
||||||
fn register_message_channel(&self, _message_channel: IoChannel<ClientIoMessage>) {}
|
fn register_message_channel(&self, _message_channel: IoChannel<ClientIoMessage>) {}
|
||||||
// TODO: sealing stuff - though might want to leave this for later.
|
|
||||||
|
/// Check if new block should be chosen as the one in chain.
|
||||||
|
fn is_new_best_block(&self, best_total_difficulty: U256, _best_header: HeaderView, parent_details: &BlockDetails, new_header: &HeaderView) -> bool {
|
||||||
|
ethash::is_new_best_block(best_total_difficulty, parent_details, new_header)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -38,6 +38,12 @@ impl NullEngine {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Default for NullEngine {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new(Default::default(), Default::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Engine for NullEngine {
|
impl Engine for NullEngine {
|
||||||
fn name(&self) -> &str {
|
fn name(&self) -> &str {
|
||||||
"NullEngine"
|
"NullEngine"
|
||||||
|
@ -21,6 +21,7 @@ use builtin::Builtin;
|
|||||||
use env_info::EnvInfo;
|
use env_info::EnvInfo;
|
||||||
use error::{BlockError, TransactionError, Error};
|
use error::{BlockError, TransactionError, Error};
|
||||||
use header::Header;
|
use header::Header;
|
||||||
|
use views::HeaderView;
|
||||||
use state::CleanupMode;
|
use state::CleanupMode;
|
||||||
use spec::CommonParams;
|
use spec::CommonParams;
|
||||||
use transaction::SignedTransaction;
|
use transaction::SignedTransaction;
|
||||||
@ -28,6 +29,7 @@ use engines::Engine;
|
|||||||
use evm::Schedule;
|
use evm::Schedule;
|
||||||
use ethjson;
|
use ethjson;
|
||||||
use rlp::{self, UntrustedRlp, View};
|
use rlp::{self, UntrustedRlp, View};
|
||||||
|
use blockchain::extras::BlockDetails;
|
||||||
|
|
||||||
/// Ethash params.
|
/// Ethash params.
|
||||||
#[derive(Debug, PartialEq)]
|
#[derive(Debug, PartialEq)]
|
||||||
@ -163,9 +165,9 @@ impl Engine for Ethash {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn signing_network_id(&self, env_info: &EnvInfo) -> Option<u8> {
|
fn signing_network_id(&self, env_info: &EnvInfo) -> Option<u64> {
|
||||||
if env_info.number >= self.ethash_params.eip155_transition && self.params().network_id < 127 {
|
if env_info.number >= self.ethash_params.eip155_transition {
|
||||||
Some(self.params().network_id as u8)
|
Some(self.params().network_id)
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
@ -314,7 +316,7 @@ impl Engine for Ethash {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let Some(n) = t.network_id() {
|
if let Some(n) = t.network_id() {
|
||||||
if header.number() < self.ethash_params.eip155_transition || n as usize != self.params().network_id {
|
if header.number() < self.ethash_params.eip155_transition || n != self.params().network_id {
|
||||||
return Err(TransactionError::InvalidNetworkId.into())
|
return Err(TransactionError::InvalidNetworkId.into())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -325,6 +327,15 @@ impl Engine for Ethash {
|
|||||||
fn verify_transaction(&self, t: &SignedTransaction, _header: &Header) -> Result<(), Error> {
|
fn verify_transaction(&self, t: &SignedTransaction, _header: &Header) -> Result<(), Error> {
|
||||||
t.sender().map(|_|()) // Perform EC recovery and cache sender
|
t.sender().map(|_|()) // Perform EC recovery and cache sender
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn is_new_best_block(&self, best_total_difficulty: U256, _best_header: HeaderView, parent_details: &BlockDetails, new_header: &HeaderView) -> bool {
|
||||||
|
is_new_best_block(best_total_difficulty, parent_details, new_header)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check if a new block should replace the best blockchain block.
|
||||||
|
pub fn is_new_best_block(best_total_difficulty: U256, parent_details: &BlockDetails, new_header: &HeaderView) -> bool {
|
||||||
|
parent_details.total_difficulty + new_header.difficulty() > best_total_difficulty
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg_attr(feature="dev", allow(wrong_self_convention))]
|
#[cfg_attr(feature="dev", allow(wrong_self_convention))]
|
||||||
|
@ -81,6 +81,7 @@ struct Restoration {
|
|||||||
struct RestorationParams<'a> {
|
struct RestorationParams<'a> {
|
||||||
manifest: ManifestData, // manifest to base restoration on.
|
manifest: ManifestData, // manifest to base restoration on.
|
||||||
pruning: Algorithm, // pruning algorithm for the database.
|
pruning: Algorithm, // pruning algorithm for the database.
|
||||||
|
engine: Arc<Engine>, // consensus engine of the chain.
|
||||||
db_path: PathBuf, // database path
|
db_path: PathBuf, // database path
|
||||||
db_config: &'a DatabaseConfig, // configuration for the database.
|
db_config: &'a DatabaseConfig, // configuration for the database.
|
||||||
writer: Option<LooseWriter>, // writer for recovered snapshot.
|
writer: Option<LooseWriter>, // writer for recovered snapshot.
|
||||||
@ -99,7 +100,7 @@ impl Restoration {
|
|||||||
let raw_db = Arc::new(try!(Database::open(params.db_config, &*params.db_path.to_string_lossy())
|
let raw_db = Arc::new(try!(Database::open(params.db_config, &*params.db_path.to_string_lossy())
|
||||||
.map_err(UtilError::SimpleString)));
|
.map_err(UtilError::SimpleString)));
|
||||||
|
|
||||||
let chain = BlockChain::new(Default::default(), params.genesis, raw_db.clone());
|
let chain = BlockChain::new(Default::default(), params.genesis, raw_db.clone(), params.engine);
|
||||||
let blocks = try!(BlockRebuilder::new(chain, raw_db.clone(), &manifest));
|
let blocks = try!(BlockRebuilder::new(chain, raw_db.clone(), &manifest));
|
||||||
|
|
||||||
let root = manifest.state_root.clone();
|
let root = manifest.state_root.clone();
|
||||||
@ -420,6 +421,7 @@ impl Service {
|
|||||||
let params = RestorationParams {
|
let params = RestorationParams {
|
||||||
manifest: manifest,
|
manifest: manifest,
|
||||||
pruning: self.pruning,
|
pruning: self.pruning,
|
||||||
|
engine: self.engine.clone(),
|
||||||
db_path: self.restoration_db(),
|
db_path: self.restoration_db(),
|
||||||
db_config: &self.db_config,
|
db_config: &self.db_config,
|
||||||
writer: writer,
|
writer: writer,
|
||||||
|
@ -37,13 +37,14 @@ fn chunk_and_restore(amount: u64) {
|
|||||||
let genesis = canon_chain.generate(&mut finalizer).unwrap();
|
let genesis = canon_chain.generate(&mut finalizer).unwrap();
|
||||||
let db_cfg = DatabaseConfig::with_columns(::db::NUM_COLUMNS);
|
let db_cfg = DatabaseConfig::with_columns(::db::NUM_COLUMNS);
|
||||||
|
|
||||||
|
let engine = Arc::new(::engines::NullEngine::default());
|
||||||
let orig_path = RandomTempPath::create_dir();
|
let orig_path = RandomTempPath::create_dir();
|
||||||
let new_path = RandomTempPath::create_dir();
|
let new_path = RandomTempPath::create_dir();
|
||||||
let mut snapshot_path = new_path.as_path().to_owned();
|
let mut snapshot_path = new_path.as_path().to_owned();
|
||||||
snapshot_path.push("SNAP");
|
snapshot_path.push("SNAP");
|
||||||
|
|
||||||
let old_db = Arc::new(Database::open(&db_cfg, orig_path.as_str()).unwrap());
|
let old_db = Arc::new(Database::open(&db_cfg, orig_path.as_str()).unwrap());
|
||||||
let bc = BlockChain::new(Default::default(), &genesis, old_db.clone());
|
let bc = BlockChain::new(Default::default(), &genesis, old_db.clone(), engine.clone());
|
||||||
|
|
||||||
// build the blockchain.
|
// build the blockchain.
|
||||||
let mut batch = old_db.transaction();
|
let mut batch = old_db.transaction();
|
||||||
@ -73,21 +74,20 @@ fn chunk_and_restore(amount: u64) {
|
|||||||
|
|
||||||
// restore it.
|
// restore it.
|
||||||
let new_db = Arc::new(Database::open(&db_cfg, new_path.as_str()).unwrap());
|
let new_db = Arc::new(Database::open(&db_cfg, new_path.as_str()).unwrap());
|
||||||
let new_chain = BlockChain::new(Default::default(), &genesis, new_db.clone());
|
let new_chain = BlockChain::new(Default::default(), &genesis, new_db.clone(), engine.clone());
|
||||||
let mut rebuilder = BlockRebuilder::new(new_chain, new_db.clone(), &manifest).unwrap();
|
let mut rebuilder = BlockRebuilder::new(new_chain, new_db.clone(), &manifest).unwrap();
|
||||||
let reader = PackedReader::new(&snapshot_path).unwrap().unwrap();
|
let reader = PackedReader::new(&snapshot_path).unwrap().unwrap();
|
||||||
let engine = ::engines::NullEngine::new(Default::default(), Default::default());
|
|
||||||
let flag = AtomicBool::new(true);
|
let flag = AtomicBool::new(true);
|
||||||
for chunk_hash in &reader.manifest().block_hashes {
|
for chunk_hash in &reader.manifest().block_hashes {
|
||||||
let compressed = reader.chunk(*chunk_hash).unwrap();
|
let compressed = reader.chunk(*chunk_hash).unwrap();
|
||||||
let chunk = snappy::decompress(&compressed).unwrap();
|
let chunk = snappy::decompress(&compressed).unwrap();
|
||||||
rebuilder.feed(&chunk, &engine, &flag).unwrap();
|
rebuilder.feed(&chunk, engine.as_ref(), &flag).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
rebuilder.finalize(HashMap::new()).unwrap();
|
rebuilder.finalize(HashMap::new()).unwrap();
|
||||||
|
|
||||||
// and test it.
|
// and test it.
|
||||||
let new_chain = BlockChain::new(Default::default(), &genesis, new_db);
|
let new_chain = BlockChain::new(Default::default(), &genesis, new_db, engine);
|
||||||
assert_eq!(new_chain.best_block_hash(), best_hash);
|
assert_eq!(new_chain.best_block_hash(), best_hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -121,8 +121,8 @@ fn checks_flag() {
|
|||||||
|
|
||||||
let db_cfg = DatabaseConfig::with_columns(::db::NUM_COLUMNS);
|
let db_cfg = DatabaseConfig::with_columns(::db::NUM_COLUMNS);
|
||||||
let db = Arc::new(Database::open(&db_cfg, path.as_str()).unwrap());
|
let db = Arc::new(Database::open(&db_cfg, path.as_str()).unwrap());
|
||||||
let chain = BlockChain::new(Default::default(), &genesis, db.clone());
|
let engine = Arc::new(::engines::NullEngine::default());
|
||||||
let engine = ::engines::NullEngine::new(Default::default(), Default::default());
|
let chain = BlockChain::new(Default::default(), &genesis, db.clone(), engine.clone());
|
||||||
|
|
||||||
let manifest = ::snapshot::ManifestData {
|
let manifest = ::snapshot::ManifestData {
|
||||||
state_hashes: Vec::new(),
|
state_hashes: Vec::new(),
|
||||||
@ -134,7 +134,7 @@ fn checks_flag() {
|
|||||||
|
|
||||||
let mut rebuilder = BlockRebuilder::new(chain, db.clone(), &manifest).unwrap();
|
let mut rebuilder = BlockRebuilder::new(chain, db.clone(), &manifest).unwrap();
|
||||||
|
|
||||||
match rebuilder.feed(&chunk, &engine, &AtomicBool::new(false)) {
|
match rebuilder.feed(&chunk, engine.as_ref(), &AtomicBool::new(false)) {
|
||||||
Err(Error::Snapshot(SnapshotError::RestorationAborted)) => {}
|
Err(Error::Snapshot(SnapshotError::RestorationAborted)) => {}
|
||||||
_ => panic!("Wrong result on abort flag set")
|
_ => panic!("Wrong result on abort flag set")
|
||||||
}
|
}
|
||||||
|
@ -30,11 +30,9 @@ pub struct Ethereum {
|
|||||||
|
|
||||||
impl Into<Generic> for Ethereum {
|
impl Into<Generic> for Ethereum {
|
||||||
fn into(self) -> Generic {
|
fn into(self) -> Generic {
|
||||||
let mut s = RlpStream::new();
|
let mut s = RlpStream::new_list(2);
|
||||||
s.append(&self.mix_hash);
|
s.append(&self.mix_hash).append(&self.nonce);
|
||||||
s.append(&self.nonce);
|
|
||||||
Generic {
|
Generic {
|
||||||
fields: 2,
|
|
||||||
rlp: s.out()
|
rlp: s.out()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -42,8 +40,6 @@ impl Into<Generic> for Ethereum {
|
|||||||
|
|
||||||
/// Generic seal.
|
/// Generic seal.
|
||||||
pub struct Generic {
|
pub struct Generic {
|
||||||
/// Number of seal fields.
|
|
||||||
pub fields: usize,
|
|
||||||
/// Seal rlp.
|
/// Seal rlp.
|
||||||
pub rlp: Vec<u8>,
|
pub rlp: Vec<u8>,
|
||||||
}
|
}
|
||||||
@ -64,7 +60,6 @@ impl From<ethjson::spec::Seal> for Seal {
|
|||||||
mix_hash: eth.mix_hash.into()
|
mix_hash: eth.mix_hash.into()
|
||||||
}),
|
}),
|
||||||
ethjson::spec::Seal::Generic(g) => Seal::Generic(Generic {
|
ethjson::spec::Seal::Generic(g) => Seal::Generic(Generic {
|
||||||
fields: g.fields,
|
|
||||||
rlp: g.rlp.into()
|
rlp: g.rlp.into()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -30,15 +30,14 @@ use ethjson;
|
|||||||
use rlp::{Rlp, RlpStream, View, Stream};
|
use rlp::{Rlp, RlpStream, View, Stream};
|
||||||
|
|
||||||
/// Parameters common to all engines.
|
/// Parameters common to all engines.
|
||||||
#[derive(Debug, PartialEq, Clone)]
|
#[derive(Debug, PartialEq, Clone, Default)]
|
||||||
#[cfg_attr(test, derive(Default))]
|
|
||||||
pub struct CommonParams {
|
pub struct CommonParams {
|
||||||
/// Account start nonce.
|
/// Account start nonce.
|
||||||
pub account_start_nonce: U256,
|
pub account_start_nonce: U256,
|
||||||
/// Maximum size of extra data.
|
/// Maximum size of extra data.
|
||||||
pub maximum_extra_data_size: usize,
|
pub maximum_extra_data_size: usize,
|
||||||
/// Network id.
|
/// Network id.
|
||||||
pub network_id: usize,
|
pub network_id: u64,
|
||||||
/// Main subprotocol name.
|
/// Main subprotocol name.
|
||||||
pub subprotocol_name: String,
|
pub subprotocol_name: String,
|
||||||
/// Minimum gas limit.
|
/// Minimum gas limit.
|
||||||
@ -94,8 +93,6 @@ pub struct Spec {
|
|||||||
pub receipts_root: H256,
|
pub receipts_root: H256,
|
||||||
/// The genesis block's extra data field.
|
/// The genesis block's extra data field.
|
||||||
pub extra_data: Bytes,
|
pub extra_data: Bytes,
|
||||||
/// The number of seal fields in the genesis block.
|
|
||||||
pub seal_fields: usize,
|
|
||||||
/// Each seal field, expressed as RLP, concatenated.
|
/// Each seal field, expressed as RLP, concatenated.
|
||||||
pub seal_rlp: Bytes,
|
pub seal_rlp: Bytes,
|
||||||
|
|
||||||
@ -127,7 +124,6 @@ impl From<ethjson::spec::Spec> for Spec {
|
|||||||
gas_used: g.gas_used,
|
gas_used: g.gas_used,
|
||||||
timestamp: g.timestamp,
|
timestamp: g.timestamp,
|
||||||
extra_data: g.extra_data,
|
extra_data: g.extra_data,
|
||||||
seal_fields: seal.fields,
|
|
||||||
seal_rlp: seal.rlp,
|
seal_rlp: seal.rlp,
|
||||||
state_root_memo: RwLock::new(g.state_root),
|
state_root_memo: RwLock::new(g.state_root),
|
||||||
genesis_state: From::from(s.accounts),
|
genesis_state: From::from(s.accounts),
|
||||||
@ -167,7 +163,7 @@ impl Spec {
|
|||||||
pub fn nodes(&self) -> &[String] { &self.nodes }
|
pub fn nodes(&self) -> &[String] { &self.nodes }
|
||||||
|
|
||||||
/// Get the configured Network ID.
|
/// Get the configured Network ID.
|
||||||
pub fn network_id(&self) -> usize { self.params.network_id }
|
pub fn network_id(&self) -> u64 { self.params.network_id }
|
||||||
|
|
||||||
/// Get the configured subprotocol name.
|
/// Get the configured subprotocol name.
|
||||||
pub fn subprotocol_name(&self) -> String { self.params.subprotocol_name.clone() }
|
pub fn subprotocol_name(&self) -> String { self.params.subprotocol_name.clone() }
|
||||||
@ -192,13 +188,8 @@ impl Spec {
|
|||||||
header.set_gas_limit(self.gas_limit.clone());
|
header.set_gas_limit(self.gas_limit.clone());
|
||||||
header.set_difficulty(self.difficulty.clone());
|
header.set_difficulty(self.difficulty.clone());
|
||||||
header.set_seal({
|
header.set_seal({
|
||||||
let seal = {
|
let r = Rlp::new(&self.seal_rlp);
|
||||||
let mut s = RlpStream::new_list(self.seal_fields);
|
r.iter().map(|f| f.as_raw().to_vec()).collect()
|
||||||
s.append_raw(&self.seal_rlp, self.seal_fields);
|
|
||||||
s.out()
|
|
||||||
};
|
|
||||||
let r = Rlp::new(&seal);
|
|
||||||
(0..self.seal_fields).map(|i| r.at(i).as_raw().to_vec()).collect()
|
|
||||||
});
|
});
|
||||||
trace!(target: "spec", "Header hash is {}", header.hash());
|
trace!(target: "spec", "Header hash is {}", header.hash());
|
||||||
header
|
header
|
||||||
@ -227,7 +218,6 @@ impl Spec {
|
|||||||
self.gas_used = g.gas_used;
|
self.gas_used = g.gas_used;
|
||||||
self.timestamp = g.timestamp;
|
self.timestamp = g.timestamp;
|
||||||
self.extra_data = g.extra_data;
|
self.extra_data = g.extra_data;
|
||||||
self.seal_fields = seal.fields;
|
|
||||||
self.seal_rlp = seal.rlp;
|
self.seal_rlp = seal.rlp;
|
||||||
self.state_root_memo = RwLock::new(g.state_root);
|
self.state_root_memo = RwLock::new(g.state_root);
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -286,7 +286,7 @@ fn new_db(path: &str) -> Arc<Database> {
|
|||||||
pub fn generate_dummy_blockchain(block_number: u32) -> GuardedTempResult<BlockChain> {
|
pub fn generate_dummy_blockchain(block_number: u32) -> GuardedTempResult<BlockChain> {
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(BlockChainConfig::default(), &create_unverifiable_block(0, H256::zero()), db.clone());
|
let bc = BlockChain::new(BlockChainConfig::default(), &create_unverifiable_block(0, H256::zero()), db.clone(), Spec::new_null().engine);
|
||||||
|
|
||||||
let mut batch = db.transaction();
|
let mut batch = db.transaction();
|
||||||
for block_order in 1..block_number {
|
for block_order in 1..block_number {
|
||||||
@ -304,7 +304,7 @@ pub fn generate_dummy_blockchain(block_number: u32) -> GuardedTempResult<BlockCh
|
|||||||
pub fn generate_dummy_blockchain_with_extra(block_number: u32) -> GuardedTempResult<BlockChain> {
|
pub fn generate_dummy_blockchain_with_extra(block_number: u32) -> GuardedTempResult<BlockChain> {
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(BlockChainConfig::default(), &create_unverifiable_block(0, H256::zero()), db.clone());
|
let bc = BlockChain::new(BlockChainConfig::default(), &create_unverifiable_block(0, H256::zero()), db.clone(), Spec::new_null().engine);
|
||||||
|
|
||||||
|
|
||||||
let mut batch = db.transaction();
|
let mut batch = db.transaction();
|
||||||
@ -323,7 +323,7 @@ pub fn generate_dummy_blockchain_with_extra(block_number: u32) -> GuardedTempRes
|
|||||||
pub fn generate_dummy_empty_blockchain() -> GuardedTempResult<BlockChain> {
|
pub fn generate_dummy_empty_blockchain() -> GuardedTempResult<BlockChain> {
|
||||||
let temp = RandomTempPath::new();
|
let temp = RandomTempPath::new();
|
||||||
let db = new_db(temp.as_str());
|
let db = new_db(temp.as_str());
|
||||||
let bc = BlockChain::new(BlockChainConfig::default(), &create_unverifiable_block(0, H256::zero()), db.clone());
|
let bc = BlockChain::new(BlockChainConfig::default(), &create_unverifiable_block(0, H256::zero()), db.clone(), Spec::new_null().engine);
|
||||||
|
|
||||||
GuardedTempResult::<BlockChain> {
|
GuardedTempResult::<BlockChain> {
|
||||||
_temp: temp,
|
_temp: temp,
|
||||||
|
@ -72,7 +72,7 @@ pub struct Transaction {
|
|||||||
|
|
||||||
impl Transaction {
|
impl Transaction {
|
||||||
/// Append object with a without signature into RLP stream
|
/// Append object with a without signature into RLP stream
|
||||||
pub fn rlp_append_unsigned_transaction(&self, s: &mut RlpStream, network_id: Option<u8>) {
|
pub fn rlp_append_unsigned_transaction(&self, s: &mut RlpStream, network_id: Option<u64>) {
|
||||||
s.begin_list(if network_id.is_none() { 6 } else { 9 });
|
s.begin_list(if network_id.is_none() { 6 } else { 9 });
|
||||||
s.append(&self.nonce);
|
s.append(&self.nonce);
|
||||||
s.append(&self.gas_price);
|
s.append(&self.gas_price);
|
||||||
@ -140,26 +140,26 @@ impl From<ethjson::transaction::Transaction> for SignedTransaction {
|
|||||||
|
|
||||||
impl Transaction {
|
impl Transaction {
|
||||||
/// The message hash of the transaction.
|
/// The message hash of the transaction.
|
||||||
pub fn hash(&self, network_id: Option<u8>) -> H256 {
|
pub fn hash(&self, network_id: Option<u64>) -> H256 {
|
||||||
let mut stream = RlpStream::new();
|
let mut stream = RlpStream::new();
|
||||||
self.rlp_append_unsigned_transaction(&mut stream, network_id);
|
self.rlp_append_unsigned_transaction(&mut stream, network_id);
|
||||||
stream.out().sha3()
|
stream.out().sha3()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Signs the transaction as coming from `sender`.
|
/// Signs the transaction as coming from `sender`.
|
||||||
pub fn sign(self, secret: &Secret, network_id: Option<u8>) -> SignedTransaction {
|
pub fn sign(self, secret: &Secret, network_id: Option<u64>) -> SignedTransaction {
|
||||||
let sig = ::ethkey::sign(secret, &self.hash(network_id))
|
let sig = ::ethkey::sign(secret, &self.hash(network_id))
|
||||||
.expect("data is valid and context has signing capabilities; qed");
|
.expect("data is valid and context has signing capabilities; qed");
|
||||||
self.with_signature(sig, network_id)
|
self.with_signature(sig, network_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Signs the transaction with signature.
|
/// Signs the transaction with signature.
|
||||||
pub fn with_signature(self, sig: Signature, network_id: Option<u8>) -> SignedTransaction {
|
pub fn with_signature(self, sig: Signature, network_id: Option<u64>) -> SignedTransaction {
|
||||||
SignedTransaction {
|
SignedTransaction {
|
||||||
unsigned: self,
|
unsigned: self,
|
||||||
r: sig.r().into(),
|
r: sig.r().into(),
|
||||||
s: sig.s().into(),
|
s: sig.s().into(),
|
||||||
v: sig.v() + if let Some(n) = network_id { 35 + n * 2 } else { 27 },
|
v: sig.v() as u64 + if let Some(n) = network_id { 35 + n * 2 } else { 27 },
|
||||||
hash: Cell::new(None),
|
hash: Cell::new(None),
|
||||||
sender: Cell::new(None),
|
sender: Cell::new(None),
|
||||||
}
|
}
|
||||||
@ -211,7 +211,7 @@ pub struct SignedTransaction {
|
|||||||
unsigned: Transaction,
|
unsigned: Transaction,
|
||||||
/// The V field of the signature; the LS bit described which half of the curve our point falls
|
/// The V field of the signature; the LS bit described which half of the curve our point falls
|
||||||
/// in. The MS bits describe which network this transaction is for. If 27/28, its for all networks.
|
/// in. The MS bits describe which network this transaction is for. If 27/28, its for all networks.
|
||||||
v: u8,
|
v: u64,
|
||||||
/// The R field of the signature; helps describe the point on the curve.
|
/// The R field of the signature; helps describe the point on the curve.
|
||||||
r: U256,
|
r: U256,
|
||||||
/// The S field of the signature; helps describe the point on the curve.
|
/// The S field of the signature; helps describe the point on the curve.
|
||||||
@ -302,13 +302,13 @@ impl SignedTransaction {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 0 if `v` would have been 27 under "Electrum" notation, 1 if 28 or 4 if invalid.
|
/// 0 if `v` would have been 27 under "Electrum" notation, 1 if 28 or 4 if invalid.
|
||||||
pub fn standard_v(&self) -> u8 { match self.v { v if v == 27 || v == 28 || v > 36 => (v - 1) % 2, _ => 4 } }
|
pub fn standard_v(&self) -> u8 { match self.v { v if v == 27 || v == 28 || v > 36 => ((v - 1) % 2) as u8, _ => 4 } }
|
||||||
|
|
||||||
/// The `v` value that appears in the RLP.
|
/// The `v` value that appears in the RLP.
|
||||||
pub fn original_v(&self) -> u8 { self.v }
|
pub fn original_v(&self) -> u8 { self.v }
|
||||||
|
|
||||||
/// The network ID, or `None` if this is a global transaction.
|
/// The network ID, or `None` if this is a global transaction.
|
||||||
pub fn network_id(&self) -> Option<u8> {
|
pub fn network_id(&self) -> Option<u64> {
|
||||||
match self.v {
|
match self.v {
|
||||||
v if v > 36 => Some((v - 35) / 2),
|
v if v > 36 => Some((v - 35) / 2),
|
||||||
_ => None,
|
_ => None,
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
//! A queue of blocks. Sits between network or other I/O and the `BlockChain`.
|
//! A queue of blocks. Sits between network or other I/O and the `BlockChain`.
|
||||||
//! Sorts them ready for blockchain insertion.
|
//! Sorts them ready for blockchain insertion.
|
||||||
|
|
||||||
use std::thread::{JoinHandle, self};
|
use std::thread::{self, JoinHandle};
|
||||||
use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering as AtomicOrdering};
|
use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering as AtomicOrdering};
|
||||||
use std::sync::{Condvar as SCondvar, Mutex as SMutex};
|
use std::sync::{Condvar as SCondvar, Mutex as SMutex};
|
||||||
use util::*;
|
use util::*;
|
||||||
@ -64,35 +64,11 @@ impl Default for Config {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct VerifierHandle {
|
// pool states
|
||||||
deleting: Arc<AtomicBool>,
|
enum State {
|
||||||
sleep: Arc<AtomicBool>,
|
// all threads with id < inner value are to work.
|
||||||
thread: JoinHandle<()>,
|
Work(usize),
|
||||||
}
|
Exit,
|
||||||
|
|
||||||
impl VerifierHandle {
|
|
||||||
// signal to the verifier thread that it should sleep.
|
|
||||||
fn sleep(&self) {
|
|
||||||
self.sleep.store(true, AtomicOrdering::SeqCst);
|
|
||||||
}
|
|
||||||
|
|
||||||
// signal to the verifier thread that it should wake up.
|
|
||||||
fn wake_up(&self) {
|
|
||||||
self.sleep.store(false, AtomicOrdering::SeqCst);
|
|
||||||
self.thread.thread().unpark();
|
|
||||||
}
|
|
||||||
|
|
||||||
// signal to the verifier thread that it should conclude its
|
|
||||||
// operations.
|
|
||||||
fn conclude(&self) {
|
|
||||||
self.wake_up();
|
|
||||||
self.deleting.store(true, AtomicOrdering::Release);
|
|
||||||
}
|
|
||||||
|
|
||||||
// join the verifier thread.
|
|
||||||
fn join(self) {
|
|
||||||
self.thread.join().expect("Verifier thread panicked");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// An item which is in the process of being verified.
|
/// An item which is in the process of being verified.
|
||||||
@ -131,7 +107,6 @@ pub struct VerificationQueue<K: Kind> {
|
|||||||
engine: Arc<Engine>,
|
engine: Arc<Engine>,
|
||||||
more_to_verify: Arc<SCondvar>,
|
more_to_verify: Arc<SCondvar>,
|
||||||
verification: Arc<Verification<K>>,
|
verification: Arc<Verification<K>>,
|
||||||
verifiers: Mutex<(Vec<VerifierHandle>, usize)>,
|
|
||||||
deleting: Arc<AtomicBool>,
|
deleting: Arc<AtomicBool>,
|
||||||
ready_signal: Arc<QueueSignal>,
|
ready_signal: Arc<QueueSignal>,
|
||||||
empty: Arc<SCondvar>,
|
empty: Arc<SCondvar>,
|
||||||
@ -139,6 +114,8 @@ pub struct VerificationQueue<K: Kind> {
|
|||||||
ticks_since_adjustment: AtomicUsize,
|
ticks_since_adjustment: AtomicUsize,
|
||||||
max_queue_size: usize,
|
max_queue_size: usize,
|
||||||
max_mem_use: usize,
|
max_mem_use: usize,
|
||||||
|
verifier_handles: Vec<JoinHandle<()>>,
|
||||||
|
state: Arc<(Mutex<State>, Condvar)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct QueueSignal {
|
struct QueueSignal {
|
||||||
@ -224,40 +201,39 @@ impl<K: Kind> VerificationQueue<K> {
|
|||||||
|
|
||||||
let max_verifiers = min(::num_cpus::get(), MAX_VERIFIERS);
|
let max_verifiers = min(::num_cpus::get(), MAX_VERIFIERS);
|
||||||
let default_amount = max(::num_cpus::get(), 3) - 2;
|
let default_amount = max(::num_cpus::get(), 3) - 2;
|
||||||
let mut verifiers = Vec::with_capacity(max_verifiers);
|
let state = Arc::new((Mutex::new(State::Work(default_amount)), Condvar::new()));
|
||||||
|
let mut verifier_handles = Vec::with_capacity(max_verifiers);
|
||||||
|
|
||||||
debug!(target: "verification", "Allocating {} verifiers, {} initially active", max_verifiers, default_amount);
|
debug!(target: "verification", "Allocating {} verifiers, {} initially active", max_verifiers, default_amount);
|
||||||
|
|
||||||
for i in 0..max_verifiers {
|
for i in 0..max_verifiers {
|
||||||
debug!(target: "verification", "Adding verification thread #{}", i);
|
debug!(target: "verification", "Adding verification thread #{}", i);
|
||||||
|
|
||||||
let deleting = deleting.clone();
|
|
||||||
let panic_handler = panic_handler.clone();
|
let panic_handler = panic_handler.clone();
|
||||||
let verification = verification.clone();
|
let verification = verification.clone();
|
||||||
let engine = engine.clone();
|
let engine = engine.clone();
|
||||||
let wait = more_to_verify.clone();
|
let wait = more_to_verify.clone();
|
||||||
let ready = ready_signal.clone();
|
let ready = ready_signal.clone();
|
||||||
let empty = empty.clone();
|
let empty = empty.clone();
|
||||||
|
let state = state.clone();
|
||||||
|
|
||||||
// enable only the first few verifiers.
|
let handle = thread::Builder::new()
|
||||||
let sleep = if i < default_amount {
|
.name(format!("Verifier #{}", i))
|
||||||
Arc::new(AtomicBool::new(false))
|
.spawn(move || {
|
||||||
} else {
|
panic_handler.catch_panic(move || {
|
||||||
Arc::new(AtomicBool::new(true))
|
VerificationQueue::verify(
|
||||||
};
|
verification,
|
||||||
|
engine,
|
||||||
verifiers.push(VerifierHandle {
|
wait,
|
||||||
deleting: deleting.clone(),
|
ready,
|
||||||
sleep: sleep.clone(),
|
empty,
|
||||||
thread: thread::Builder::new()
|
state,
|
||||||
.name(format!("Verifier #{}", i))
|
i,
|
||||||
.spawn(move || {
|
)
|
||||||
panic_handler.catch_panic(move || {
|
}).unwrap()
|
||||||
VerificationQueue::verify(verification, engine, wait, ready, deleting, empty, sleep)
|
})
|
||||||
}).unwrap()
|
.expect("Failed to create verifier thread.");
|
||||||
})
|
verifier_handles.push(handle);
|
||||||
.expect("Failed to create verifier thread.")
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
VerificationQueue {
|
VerificationQueue {
|
||||||
@ -266,13 +242,14 @@ impl<K: Kind> VerificationQueue<K> {
|
|||||||
ready_signal: ready_signal,
|
ready_signal: ready_signal,
|
||||||
more_to_verify: more_to_verify,
|
more_to_verify: more_to_verify,
|
||||||
verification: verification,
|
verification: verification,
|
||||||
verifiers: Mutex::new((verifiers, default_amount)),
|
|
||||||
deleting: deleting,
|
deleting: deleting,
|
||||||
processing: RwLock::new(HashSet::new()),
|
processing: RwLock::new(HashSet::new()),
|
||||||
empty: empty,
|
empty: empty,
|
||||||
ticks_since_adjustment: AtomicUsize::new(0),
|
ticks_since_adjustment: AtomicUsize::new(0),
|
||||||
max_queue_size: max(config.max_queue_size, MIN_QUEUE_LIMIT),
|
max_queue_size: max(config.max_queue_size, MIN_QUEUE_LIMIT),
|
||||||
max_mem_use: max(config.max_mem_use, MIN_MEM_LIMIT),
|
max_mem_use: max(config.max_mem_use, MIN_MEM_LIMIT),
|
||||||
|
verifier_handles: verifier_handles,
|
||||||
|
state: state,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -281,23 +258,30 @@ impl<K: Kind> VerificationQueue<K> {
|
|||||||
engine: Arc<Engine>,
|
engine: Arc<Engine>,
|
||||||
wait: Arc<SCondvar>,
|
wait: Arc<SCondvar>,
|
||||||
ready: Arc<QueueSignal>,
|
ready: Arc<QueueSignal>,
|
||||||
deleting: Arc<AtomicBool>,
|
|
||||||
empty: Arc<SCondvar>,
|
empty: Arc<SCondvar>,
|
||||||
sleep: Arc<AtomicBool>,
|
state: Arc<(Mutex<State>, Condvar)>,
|
||||||
|
id: usize,
|
||||||
) {
|
) {
|
||||||
while !deleting.load(AtomicOrdering::Acquire) {
|
loop {
|
||||||
|
// check current state.
|
||||||
{
|
{
|
||||||
while sleep.load(AtomicOrdering::SeqCst) {
|
let mut cur_state = state.0.lock();
|
||||||
trace!(target: "verification", "Verifier sleeping");
|
while let State::Work(x) = *cur_state {
|
||||||
::std::thread::park();
|
// sleep until this thread is required.
|
||||||
trace!(target: "verification", "Verifier waking up");
|
if id < x { break }
|
||||||
|
|
||||||
if deleting.load(AtomicOrdering::Acquire) {
|
debug!(target: "verification", "verifier {} sleeping", id);
|
||||||
return;
|
state.1.wait(&mut cur_state);
|
||||||
}
|
debug!(target: "verification", "verifier {} waking up", id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let State::Exit = *cur_state {
|
||||||
|
debug!(target: "verification", "verifier {} exiting", id);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// wait for work if empty.
|
||||||
{
|
{
|
||||||
let mut more_to_verify = verification.more_to_verify.lock().unwrap();
|
let mut more_to_verify = verification.more_to_verify.lock().unwrap();
|
||||||
|
|
||||||
@ -305,15 +289,22 @@ impl<K: Kind> VerificationQueue<K> {
|
|||||||
empty.notify_all();
|
empty.notify_all();
|
||||||
}
|
}
|
||||||
|
|
||||||
while verification.unverified.lock().is_empty() && !deleting.load(AtomicOrdering::Acquire) {
|
while verification.unverified.lock().is_empty() {
|
||||||
|
if let State::Exit = *state.0.lock() {
|
||||||
|
debug!(target: "verification", "verifier {} exiting", id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
more_to_verify = wait.wait(more_to_verify).unwrap();
|
more_to_verify = wait.wait(more_to_verify).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
if deleting.load(AtomicOrdering::Acquire) {
|
if let State::Exit = *state.0.lock() {
|
||||||
|
debug!(target: "verification", "verifier {} exiting", id);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// do work.
|
||||||
let item = {
|
let item = {
|
||||||
// acquire these locks before getting the item to verify.
|
// acquire these locks before getting the item to verify.
|
||||||
let mut unverified = verification.unverified.lock();
|
let mut unverified = verification.unverified.lock();
|
||||||
@ -568,6 +559,14 @@ impl<K: Kind> VerificationQueue<K> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get the current number of working verifiers.
|
||||||
|
pub fn num_verifiers(&self) -> usize {
|
||||||
|
match *self.state.0.lock() {
|
||||||
|
State::Work(x) => x,
|
||||||
|
State::Exit => panic!("state only set to exit on drop; queue live now; qed"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Optimise memory footprint of the heap fields, and adjust the number of threads
|
/// Optimise memory footprint of the heap fields, and adjust the number of threads
|
||||||
/// to better suit the workload.
|
/// to better suit the workload.
|
||||||
pub fn collect_garbage(&self) {
|
pub fn collect_garbage(&self) {
|
||||||
@ -604,7 +603,7 @@ impl<K: Kind> VerificationQueue<K> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let current = self.verifiers.lock().1;
|
let current = self.num_verifiers();
|
||||||
|
|
||||||
let diff = (v_len - u_len).abs();
|
let diff = (v_len - u_len).abs();
|
||||||
let total = v_len + u_len;
|
let total = v_len + u_len;
|
||||||
@ -626,27 +625,14 @@ impl<K: Kind> VerificationQueue<K> {
|
|||||||
// possible, never going over the amount of initially allocated threads
|
// possible, never going over the amount of initially allocated threads
|
||||||
// or below 1.
|
// or below 1.
|
||||||
fn scale_verifiers(&self, target: usize) {
|
fn scale_verifiers(&self, target: usize) {
|
||||||
let mut verifiers = self.verifiers.lock();
|
let current = self.num_verifiers();
|
||||||
let &mut (ref mut verifiers, ref mut verifier_count) = &mut *verifiers;
|
let target = min(self.verifier_handles.len(), target);
|
||||||
|
|
||||||
let target = min(verifiers.len(), target);
|
|
||||||
let target = max(1, target);
|
let target = max(1, target);
|
||||||
|
|
||||||
debug!(target: "verification", "Scaling from {} to {} verifiers", verifier_count, target);
|
debug!(target: "verification", "Scaling from {} to {} verifiers", current, target);
|
||||||
|
|
||||||
// scaling up
|
*self.state.0.lock() = State::Work(target);
|
||||||
for i in *verifier_count..target {
|
self.state.1.notify_all();
|
||||||
debug!(target: "verification", "Waking up verifier {}", i);
|
|
||||||
verifiers[i].wake_up();
|
|
||||||
}
|
|
||||||
|
|
||||||
// scaling down.
|
|
||||||
for i in target..*verifier_count {
|
|
||||||
debug!(target: "verification", "Putting verifier {} to sleep", i);
|
|
||||||
verifiers[i].sleep();
|
|
||||||
}
|
|
||||||
|
|
||||||
*verifier_count = target;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -660,22 +646,18 @@ impl<K: Kind> Drop for VerificationQueue<K> {
|
|||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
trace!(target: "shutdown", "[VerificationQueue] Closing...");
|
trace!(target: "shutdown", "[VerificationQueue] Closing...");
|
||||||
self.clear();
|
self.clear();
|
||||||
self.deleting.store(true, AtomicOrdering::Release);
|
self.deleting.store(true, AtomicOrdering::SeqCst);
|
||||||
|
|
||||||
let mut verifiers = self.verifiers.get_mut();
|
// set exit state; should be done before `more_to_verify` notification.
|
||||||
let mut verifiers = &mut verifiers.0;
|
*self.state.0.lock() = State::Exit;
|
||||||
|
self.state.1.notify_all();
|
||||||
// first pass to signal conclusion. must be done before
|
|
||||||
// notify or deadlock possible.
|
|
||||||
for handle in verifiers.iter() {
|
|
||||||
handle.conclude();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// wake up all threads waiting for more work.
|
||||||
self.more_to_verify.notify_all();
|
self.more_to_verify.notify_all();
|
||||||
|
|
||||||
// second pass to join.
|
// wait for all verifier threads to join.
|
||||||
for handle in verifiers.drain(..) {
|
for thread in self.verifier_handles.drain(..) {
|
||||||
handle.join();
|
thread.join().expect("Propagating verifier thread panic on shutdown");
|
||||||
}
|
}
|
||||||
|
|
||||||
trace!(target: "shutdown", "[VerificationQueue] Closed.");
|
trace!(target: "shutdown", "[VerificationQueue] Closed.");
|
||||||
@ -687,7 +669,7 @@ mod tests {
|
|||||||
use util::*;
|
use util::*;
|
||||||
use io::*;
|
use io::*;
|
||||||
use spec::*;
|
use spec::*;
|
||||||
use super::{BlockQueue, Config};
|
use super::{BlockQueue, Config, State};
|
||||||
use super::kind::blocks::Unverified;
|
use super::kind::blocks::Unverified;
|
||||||
use tests::helpers::*;
|
use tests::helpers::*;
|
||||||
use error::*;
|
use error::*;
|
||||||
@ -784,11 +766,11 @@ mod tests {
|
|||||||
let queue = get_test_queue();
|
let queue = get_test_queue();
|
||||||
queue.scale_verifiers(MAX_VERIFIERS + 1);
|
queue.scale_verifiers(MAX_VERIFIERS + 1);
|
||||||
|
|
||||||
assert!(queue.verifiers.lock().1 < MAX_VERIFIERS + 1);
|
assert!(queue.num_verifiers() < MAX_VERIFIERS + 1);
|
||||||
|
|
||||||
queue.scale_verifiers(0);
|
queue.scale_verifiers(0);
|
||||||
|
|
||||||
assert!(queue.verifiers.lock().1 == 1);
|
assert!(queue.num_verifiers() == 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -797,14 +779,7 @@ mod tests {
|
|||||||
|
|
||||||
// put all the verifiers to sleep to ensure
|
// put all the verifiers to sleep to ensure
|
||||||
// the test isn't timing sensitive.
|
// the test isn't timing sensitive.
|
||||||
let num_verifiers = {
|
*queue.state.0.lock() = State::Work(0);
|
||||||
let verifiers = queue.verifiers.lock();
|
|
||||||
for i in 0..verifiers.1 {
|
|
||||||
verifiers.0[i].sleep();
|
|
||||||
}
|
|
||||||
|
|
||||||
verifiers.1
|
|
||||||
};
|
|
||||||
|
|
||||||
for block in get_good_dummy_block_seq(5000) {
|
for block in get_good_dummy_block_seq(5000) {
|
||||||
queue.import(Unverified::new(block)).expect("Block good by definition; qed");
|
queue.import(Unverified::new(block)).expect("Block good by definition; qed");
|
||||||
@ -812,20 +787,12 @@ mod tests {
|
|||||||
|
|
||||||
// almost all unverified == bump verifier count.
|
// almost all unverified == bump verifier count.
|
||||||
queue.collect_garbage();
|
queue.collect_garbage();
|
||||||
assert_eq!(queue.verifiers.lock().1, num_verifiers + 1);
|
assert_eq!(queue.num_verifiers(), 1);
|
||||||
|
|
||||||
// wake them up again and verify everything.
|
|
||||||
{
|
|
||||||
let verifiers = queue.verifiers.lock();
|
|
||||||
for i in 0..verifiers.1 {
|
|
||||||
verifiers.0[i].wake_up();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
queue.flush();
|
queue.flush();
|
||||||
|
|
||||||
// nothing to verify == use minimum number of verifiers.
|
// nothing to verify == use minimum number of verifiers.
|
||||||
queue.collect_garbage();
|
queue.collect_garbage();
|
||||||
assert_eq!(queue.verifiers.lock().1, 1);
|
assert_eq!(queue.num_verifiers(), 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
234
js/package.json
234
js/package.json
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "parity.js",
|
"name": "parity.js",
|
||||||
"version": "0.2.82",
|
"version": "0.2.92",
|
||||||
"main": "release/index.js",
|
"main": "release/index.js",
|
||||||
"jsnext:main": "src/index.js",
|
"jsnext:main": "src/index.js",
|
||||||
"author": "Parity Team <admin@parity.io>",
|
"author": "Parity Team <admin@parity.io>",
|
||||||
@ -47,133 +47,127 @@
|
|||||||
"prepush": "npm run lint:cached"
|
"prepush": "npm run lint:cached"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"babel-cli": "~6.18.0",
|
"babel-cli": "6.18.0",
|
||||||
"babel-core": "~6.18.2",
|
"babel-core": "6.18.2",
|
||||||
"babel-eslint": "~7.1.0",
|
"babel-eslint": "7.1.1",
|
||||||
"babel-loader": "~6.2.3",
|
"babel-loader": "6.2.8",
|
||||||
"babel-plugin-lodash": "~3.2.2",
|
"babel-plugin-lodash": "3.2.10",
|
||||||
"babel-plugin-transform-class-properties": "~6.19.0",
|
"babel-plugin-transform-class-properties": "6.19.0",
|
||||||
"babel-plugin-transform-decorators-legacy": "~1.3.4",
|
"babel-plugin-transform-decorators-legacy": "1.3.4",
|
||||||
"babel-plugin-transform-react-remove-prop-types": "~0.2.9",
|
"babel-plugin-transform-react-remove-prop-types": "0.2.11",
|
||||||
"babel-plugin-transform-runtime": "~6.15.0",
|
"babel-plugin-transform-runtime": "6.15.0",
|
||||||
"babel-polyfill": "~6.16.0",
|
"babel-polyfill": "6.16.0",
|
||||||
"babel-preset-es2015": "~6.18.0",
|
"babel-preset-es2015": "6.18.0",
|
||||||
"babel-preset-es2015-rollup": "~1.2.0",
|
"babel-preset-es2015-rollup": "1.2.0",
|
||||||
"babel-preset-es2016": "~6.16.0",
|
"babel-preset-es2016": "6.16.0",
|
||||||
"babel-preset-es2017": "~6.16.0",
|
"babel-preset-es2017": "6.16.0",
|
||||||
"babel-preset-react": "~6.16.0",
|
"babel-preset-react": "6.16.0",
|
||||||
"babel-preset-stage-0": "~6.16.0",
|
"babel-preset-stage-0": "6.16.0",
|
||||||
"babel-register": "6.18.0",
|
"babel-register": "6.18.0",
|
||||||
"babel-runtime": "~6.18.0",
|
"babel-runtime": "6.18.0",
|
||||||
"chai": "~3.5.0",
|
"chai": "3.5.0",
|
||||||
"chai-enzyme": "0.4.2",
|
"chai-enzyme": "0.6.1",
|
||||||
"cheerio": "0.20.0",
|
"copy-webpack-plugin": "4.0.1",
|
||||||
"copy-webpack-plugin": "~4.0.0",
|
"core-js": "2.4.1",
|
||||||
"core-js": "~2.4.1",
|
"coveralls": "2.11.15",
|
||||||
"coveralls": "~2.11.11",
|
"css-loader": "0.26.1",
|
||||||
"css-loader": "~0.26.0",
|
"ejs-loader": "0.3.0",
|
||||||
"ejs-loader": "~0.3.0",
|
"enzyme": "2.6.0",
|
||||||
"enzyme": "2.3.0",
|
"eslint": "3.11.1",
|
||||||
"eslint": "~3.10.2",
|
"eslint-config-semistandard": "7.0.0",
|
||||||
"eslint-config-semistandard": "~7.0.0",
|
"eslint-config-standard": "6.2.1",
|
||||||
"eslint-config-standard": "~6.2.1",
|
"eslint-config-standard-react": "4.2.0",
|
||||||
"eslint-config-standard-react": "~4.2.0",
|
"eslint-plugin-promise": "3.4.0",
|
||||||
"eslint-plugin-promise": "~3.4.0",
|
"eslint-plugin-react": "6.7.1",
|
||||||
"eslint-plugin-react": "~6.7.1",
|
"eslint-plugin-standard": "2.0.1",
|
||||||
"eslint-plugin-standard": "~2.0.0",
|
"express": "4.14.0",
|
||||||
"express": "~4.14.0",
|
|
||||||
"extract-loader": "0.1.0",
|
"extract-loader": "0.1.0",
|
||||||
"extract-text-webpack-plugin": "~2.0.0-beta.4",
|
"extract-text-webpack-plugin": "2.0.0-beta.4",
|
||||||
"file-loader": "~0.9.0",
|
"file-loader": "0.9.0",
|
||||||
"fs-extra": "~0.30.0",
|
"happypack": "3.0.0",
|
||||||
"happypack": "~3.0.0",
|
"html-loader": "0.4.4",
|
||||||
"history": "~2.0.0",
|
"html-webpack-plugin": "2.24.1",
|
||||||
"html-loader": "~0.4.4",
|
"http-proxy-middleware": "0.17.2",
|
||||||
"html-webpack-plugin": "~2.24.1",
|
"husky": "0.11.9",
|
||||||
"http-proxy-middleware": "~0.17.2",
|
"ignore-styles": "5.0.1",
|
||||||
"husky": "~0.11.9",
|
"image-webpack-loader": "3.0.0",
|
||||||
"ignore-styles": "2.0.0",
|
"istanbul": "1.0.0-alpha.2",
|
||||||
"image-webpack-loader": "~3.0.0",
|
"jsdom": "9.8.3",
|
||||||
"istanbul": "~1.0.0-alpha.2",
|
"json-loader": "0.5.4",
|
||||||
"jsdom": "9.2.1",
|
"mocha": "3.2.0",
|
||||||
"json-loader": "~0.5.4",
|
|
||||||
"mocha": "~3.0.0-1",
|
|
||||||
"mock-local-storage": "1.0.2",
|
"mock-local-storage": "1.0.2",
|
||||||
"mock-socket": "~3.0.1",
|
"mock-socket": "6.0.3",
|
||||||
"nock": "~8.0.0",
|
"nock": "9.0.2",
|
||||||
"postcss-import": "8.1.0",
|
"postcss-import": "8.1.0",
|
||||||
"postcss-loader": "~1.1.1",
|
"postcss-loader": "1.1.1",
|
||||||
"postcss-nested": "~1.0.0",
|
"postcss-nested": "1.0.0",
|
||||||
"postcss-simple-vars": "~3.0.0",
|
"postcss-simple-vars": "3.0.0",
|
||||||
"progress": "~1.1.8",
|
"progress": "1.1.8",
|
||||||
"raw-loader": "~0.5.1",
|
"raw-loader": "0.5.1",
|
||||||
"react-addons-perf": "~15.3.2",
|
"react-addons-perf": "15.4.1",
|
||||||
"react-addons-test-utils": "~15.3.2",
|
"react-addons-test-utils": "15.4.1",
|
||||||
"react-dom": "~15.3.2",
|
"react-hot-loader": "3.0.0-beta.6",
|
||||||
"react-hot-loader": "~3.0.0-beta.6",
|
"rucksack-css": "0.9.1",
|
||||||
"rucksack-css": "~0.8.6",
|
"sinon": "1.17.6",
|
||||||
"sinon": "~1.17.4",
|
"sinon-as-promised": "4.0.2",
|
||||||
"sinon-as-promised": "~4.0.2",
|
"sinon-chai": "2.8.0",
|
||||||
"sinon-chai": "~2.8.0",
|
"style-loader": "0.13.1",
|
||||||
"style-loader": "~0.13.0",
|
"url-loader": "0.5.7",
|
||||||
"url-loader": "~0.5.7",
|
"webpack": "2.1.0-beta.27",
|
||||||
"webpack": "~2.1.0-beta.27",
|
"webpack-dev-middleware": "1.8.4",
|
||||||
"webpack-dev-middleware": "~1.8.4",
|
|
||||||
"webpack-error-notification": "0.1.6",
|
"webpack-error-notification": "0.1.6",
|
||||||
"webpack-hot-middleware": "~2.13.2",
|
"webpack-hot-middleware": "2.13.2",
|
||||||
"websocket": "~1.0.23"
|
"websocket": "1.0.23"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"bignumber.js": "~2.3.0",
|
"bignumber.js": "3.0.1",
|
||||||
"blockies": "0.0.2",
|
"blockies": "0.0.2",
|
||||||
"brace": "~0.9.0",
|
"brace": "0.9.0",
|
||||||
"bytes": "~2.4.0",
|
"bytes": "2.4.0",
|
||||||
"chart.js": "~2.3.0",
|
"es6-error": "4.0.0",
|
||||||
"es6-error": "~4.0.0",
|
"es6-promise": "4.0.5",
|
||||||
"es6-promise": "~3.2.1",
|
"ethereumjs-tx": "1.1.4",
|
||||||
"ethereumjs-tx": "~1.1.2",
|
"eventemitter3": "2.0.2",
|
||||||
"eventemitter3": "~2.0.2",
|
"file-saver": "1.3.3",
|
||||||
"file-saver": "~1.3.3",
|
"format-json": "1.0.3",
|
||||||
"format-json": "~1.0.3",
|
"format-number": "2.0.1",
|
||||||
"format-number": "~2.0.1",
|
"geopattern": "1.2.3",
|
||||||
"geopattern": "~1.2.3",
|
"isomorphic-fetch": "2.2.1",
|
||||||
"isomorphic-fetch": "~2.2.1",
|
"js-sha3": "0.5.5",
|
||||||
"js-sha3": "~0.5.2",
|
"lodash": "4.17.2",
|
||||||
"lodash": "~4.11.1",
|
"marked": "0.3.6",
|
||||||
"marked": "~0.3.6",
|
"material-ui": "0.16.4",
|
||||||
"material-ui": "0.16.1",
|
"material-ui-chip-input": "0.11.1",
|
||||||
"material-ui-chip-input": "~0.8.0",
|
"mobx": "2.6.4",
|
||||||
"mobx": "~2.6.1",
|
"mobx-react": "4.0.3",
|
||||||
"mobx-react": "~3.5.8",
|
"mobx-react-devtools": "4.2.10",
|
||||||
"mobx-react-devtools": "~4.2.9",
|
"moment": "2.17.0",
|
||||||
"moment": "~2.14.1",
|
"phoneformat.js": "1.0.3",
|
||||||
"phoneformat.js": "~1.0.3",
|
"qs": "6.3.0",
|
||||||
"qs": "~6.3.0",
|
"react": "15.4.1",
|
||||||
"react": "~15.3.2",
|
"react-ace": "4.1.0",
|
||||||
"react-ace": "~4.0.0",
|
"react-addons-css-transition-group": "15.4.1",
|
||||||
"react-addons-css-transition-group": "~15.3.2",
|
"react-copy-to-clipboard": "4.2.3",
|
||||||
"react-chartjs-2": "~1.5.0",
|
"react-dom": "15.4.1",
|
||||||
"react-copy-to-clipboard": "~4.2.3",
|
"react-dropzone": "3.7.3",
|
||||||
"react-dom": "~15.3.2",
|
"react-redux": "4.4.6",
|
||||||
"react-dropzone": "~3.7.3",
|
"react-router": "3.0.0",
|
||||||
"react-redux": "~4.4.5",
|
"react-router-redux": "4.0.7",
|
||||||
"react-router": "~2.6.1",
|
"react-tap-event-plugin": "2.0.1",
|
||||||
"react-router-redux": "~4.0.5",
|
"react-tooltip": "3.2.2",
|
||||||
"react-tap-event-plugin": "~1.0.0",
|
"recharts": "0.15.2",
|
||||||
"react-tooltip": "~2.0.3",
|
"redux": "3.6.0",
|
||||||
"recharts": "~0.15.2",
|
"redux-actions": "1.1.0",
|
||||||
"redux": "~3.5.2",
|
"redux-thunk": "2.1.0",
|
||||||
"redux-actions": "~0.10.1",
|
"rlp": "2.0.0",
|
||||||
"redux-thunk": "~2.1.0",
|
"scryptsy": "2.0.0",
|
||||||
"rlp": "~2.0.0",
|
|
||||||
"scryptsy": "~2.0.0",
|
|
||||||
"solc": "ngotchac/solc-js",
|
"solc": "ngotchac/solc-js",
|
||||||
"store": "~1.3.20",
|
"store": "1.3.20",
|
||||||
"utf8": "~2.1.1",
|
"utf8": "2.1.2",
|
||||||
"valid-url": "~1.0.9",
|
"valid-url": "1.0.9",
|
||||||
"validator": "~5.7.0",
|
"validator": "6.2.0",
|
||||||
"web3": "~0.17.0-beta",
|
"web3": "0.17.0-beta",
|
||||||
"whatwg-fetch": "~1.0.0",
|
"whatwg-fetch": "2.0.1",
|
||||||
"worker-loader": "~0.7.1"
|
"worker-loader": "0.7.1"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -27,9 +27,5 @@ export function sliceData (_data) {
|
|||||||
data = padAddress('');
|
data = padAddress('');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data.length % 64) {
|
|
||||||
throw new Error(`Invalid data length (not mod 64) passed to sliceData, ${data}, % 64 == ${data.length % 64}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
return data.match(/.{1,64}/g);
|
return data.match(/.{1,64}/g);
|
||||||
}
|
}
|
||||||
|
@ -21,10 +21,6 @@ describe('abi/util/slice', () => {
|
|||||||
const slice1 = '131a3afc00d1b1e3461b955e53fc866dcf303b3eb9f4c16f89e388930f48134b';
|
const slice1 = '131a3afc00d1b1e3461b955e53fc866dcf303b3eb9f4c16f89e388930f48134b';
|
||||||
const slice2 = '2124768576358735263578356373526387638357635873563586353756358763';
|
const slice2 = '2124768576358735263578356373526387638357635873563586353756358763';
|
||||||
|
|
||||||
it('throws an error on mod 64 != 0', () => {
|
|
||||||
expect(() => sliceData('123')).to.throw(/sliceData/);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('returns an empty array when length === 0', () => {
|
it('returns an empty array when length === 0', () => {
|
||||||
expect(sliceData('')).to.deep.equal([]);
|
expect(sliceData('')).to.deep.equal([]);
|
||||||
});
|
});
|
||||||
|
@ -240,9 +240,29 @@ export default class Contract {
|
|||||||
return this.unsubscribe(subscriptionId);
|
return this.unsubscribe(subscriptionId);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
event.getAllLogs = (options = {}) => {
|
||||||
|
return this.getAllLogs(event);
|
||||||
|
};
|
||||||
|
|
||||||
return event;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getAllLogs (event, _options) {
|
||||||
|
// Options as first parameter
|
||||||
|
if (!_options && event && event.topics) {
|
||||||
|
return this.getAllLogs(null, event);
|
||||||
|
}
|
||||||
|
|
||||||
|
const options = this._getFilterOptions(event, _options);
|
||||||
|
return this._api.eth
|
||||||
|
.getLogs({
|
||||||
|
fromBlock: 0,
|
||||||
|
toBlock: 'latest',
|
||||||
|
...options
|
||||||
|
})
|
||||||
|
.then((logs) => this.parseEventLogs(logs));
|
||||||
|
}
|
||||||
|
|
||||||
_findEvent (eventName = null) {
|
_findEvent (eventName = null) {
|
||||||
const event = eventName
|
const event = eventName
|
||||||
? this._events.find((evt) => evt.name === eventName)
|
? this._events.find((evt) => evt.name === eventName)
|
||||||
@ -256,7 +276,7 @@ export default class Contract {
|
|||||||
return event;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
_createEthFilter (event = null, _options) {
|
_getFilterOptions (event = null, _options = {}) {
|
||||||
const optionTopics = _options.topics || [];
|
const optionTopics = _options.topics || [];
|
||||||
const signature = event && event.signature || null;
|
const signature = event && event.signature || null;
|
||||||
|
|
||||||
@ -271,6 +291,11 @@ export default class Contract {
|
|||||||
topics
|
topics
|
||||||
});
|
});
|
||||||
|
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
|
||||||
|
_createEthFilter (event = null, _options) {
|
||||||
|
const options = this._getFilterOptions(event, _options);
|
||||||
return this._api.eth.newFilter(options);
|
return this._api.eth.newFilter(options);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -146,7 +146,8 @@ export default class Eth {
|
|||||||
|
|
||||||
getLogs (options) {
|
getLogs (options) {
|
||||||
return this._transport
|
return this._transport
|
||||||
.execute('eth_getLogs', inFilter(options));
|
.execute('eth_getLogs', inFilter(options))
|
||||||
|
.then((logs) => logs.map(outLog));
|
||||||
}
|
}
|
||||||
|
|
||||||
getLogsEx (options) {
|
getLogsEx (options) {
|
||||||
|
@ -32,6 +32,10 @@ export function hex2Ascii (_hex) {
|
|||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function bytesToAscii (bytes) {
|
||||||
|
return bytes.map((b) => String.fromCharCode(b % 512)).join('');
|
||||||
|
}
|
||||||
|
|
||||||
export function asciiToHex (string) {
|
export function asciiToHex (string) {
|
||||||
return '0x' + string.split('').map((s) => s.charCodeAt(0).toString(16)).join('');
|
return '0x' + string.split('').map((s) => s.charCodeAt(0).toString(16)).join('');
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import { bytesToHex, hex2Ascii } from '../api/util/format';
|
import { bytesToHex, hex2Ascii } from '~/api/util/format';
|
||||||
|
|
||||||
import ABI from './abi/certifier.json';
|
import ABI from './abi/certifier.json';
|
||||||
|
|
||||||
|
@ -16,8 +16,7 @@
|
|||||||
|
|
||||||
import ReactDOM from 'react-dom';
|
import ReactDOM from 'react-dom';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { createHashHistory } from 'history';
|
import { Redirect, Router, Route, hashHistory } from 'react-router';
|
||||||
import { Redirect, Router, Route, useRouterHistory } from 'react-router';
|
|
||||||
|
|
||||||
import injectTapEventPlugin from 'react-tap-event-plugin';
|
import injectTapEventPlugin from 'react-tap-event-plugin';
|
||||||
injectTapEventPlugin();
|
injectTapEventPlugin();
|
||||||
@ -27,14 +26,12 @@ import Application from './basiccoin/Application';
|
|||||||
import Overview from './basiccoin/Overview';
|
import Overview from './basiccoin/Overview';
|
||||||
import Transfer from './basiccoin/Transfer';
|
import Transfer from './basiccoin/Transfer';
|
||||||
|
|
||||||
const routerHistory = useRouterHistory(createHashHistory)({});
|
|
||||||
|
|
||||||
import '../../assets/fonts/Roboto/font.css';
|
import '../../assets/fonts/Roboto/font.css';
|
||||||
import '../../assets/fonts/RobotoMono/font.css';
|
import '../../assets/fonts/RobotoMono/font.css';
|
||||||
import './style.css';
|
import './style.css';
|
||||||
|
|
||||||
ReactDOM.render(
|
ReactDOM.render(
|
||||||
<Router history={ routerHistory }>
|
<Router history={ hashHistory }>
|
||||||
<Redirect from='/' to='/overview' />
|
<Redirect from='/' to='/overview' />
|
||||||
<Route path='/' component={ Application }>
|
<Route path='/' component={ Application }>
|
||||||
<Route path='deploy' component={ Deploy } />
|
<Route path='deploy' component={ Deploy } />
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import BigNumber from 'bignumber.js';
|
import BigNumber from 'bignumber.js';
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
|
|
||||||
import { eip20 } from '../../../../contracts/abi';
|
import { eip20 } from '~/contracts/abi';
|
||||||
|
|
||||||
import { api } from '../../parity';
|
import { api } from '../../parity';
|
||||||
import { loadBalances } from '../../services';
|
import { loadBalances } from '../../services';
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
import BigNumber from 'bignumber.js';
|
import BigNumber from 'bignumber.js';
|
||||||
|
|
||||||
import * as abis from '../../contracts/abi';
|
import * as abis from '~/contracts/abi';
|
||||||
import { api } from './parity';
|
import { api } from './parity';
|
||||||
|
|
||||||
let managerInstance;
|
let managerInstance;
|
||||||
|
@ -17,8 +17,8 @@
|
|||||||
import BigNumber from 'bignumber.js';
|
import BigNumber from 'bignumber.js';
|
||||||
import { action, computed, observable, transaction } from 'mobx';
|
import { action, computed, observable, transaction } from 'mobx';
|
||||||
|
|
||||||
import * as abis from '../../contracts/abi';
|
import * as abis from '~/contracts/abi';
|
||||||
import builtins from '../../views/Dapps/builtin.json';
|
import builtins from '~/views/Dapps/builtin.json';
|
||||||
|
|
||||||
import { api } from './parity';
|
import { api } from './parity';
|
||||||
|
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import * as abis from '../../contracts/abi';
|
import * as abis from '~/contracts/abi';
|
||||||
import { api } from './parity';
|
import { api } from './parity';
|
||||||
|
|
||||||
export function attachInterface () {
|
export function attachInterface () {
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
<style>
|
<style>
|
||||||
html, body, #container {
|
html, body, #container {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
min-height: 100%;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
background: white;
|
background: white;
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import { registry as registryAbi } from '../../contracts/abi';
|
import { registry as registryAbi } from '~/contracts/abi';
|
||||||
|
|
||||||
import { api } from './parity.js';
|
import { api } from './parity.js';
|
||||||
import * as addresses from './addresses/actions.js';
|
import * as addresses from './addresses/actions.js';
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import * as abis from '../../contracts/abi';
|
import * as abis from '~/contracts/abi';
|
||||||
import { api } from './parity';
|
import { api } from './parity';
|
||||||
|
|
||||||
const sortEvents = (a, b) => b.blockNumber.cmp(a.blockNumber) || b.logIndex.cmp(a.logIndex);
|
const sortEvents = (a, b) => b.blockNumber.cmp(a.blockNumber) || b.logIndex.cmp(a.logIndex);
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import Contracts from '../../../contracts';
|
import Contracts from '~/contracts';
|
||||||
|
|
||||||
import { loadToken, setTokenPending, deleteToken, setTokenData } from '../Tokens/actions';
|
import { loadToken, setTokenPending, deleteToken, setTokenData } from '../Tokens/actions';
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
import { api } from './parity';
|
import { api } from './parity';
|
||||||
|
|
||||||
import { eip20 as eip20Abi } from '../../contracts/abi';
|
import { eip20 as eip20Abi } from '~/contracts/abi';
|
||||||
|
|
||||||
export const getTokenTotalSupply = (tokenAddress) => {
|
export const getTokenTotalSupply = (tokenAddress) => {
|
||||||
return api
|
return api
|
||||||
|
@ -25,19 +25,18 @@ import ReactDOM from 'react-dom';
|
|||||||
import { AppContainer } from 'react-hot-loader';
|
import { AppContainer } from 'react-hot-loader';
|
||||||
|
|
||||||
import injectTapEventPlugin from 'react-tap-event-plugin';
|
import injectTapEventPlugin from 'react-tap-event-plugin';
|
||||||
import { createHashHistory } from 'history';
|
import { hashHistory } from 'react-router';
|
||||||
import { useRouterHistory } from 'react-router';
|
|
||||||
import qs from 'querystring';
|
import qs from 'querystring';
|
||||||
|
|
||||||
import SecureApi from './secureApi';
|
import SecureApi from './secureApi';
|
||||||
import ContractInstances from './contracts';
|
import ContractInstances from '~/contracts';
|
||||||
|
|
||||||
import { initStore } from './redux';
|
import { initStore } from './redux';
|
||||||
import ContextProvider from './ui/ContextProvider';
|
import ContextProvider from '~/ui/ContextProvider';
|
||||||
import muiTheme from './ui/Theme';
|
import muiTheme from '~/ui/Theme';
|
||||||
import MainApplication from './main';
|
import MainApplication from './main';
|
||||||
|
|
||||||
import { setApi } from './redux/providers/apiActions';
|
import { setApi } from '~/redux/providers/apiActions';
|
||||||
|
|
||||||
import './environment';
|
import './environment';
|
||||||
|
|
||||||
@ -74,13 +73,11 @@ store.dispatch(setApi(api));
|
|||||||
|
|
||||||
window.secureApi = api;
|
window.secureApi = api;
|
||||||
|
|
||||||
const routerHistory = useRouterHistory(createHashHistory)({});
|
|
||||||
|
|
||||||
ReactDOM.render(
|
ReactDOM.render(
|
||||||
<AppContainer>
|
<AppContainer>
|
||||||
<ContextProvider api={ api } muiTheme={ muiTheme } store={ store }>
|
<ContextProvider api={ api } muiTheme={ muiTheme } store={ store }>
|
||||||
<MainApplication
|
<MainApplication
|
||||||
routerHistory={ routerHistory }
|
routerHistory={ hashHistory }
|
||||||
/>
|
/>
|
||||||
</ContextProvider>
|
</ContextProvider>
|
||||||
</AppContainer>,
|
</AppContainer>,
|
||||||
@ -88,24 +85,6 @@ ReactDOM.render(
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (module.hot) {
|
if (module.hot) {
|
||||||
// module.hot.accept('./redux', () => {
|
|
||||||
// // redux store has a method replaceReducer
|
|
||||||
// // const newStore = initStore(api);
|
|
||||||
// console.warn('REDUX UPDATE');
|
|
||||||
// // store.replaceReducer(appReducer);
|
|
||||||
|
|
||||||
// // ReactDOM.render(
|
|
||||||
// // <AppContainer>
|
|
||||||
// // <ContextProvider api={ api } muiTheme={ muiTheme } store={ newStore }>
|
|
||||||
// // <MainApplication
|
|
||||||
// // routerHistory={ routerHistory }
|
|
||||||
// // />
|
|
||||||
// // </ContextProvider>
|
|
||||||
// // </AppContainer>,
|
|
||||||
// // document.querySelector('#container')
|
|
||||||
// // );
|
|
||||||
// });
|
|
||||||
|
|
||||||
module.hot.accept('./main.js', () => {
|
module.hot.accept('./main.js', () => {
|
||||||
require('./main.js');
|
require('./main.js');
|
||||||
|
|
||||||
@ -113,7 +92,7 @@ if (module.hot) {
|
|||||||
<AppContainer>
|
<AppContainer>
|
||||||
<ContextProvider api={ api } muiTheme={ muiTheme } store={ store }>
|
<ContextProvider api={ api } muiTheme={ muiTheme } store={ store }>
|
||||||
<MainApplication
|
<MainApplication
|
||||||
routerHistory={ routerHistory }
|
routerHistory={ hashHistory }
|
||||||
/>
|
/>
|
||||||
</ContextProvider>
|
</ContextProvider>
|
||||||
</AppContainer>,
|
</AppContainer>,
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { Redirect, Router, Route } from 'react-router';
|
import { Redirect, Router, Route } from 'react-router';
|
||||||
|
|
||||||
import { Accounts, Account, Addresses, Address, Application, Contract, Contracts, WriteContract, Dapp, Dapps, Settings, SettingsBackground, SettingsParity, SettingsProxy, SettingsViews, Signer, Status } from './views';
|
import { Accounts, Account, Addresses, Address, Application, Contract, Contracts, WriteContract, Dapp, Dapps, Settings, SettingsBackground, SettingsParity, SettingsProxy, SettingsViews, Signer, Status } from '~/views';
|
||||||
|
|
||||||
import styles from './reset.css';
|
import styles from './reset.css';
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import ContentAdd from 'material-ui/svg-icons/content/add';
|
import ContentAdd from 'material-ui/svg-icons/content/add';
|
||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
|
|
||||||
import { Button, Modal, Form, Input, InputAddress } from '../../ui';
|
import { Button, Modal, Form, Input, InputAddress } from '~/ui';
|
||||||
import { ERRORS, validateAddress, validateName } from '../../util/validation';
|
import { ERRORS, validateAddress, validateName } from '../../util/validation';
|
||||||
|
|
||||||
export default class AddAddress extends Component {
|
export default class AddAddress extends Component {
|
||||||
|
@ -20,10 +20,10 @@ import ContentClear from 'material-ui/svg-icons/content/clear';
|
|||||||
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
||||||
import NavigationArrowBack from 'material-ui/svg-icons/navigation/arrow-back';
|
import NavigationArrowBack from 'material-ui/svg-icons/navigation/arrow-back';
|
||||||
|
|
||||||
import { Button, Modal, Form, Input, InputAddress, RadioButtons } from '../../ui';
|
import { Button, Modal, Form, Input, InputAddress, RadioButtons } from '~/ui';
|
||||||
import { ERRORS, validateAbi, validateAddress, validateName } from '../../util/validation';
|
import { ERRORS, validateAbi, validateAddress, validateName } from '../../util/validation';
|
||||||
|
|
||||||
import { eip20, wallet } from '../../contracts/abi';
|
import { eip20, wallet } from '~/contracts/abi';
|
||||||
|
|
||||||
const ABI_TYPES = [
|
const ABI_TYPES = [
|
||||||
{
|
{
|
||||||
|
@ -15,15 +15,8 @@
|
|||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import PrintIcon from 'material-ui/svg-icons/action/print';
|
|
||||||
|
|
||||||
import { Form, Input, InputAddress } from '../../../ui';
|
import { Form, Input, InputAddress } from '~/ui';
|
||||||
import Button from '../../../ui/Button';
|
|
||||||
|
|
||||||
import { createIdentityImg } from '../../../api/util/identity';
|
|
||||||
import print from './print';
|
|
||||||
import recoveryPage from './recovery-page.ejs';
|
|
||||||
import ParityLogo from '../../../../assets/images/parity-logo-black-no-text.svg';
|
|
||||||
|
|
||||||
export default class AccountDetails extends Component {
|
export default class AccountDetails extends Component {
|
||||||
static propTypes = {
|
static propTypes = {
|
||||||
@ -49,7 +42,6 @@ export default class AccountDetails extends Component {
|
|||||||
label='address'
|
label='address'
|
||||||
value={ address } />
|
value={ address } />
|
||||||
{ this.renderPhrase() }
|
{ this.renderPhrase() }
|
||||||
{ this.renderPhraseCopyButton() }
|
|
||||||
</Form>
|
</Form>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -70,26 +62,4 @@ export default class AccountDetails extends Component {
|
|||||||
value={ phrase } />
|
value={ phrase } />
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
renderPhraseCopyButton () {
|
|
||||||
const { phrase } = this.props;
|
|
||||||
if (!phrase) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Button
|
|
||||||
icon={ <PrintIcon /> }
|
|
||||||
label={ 'print recovery phrase' }
|
|
||||||
onClick={ this.printPhrase }
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
printPhrase = () => {
|
|
||||||
const { address, phrase, name } = this.props;
|
|
||||||
const identity = createIdentityImg(address);
|
|
||||||
|
|
||||||
print(recoveryPage({ phrase, name, identity, address, logo: ParityLogo }));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,7 @@ import IconButton from 'material-ui/IconButton';
|
|||||||
import { RadioButton, RadioButtonGroup } from 'material-ui/RadioButton';
|
import { RadioButton, RadioButtonGroup } from 'material-ui/RadioButton';
|
||||||
import ActionAutorenew from 'material-ui/svg-icons/action/autorenew';
|
import ActionAutorenew from 'material-ui/svg-icons/action/autorenew';
|
||||||
|
|
||||||
import { Form, Input, IdentityIcon } from '../../../ui';
|
import { Form, Input, IdentityIcon } from '~/ui';
|
||||||
|
|
||||||
import styles from '../createAccount.css';
|
import styles from '../createAccount.css';
|
||||||
|
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { Checkbox } from 'material-ui';
|
import { Checkbox } from 'material-ui';
|
||||||
|
|
||||||
import { IdentityIcon } from '../../../ui';
|
import { IdentityIcon } from '~/ui';
|
||||||
|
|
||||||
import styles from './newGeth.css';
|
import styles from './newGeth.css';
|
||||||
|
|
||||||
|
@ -19,7 +19,7 @@ import ReactDOM from 'react-dom';
|
|||||||
import { FloatingActionButton } from 'material-ui';
|
import { FloatingActionButton } from 'material-ui';
|
||||||
import EditorAttachFile from 'material-ui/svg-icons/editor/attach-file';
|
import EditorAttachFile from 'material-ui/svg-icons/editor/attach-file';
|
||||||
|
|
||||||
import { Form, Input } from '../../../ui';
|
import { Form, Input } from '~/ui';
|
||||||
|
|
||||||
import styles from '../createAccount.css';
|
import styles from '../createAccount.css';
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
|
|
||||||
import { Form, Input } from '../../../ui';
|
import { Form, Input } from '~/ui';
|
||||||
|
|
||||||
import styles from '../createAccount.css';
|
import styles from '../createAccount.css';
|
||||||
|
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { Checkbox } from 'material-ui';
|
import { Checkbox } from 'material-ui';
|
||||||
|
|
||||||
import { Form, Input } from '../../../ui';
|
import { Form, Input } from '~/ui';
|
||||||
|
|
||||||
import styles from '../createAccount.css';
|
import styles from '../createAccount.css';
|
||||||
|
|
||||||
|
@ -20,8 +20,9 @@ import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
|||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
import NavigationArrowBack from 'material-ui/svg-icons/navigation/arrow-back';
|
import NavigationArrowBack from 'material-ui/svg-icons/navigation/arrow-back';
|
||||||
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
||||||
|
import PrintIcon from 'material-ui/svg-icons/action/print';
|
||||||
|
|
||||||
import { Button, Modal } from '../../ui';
|
import { Button, Modal } from '~/ui';
|
||||||
|
|
||||||
import AccountDetails from './AccountDetails';
|
import AccountDetails from './AccountDetails';
|
||||||
import AccountDetailsGeth from './AccountDetailsGeth';
|
import AccountDetailsGeth from './AccountDetailsGeth';
|
||||||
@ -32,6 +33,11 @@ import NewImport from './NewImport';
|
|||||||
import RawKey from './RawKey';
|
import RawKey from './RawKey';
|
||||||
import RecoveryPhrase from './RecoveryPhrase';
|
import RecoveryPhrase from './RecoveryPhrase';
|
||||||
|
|
||||||
|
import { createIdentityImg } from '~/api/util/identity';
|
||||||
|
import print from './print';
|
||||||
|
import recoveryPage from './recovery-page.ejs';
|
||||||
|
import ParityLogo from '../../../assets/images/parity-logo-black-no-text.svg';
|
||||||
|
|
||||||
const TITLES = {
|
const TITLES = {
|
||||||
type: 'creation type',
|
type: 'creation type',
|
||||||
create: 'create account',
|
create: 'create account',
|
||||||
@ -179,12 +185,18 @@ export default class CreateAccount extends Component {
|
|||||||
];
|
];
|
||||||
|
|
||||||
case 2:
|
case 2:
|
||||||
return (
|
return [
|
||||||
|
createType === 'fromNew' || createType === 'fromPhrase' ? (
|
||||||
|
<Button
|
||||||
|
icon={ <PrintIcon /> }
|
||||||
|
label='Print Phrase'
|
||||||
|
onClick={ this.printPhrase } />
|
||||||
|
) : null,
|
||||||
<Button
|
<Button
|
||||||
icon={ <ActionDoneAll /> }
|
icon={ <ActionDoneAll /> }
|
||||||
label='Close'
|
label='Close'
|
||||||
onClick={ this.onClose } />
|
onClick={ this.onClose } />
|
||||||
);
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -377,4 +389,11 @@ export default class CreateAccount extends Component {
|
|||||||
|
|
||||||
store.dispatch({ type: 'newError', error });
|
store.dispatch({ type: 'newError', error });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
printPhrase = () => {
|
||||||
|
const { address, phrase, name } = this.state;
|
||||||
|
const identity = createIdentityImg(address);
|
||||||
|
|
||||||
|
print(recoveryPage({ phrase, name, identity, address, logo: ParityLogo }));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,7 +18,7 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { bindActionCreators } from 'redux';
|
import { bindActionCreators } from 'redux';
|
||||||
|
|
||||||
import { ConfirmDialog, IdentityIcon, IdentityName, Input } from '../../ui';
|
import { ConfirmDialog, IdentityIcon, IdentityName, Input } from '~/ui';
|
||||||
import { newError } from '../../redux/actions';
|
import { newError } from '../../redux/actions';
|
||||||
|
|
||||||
import styles from './deleteAccount.css';
|
import styles from './deleteAccount.css';
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { MenuItem } from 'material-ui';
|
import { MenuItem } from 'material-ui';
|
||||||
|
|
||||||
import { AddressSelect, Form, Input, Select } from '../../../ui';
|
import { AddressSelect, Form, Input, Select } from '~/ui';
|
||||||
import { validateAbi } from '../../../util/validation';
|
import { validateAbi } from '../../../util/validation';
|
||||||
import { parseAbiType } from '../../../util/abi';
|
import { parseAbiType } from '../../../util/abi';
|
||||||
|
|
||||||
|
@ -31,7 +31,7 @@
|
|||||||
|
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
|
|
||||||
import { Form, TypedInput } from '../../../ui';
|
import { Form, TypedInput } from '~/ui';
|
||||||
import { parseAbiType } from '../../../util/abi';
|
import { parseAbiType } from '../../../util/abi';
|
||||||
|
|
||||||
import styles from '../deployContract.css';
|
import styles from '../deployContract.css';
|
||||||
|
@ -18,7 +18,7 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
|
|
||||||
import { BusyStep, CompletedStep, CopyToClipboard, Button, IdentityIcon, Modal, TxHash } from '../../ui';
|
import { BusyStep, CompletedStep, CopyToClipboard, Button, IdentityIcon, Modal, TxHash } from '~/ui';
|
||||||
import { ERRORS, validateAbi, validateCode, validateName } from '../../util/validation';
|
import { ERRORS, validateAbi, validateCode, validateName } from '../../util/validation';
|
||||||
|
|
||||||
import DetailsStep from './DetailsStep';
|
import DetailsStep from './DetailsStep';
|
||||||
@ -27,7 +27,7 @@ import ErrorStep from './ErrorStep';
|
|||||||
|
|
||||||
import styles from './deployContract.css';
|
import styles from './deployContract.css';
|
||||||
|
|
||||||
import { ERROR_CODES } from '../../api/transport/error';
|
import { ERROR_CODES } from '~/api/transport/error';
|
||||||
|
|
||||||
const STEPS = {
|
const STEPS = {
|
||||||
CONTRACT_DETAILS: { title: 'contract details' },
|
CONTRACT_DETAILS: { title: 'contract details' },
|
||||||
|
@ -18,7 +18,7 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
import ContentSave from 'material-ui/svg-icons/content/save';
|
import ContentSave from 'material-ui/svg-icons/content/save';
|
||||||
|
|
||||||
import { Button, Form, Input, InputChip, Modal } from '../../ui';
|
import { Button, Form, Input, InputChip, Modal } from '~/ui';
|
||||||
import { validateName } from '../../util/validation';
|
import { validateName } from '../../util/validation';
|
||||||
|
|
||||||
export default class EditMeta extends Component {
|
export default class EditMeta extends Component {
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { MenuItem } from 'material-ui';
|
import { MenuItem } from 'material-ui';
|
||||||
|
|
||||||
import { AddressSelect, Form, Input, InputAddressSelect, Select } from '../../../ui';
|
import { AddressSelect, Form, Input, InputAddressSelect, Select } from '~/ui';
|
||||||
|
|
||||||
import styles from '../executeContract.css';
|
import styles from '../executeContract.css';
|
||||||
|
|
||||||
|
@ -20,14 +20,14 @@ import { bindActionCreators } from 'redux';
|
|||||||
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
|
|
||||||
import { BusyStep, CompletedStep, Button, IdentityIcon, Modal, TxHash } from '../../ui';
|
import { BusyStep, CompletedStep, Button, IdentityIcon, Modal, TxHash } from '~/ui';
|
||||||
import { MAX_GAS_ESTIMATION } from '../../util/constants';
|
import { MAX_GAS_ESTIMATION } from '../../util/constants';
|
||||||
import { validateAddress, validateUint } from '../../util/validation';
|
import { validateAddress, validateUint } from '../../util/validation';
|
||||||
|
|
||||||
import DetailsStep from './DetailsStep';
|
import DetailsStep from './DetailsStep';
|
||||||
|
|
||||||
import ERRORS from '../Transfer/errors';
|
import ERRORS from '../Transfer/errors';
|
||||||
import { ERROR_CODES } from '../../api/transport/error';
|
import { ERROR_CODES } from '~/api/transport/error';
|
||||||
|
|
||||||
class ExecuteContract extends Component {
|
class ExecuteContract extends Component {
|
||||||
static contextTypes = {
|
static contextTypes = {
|
||||||
|
@ -18,8 +18,9 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import ActionDone from 'material-ui/svg-icons/action/done';
|
import ActionDone from 'material-ui/svg-icons/action/done';
|
||||||
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
||||||
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
||||||
|
import PrintIcon from 'material-ui/svg-icons/action/print';
|
||||||
|
|
||||||
import { Button, Modal } from '../../ui';
|
import { Button, Modal } from '~/ui';
|
||||||
|
|
||||||
import { NewAccount, AccountDetails } from '../CreateAccount';
|
import { NewAccount, AccountDetails } from '../CreateAccount';
|
||||||
|
|
||||||
@ -27,6 +28,11 @@ import Completed from './Completed';
|
|||||||
import TnC from './TnC';
|
import TnC from './TnC';
|
||||||
import Welcome from './Welcome';
|
import Welcome from './Welcome';
|
||||||
|
|
||||||
|
import { createIdentityImg } from '~/api/util/identity';
|
||||||
|
import print from '../CreateAccount/print';
|
||||||
|
import recoveryPage from '../CreateAccount/recovery-page.ejs';
|
||||||
|
import ParityLogo from '../../../assets/images/parity-logo-black-no-text.svg';
|
||||||
|
|
||||||
const STAGE_NAMES = ['welcome', 'terms', 'new account', 'recovery', 'completed'];
|
const STAGE_NAMES = ['welcome', 'terms', 'new account', 'recovery', 'completed'];
|
||||||
|
|
||||||
export default class FirstRun extends Component {
|
export default class FirstRun extends Component {
|
||||||
@ -107,7 +113,6 @@ export default class FirstRun extends Component {
|
|||||||
|
|
||||||
switch (stage) {
|
switch (stage) {
|
||||||
case 0:
|
case 0:
|
||||||
case 3:
|
|
||||||
return (
|
return (
|
||||||
<Button
|
<Button
|
||||||
icon={ <NavigationArrowForward /> }
|
icon={ <NavigationArrowForward /> }
|
||||||
@ -133,6 +138,20 @@ export default class FirstRun extends Component {
|
|||||||
onClick={ this.onCreate } />
|
onClick={ this.onCreate } />
|
||||||
);
|
);
|
||||||
|
|
||||||
|
case 3:
|
||||||
|
return [
|
||||||
|
<Button
|
||||||
|
icon={ <PrintIcon /> }
|
||||||
|
label='Print Phrase'
|
||||||
|
onClick={ this.printPhrase }
|
||||||
|
/>,
|
||||||
|
<Button
|
||||||
|
icon={ <NavigationArrowForward /> }
|
||||||
|
label='Next'
|
||||||
|
onClick={ this.onNext }
|
||||||
|
/>
|
||||||
|
];
|
||||||
|
|
||||||
case 4:
|
case 4:
|
||||||
return (
|
return (
|
||||||
<Button
|
<Button
|
||||||
@ -205,4 +224,11 @@ export default class FirstRun extends Component {
|
|||||||
|
|
||||||
store.dispatch({ type: 'newError', error });
|
store.dispatch({ type: 'newError', error });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
printPhrase = () => {
|
||||||
|
const { address, phrase, name } = this.state;
|
||||||
|
const identity = createIdentityImg(address);
|
||||||
|
|
||||||
|
print(recoveryPage({ phrase, name, identity, address, logo: ParityLogo }));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -24,7 +24,7 @@ import { List, ListItem, makeSelectable } from 'material-ui/List';
|
|||||||
import { Subheader, IconButton, Tabs, Tab } from 'material-ui';
|
import { Subheader, IconButton, Tabs, Tab } from 'material-ui';
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
|
|
||||||
import { Button, Modal, Editor } from '../../ui';
|
import { Button, Modal, Editor } from '~/ui';
|
||||||
|
|
||||||
import styles from './loadContract.css';
|
import styles from './loadContract.css';
|
||||||
|
|
||||||
|
@ -24,10 +24,10 @@ import Paper from 'material-ui/Paper';
|
|||||||
|
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { bindActionCreators } from 'redux';
|
import { bindActionCreators } from 'redux';
|
||||||
import { showSnackbar } from '../../redux/providers/snackbarActions';
|
import { showSnackbar } from '~/redux/providers/snackbarActions';
|
||||||
|
|
||||||
import Form, { Input } from '../../ui/Form';
|
import Form, { Input } from '~/ui/Form';
|
||||||
import { Button, Modal, IdentityName, IdentityIcon } from '../../ui';
|
import { Button, Modal, IdentityName, IdentityIcon } from '~/ui';
|
||||||
|
|
||||||
import styles from './passwordManager.css';
|
import styles from './passwordManager.css';
|
||||||
|
|
||||||
|
@ -22,8 +22,8 @@ import InfoIcon from 'material-ui/svg-icons/action/info-outline';
|
|||||||
import SuccessIcon from 'material-ui/svg-icons/navigation/check';
|
import SuccessIcon from 'material-ui/svg-icons/navigation/check';
|
||||||
import ErrorIcon from 'material-ui/svg-icons/navigation/close';
|
import ErrorIcon from 'material-ui/svg-icons/navigation/close';
|
||||||
|
|
||||||
import { fromWei } from '../../../api/util/wei';
|
import { fromWei } from '~/api/util/wei';
|
||||||
import { Form, Input } from '../../../ui';
|
import { Form, Input } from '~/ui';
|
||||||
|
|
||||||
import { termsOfService } from '../../../3rdparty/sms-verification';
|
import { termsOfService } from '../../../3rdparty/sms-verification';
|
||||||
import styles from './gatherData.css';
|
import styles from './gatherData.css';
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
|
|
||||||
import { Form, Input } from '../../../ui';
|
import { Form, Input } from '~/ui';
|
||||||
|
|
||||||
export default class QueryCode extends Component {
|
export default class QueryCode extends Component {
|
||||||
static propTypes = {
|
static propTypes = {
|
||||||
|
@ -19,7 +19,7 @@ import { observer } from 'mobx-react';
|
|||||||
import DoneIcon from 'material-ui/svg-icons/action/done-all';
|
import DoneIcon from 'material-ui/svg-icons/action/done-all';
|
||||||
import CancelIcon from 'material-ui/svg-icons/content/clear';
|
import CancelIcon from 'material-ui/svg-icons/content/clear';
|
||||||
|
|
||||||
import { Button, IdentityIcon, Modal } from '../../ui';
|
import { Button, IdentityIcon, Modal } from '~/ui';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
LOADING,
|
LOADING,
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import nullable from '../../../util/nullable-proptype';
|
import nullable from '../../../util/nullable-proptype';
|
||||||
|
|
||||||
import TxHash from '../../../ui/TxHash';
|
import TxHash from '~/ui/TxHash';
|
||||||
import {
|
import {
|
||||||
POSTING_CONFIRMATION, POSTED_CONFIRMATION
|
POSTING_CONFIRMATION, POSTED_CONFIRMATION
|
||||||
} from '../store';
|
} from '../store';
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import nullable from '../../../util/nullable-proptype';
|
import nullable from '../../../util/nullable-proptype';
|
||||||
|
|
||||||
import TxHash from '../../../ui/TxHash';
|
import TxHash from '~/ui/TxHash';
|
||||||
import {
|
import {
|
||||||
POSTING_REQUEST, POSTED_REQUEST, REQUESTING_SMS
|
POSTING_REQUEST, POSTED_REQUEST, REQUESTING_SMS
|
||||||
} from '../store';
|
} from '../store';
|
||||||
|
@ -16,11 +16,11 @@
|
|||||||
|
|
||||||
import { observable, computed, autorun, action } from 'mobx';
|
import { observable, computed, autorun, action } from 'mobx';
|
||||||
import phone from 'phoneformat.js';
|
import phone from 'phoneformat.js';
|
||||||
import { sha3 } from '../../api/util/sha3';
|
import { sha3 } from '~/api/util/sha3';
|
||||||
|
|
||||||
import Contracts from '../../contracts';
|
import Contracts from '~/contracts';
|
||||||
|
|
||||||
import { checkIfVerified, checkIfRequested, awaitPuzzle } from '../../contracts/sms-verification';
|
import { checkIfVerified, checkIfRequested, awaitPuzzle } from '~/contracts/sms-verification';
|
||||||
import { postToServer } from '../../3rdparty/sms-verification';
|
import { postToServer } from '../../3rdparty/sms-verification';
|
||||||
import checkIfTxFailed from '../../util/check-if-tx-failed';
|
import checkIfTxFailed from '../../util/check-if-tx-failed';
|
||||||
import waitForConfirmations from '../../util/wait-for-block-confirmations';
|
import waitForConfirmations from '../../util/wait-for-block-confirmations';
|
||||||
|
@ -19,7 +19,7 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import SaveIcon from 'material-ui/svg-icons/content/save';
|
import SaveIcon from 'material-ui/svg-icons/content/save';
|
||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
|
|
||||||
import { Button, Modal, Editor, Form, Input } from '../../ui';
|
import { Button, Modal, Editor, Form, Input } from '~/ui';
|
||||||
import { ERRORS, validateName } from '../../util/validation';
|
import { ERRORS, validateName } from '../../util/validation';
|
||||||
|
|
||||||
import styles from './saveContract.css';
|
import styles from './saveContract.css';
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { Checkbox, MenuItem } from 'material-ui';
|
import { Checkbox, MenuItem } from 'material-ui';
|
||||||
|
|
||||||
import { Form, Input, Select } from '../../../ui';
|
import { Form, Input, Select } from '~/ui';
|
||||||
|
|
||||||
import Price from '../Price';
|
import Price from '../Price';
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
|
|
||||||
import { Button, IdentityIcon, Modal } from '../../ui';
|
import { Button, IdentityIcon, Modal } from '~/ui';
|
||||||
import initShapeshift from '../../3rdparty/shapeshift';
|
import initShapeshift from '../../3rdparty/shapeshift';
|
||||||
import shapeshiftLogo from '../../../assets/images/shapeshift-logo.png';
|
import shapeshiftLogo from '../../../assets/images/shapeshift-logo.png';
|
||||||
|
|
||||||
|
@ -18,7 +18,9 @@ import BigNumber from 'bignumber.js';
|
|||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { Checkbox, MenuItem } from 'material-ui';
|
import { Checkbox, MenuItem } from 'material-ui';
|
||||||
|
|
||||||
import Form, { Input, InputAddressSelect, Select } from '../../../ui/Form';
|
import { isEqual } from 'lodash';
|
||||||
|
|
||||||
|
import Form, { Input, InputAddressSelect, Select } from '~/ui/Form';
|
||||||
|
|
||||||
import imageUnknown from '../../../../assets/images/contracts/unknown-64x64.png';
|
import imageUnknown from '../../../../assets/images/contracts/unknown-64x64.png';
|
||||||
import styles from '../transfer.css';
|
import styles from '../transfer.css';
|
||||||
@ -29,11 +31,101 @@ const CHECK_STYLE = {
|
|||||||
left: '1em'
|
left: '1em'
|
||||||
};
|
};
|
||||||
|
|
||||||
export default class Details extends Component {
|
class TokenSelect extends Component {
|
||||||
static contextTypes = {
|
static contextTypes = {
|
||||||
api: PropTypes.object
|
api: PropTypes.object
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static propTypes = {
|
||||||
|
onChange: PropTypes.func.isRequired,
|
||||||
|
balance: PropTypes.object.isRequired,
|
||||||
|
images: PropTypes.object.isRequired,
|
||||||
|
tag: PropTypes.string.isRequired
|
||||||
|
};
|
||||||
|
|
||||||
|
componentWillMount () {
|
||||||
|
this.computeTokens();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillReceiveProps (nextProps) {
|
||||||
|
const prevTokens = this.props.balance.tokens.map((t) => `${t.token.tag}_${t.value.toNumber()}`);
|
||||||
|
const nextTokens = nextProps.balance.tokens.map((t) => `${t.token.tag}_${t.value.toNumber()}`);
|
||||||
|
|
||||||
|
if (!isEqual(prevTokens, nextTokens)) {
|
||||||
|
this.computeTokens(nextProps);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
computeTokens (props = this.props) {
|
||||||
|
const { api } = this.context;
|
||||||
|
const { balance, images } = this.props;
|
||||||
|
|
||||||
|
const items = balance.tokens
|
||||||
|
.filter((token, index) => !index || token.value.gt(0))
|
||||||
|
.map((balance, index) => {
|
||||||
|
const token = balance.token;
|
||||||
|
const isEth = index === 0;
|
||||||
|
let imagesrc = token.image;
|
||||||
|
if (!imagesrc) {
|
||||||
|
imagesrc =
|
||||||
|
images[token.address]
|
||||||
|
? `${api.dappsUrl}${images[token.address]}`
|
||||||
|
: imageUnknown;
|
||||||
|
}
|
||||||
|
let value = 0;
|
||||||
|
|
||||||
|
if (isEth) {
|
||||||
|
value = api.util.fromWei(balance.value).toFormat(3);
|
||||||
|
} else {
|
||||||
|
const format = balance.token.format || 1;
|
||||||
|
const decimals = format === 1 ? 0 : Math.min(3, Math.floor(format / 10));
|
||||||
|
value = new BigNumber(balance.value).div(format).toFormat(decimals);
|
||||||
|
}
|
||||||
|
|
||||||
|
const label = (
|
||||||
|
<div className={ styles.token }>
|
||||||
|
<img src={ imagesrc } />
|
||||||
|
<div className={ styles.tokenname }>
|
||||||
|
{ token.name }
|
||||||
|
</div>
|
||||||
|
<div className={ styles.tokenbalance }>
|
||||||
|
{ value }<small> { token.tag }</small>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MenuItem
|
||||||
|
key={ token.tag }
|
||||||
|
value={ token.tag }
|
||||||
|
label={ label }>
|
||||||
|
{ label }
|
||||||
|
</MenuItem>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.setState({ items });
|
||||||
|
}
|
||||||
|
|
||||||
|
render () {
|
||||||
|
const { tag, onChange } = this.props;
|
||||||
|
const { items } = this.state;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Select
|
||||||
|
className={ styles.tokenSelect }
|
||||||
|
label='type of token transfer'
|
||||||
|
hint='type of token to transfer'
|
||||||
|
value={ tag }
|
||||||
|
onChange={ onChange }
|
||||||
|
>
|
||||||
|
{ items }
|
||||||
|
</Select>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default class Details extends Component {
|
||||||
static propTypes = {
|
static propTypes = {
|
||||||
address: PropTypes.string,
|
address: PropTypes.string,
|
||||||
balance: PropTypes.object,
|
balance: PropTypes.object,
|
||||||
@ -115,62 +207,15 @@ export default class Details extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
renderTokenSelect () {
|
renderTokenSelect () {
|
||||||
const { api } = this.context;
|
|
||||||
const { balance, images, tag } = this.props;
|
const { balance, images, tag } = this.props;
|
||||||
|
|
||||||
const items = balance.tokens
|
|
||||||
.filter((token, index) => !index || token.value.gt(0))
|
|
||||||
.map((balance, index) => {
|
|
||||||
const token = balance.token;
|
|
||||||
const isEth = index === 0;
|
|
||||||
let imagesrc = token.image;
|
|
||||||
if (!imagesrc) {
|
|
||||||
imagesrc =
|
|
||||||
images[token.address]
|
|
||||||
? `${api.dappsUrl}${images[token.address]}`
|
|
||||||
: imageUnknown;
|
|
||||||
}
|
|
||||||
let value = 0;
|
|
||||||
|
|
||||||
if (isEth) {
|
|
||||||
value = api.util.fromWei(balance.value).toFormat(3);
|
|
||||||
} else {
|
|
||||||
const format = balance.token.format || 1;
|
|
||||||
const decimals = format === 1 ? 0 : Math.min(3, Math.floor(format / 10));
|
|
||||||
value = new BigNumber(balance.value).div(format).toFormat(decimals);
|
|
||||||
}
|
|
||||||
|
|
||||||
const label = (
|
|
||||||
<div className={ styles.token }>
|
|
||||||
<img src={ imagesrc } />
|
|
||||||
<div className={ styles.tokenname }>
|
|
||||||
{ token.name }
|
|
||||||
</div>
|
|
||||||
<div className={ styles.tokenbalance }>
|
|
||||||
{ value }<small> { token.tag }</small>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<MenuItem
|
|
||||||
key={ token.tag }
|
|
||||||
value={ token.tag }
|
|
||||||
label={ label }>
|
|
||||||
{ label }
|
|
||||||
</MenuItem>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Select
|
<TokenSelect
|
||||||
className={ styles.tokenSelect }
|
balance={ balance }
|
||||||
label='type of token transfer'
|
images={ images }
|
||||||
hint='type of token to transfer'
|
tag={ tag }
|
||||||
value={ tag }
|
onChange={ this.onChangeToken }
|
||||||
onChange={ this.onChangeToken }>
|
/>
|
||||||
{ items }
|
|
||||||
</Select>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
|
|
||||||
import Form, { Input } from '../../../ui/Form';
|
import Form, { Input } from '~/ui/Form';
|
||||||
import GasPriceSelector from '../GasPriceSelector';
|
import GasPriceSelector from '../GasPriceSelector';
|
||||||
|
|
||||||
import styles from '../transfer.css';
|
import styles from '../transfer.css';
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
const ERRORS = {
|
const ERRORS = {
|
||||||
|
requireSender: 'a valid sender is required for the transaction',
|
||||||
requireRecipient: 'a recipient network address is required for the transaction',
|
requireRecipient: 'a recipient network address is required for the transaction',
|
||||||
invalidAddress: 'the supplied address is an invalid network address',
|
invalidAddress: 'the supplied address is an invalid network address',
|
||||||
invalidAmount: 'the supplied amount should be a valid positive number',
|
invalidAmount: 'the supplied amount should be a valid positive number',
|
||||||
|
463
js/src/modals/Transfer/store.js
Normal file
463
js/src/modals/Transfer/store.js
Normal file
@ -0,0 +1,463 @@
|
|||||||
|
// Copyright 2015, 2016 Ethcore (UK) Ltd.
|
||||||
|
// This file is part of Parity.
|
||||||
|
|
||||||
|
// Parity is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
|
||||||
|
// Parity is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import { observable, computed, action, transaction } from 'mobx';
|
||||||
|
import BigNumber from 'bignumber.js';
|
||||||
|
|
||||||
|
import ERRORS from './errors';
|
||||||
|
import { ERROR_CODES } from '~/api/transport/error';
|
||||||
|
import { DEFAULT_GAS, DEFAULT_GASPRICE, MAX_GAS_ESTIMATION } from '../../util/constants';
|
||||||
|
|
||||||
|
const TITLES = {
|
||||||
|
transfer: 'transfer details',
|
||||||
|
sending: 'sending',
|
||||||
|
complete: 'complete',
|
||||||
|
extras: 'extra information',
|
||||||
|
rejected: 'rejected'
|
||||||
|
};
|
||||||
|
const STAGES_BASIC = [TITLES.transfer, TITLES.sending, TITLES.complete];
|
||||||
|
const STAGES_EXTRA = [TITLES.transfer, TITLES.extras, TITLES.sending, TITLES.complete];
|
||||||
|
|
||||||
|
export default class TransferStore {
|
||||||
|
@observable stage = 0;
|
||||||
|
@observable data = '';
|
||||||
|
@observable dataError = null;
|
||||||
|
@observable extras = false;
|
||||||
|
@observable gas = DEFAULT_GAS;
|
||||||
|
@observable gasEst = '0';
|
||||||
|
@observable gasError = null;
|
||||||
|
@observable gasLimitError = null;
|
||||||
|
@observable gasPrice = DEFAULT_GASPRICE;
|
||||||
|
@observable gasPriceError = null;
|
||||||
|
@observable recipient = '';
|
||||||
|
@observable recipientError = ERRORS.requireRecipient;
|
||||||
|
@observable sending = false;
|
||||||
|
@observable tag = 'ETH';
|
||||||
|
@observable total = '0.0';
|
||||||
|
@observable totalError = null;
|
||||||
|
@observable value = '0.0';
|
||||||
|
@observable valueAll = false;
|
||||||
|
@observable valueError = null;
|
||||||
|
@observable isEth = true;
|
||||||
|
@observable busyState = null;
|
||||||
|
@observable rejected = false;
|
||||||
|
|
||||||
|
gasPriceHistogram = {};
|
||||||
|
|
||||||
|
account = null;
|
||||||
|
balance = null;
|
||||||
|
gasLimit = null;
|
||||||
|
onClose = null;
|
||||||
|
|
||||||
|
@computed get steps () {
|
||||||
|
const steps = [].concat(this.extras ? STAGES_EXTRA : STAGES_BASIC);
|
||||||
|
|
||||||
|
if (this.rejected) {
|
||||||
|
steps[steps.length - 1] = TITLES.rejected;
|
||||||
|
}
|
||||||
|
|
||||||
|
return steps;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get isValid () {
|
||||||
|
const detailsValid = !this.recipientError && !this.valueError && !this.totalError;
|
||||||
|
const extrasValid = !this.gasError && !this.gasPriceError && !this.totalError;
|
||||||
|
const verifyValid = !this.passwordError;
|
||||||
|
|
||||||
|
switch (this.stage) {
|
||||||
|
case 0:
|
||||||
|
return detailsValid;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
return this.extras ? extrasValid : verifyValid;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
return verifyValid;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor (api, props) {
|
||||||
|
this.api = api;
|
||||||
|
|
||||||
|
const { account, balance, gasLimit, onClose } = props;
|
||||||
|
|
||||||
|
this.account = account;
|
||||||
|
this.balance = balance;
|
||||||
|
this.gasLimit = gasLimit;
|
||||||
|
this.onClose = onClose;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action onNext = () => {
|
||||||
|
this.stage += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action onPrev = () => {
|
||||||
|
this.stage -= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action onClose = () => {
|
||||||
|
this.onClose && this.onClose();
|
||||||
|
this.stage = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action onUpdateDetails = (type, value) => {
|
||||||
|
switch (type) {
|
||||||
|
case 'all':
|
||||||
|
return this._onUpdateAll(value);
|
||||||
|
|
||||||
|
case 'extras':
|
||||||
|
return this._onUpdateExtras(value);
|
||||||
|
|
||||||
|
case 'data':
|
||||||
|
return this._onUpdateData(value);
|
||||||
|
|
||||||
|
case 'gas':
|
||||||
|
return this._onUpdateGas(value);
|
||||||
|
|
||||||
|
case 'gasPrice':
|
||||||
|
return this._onUpdateGasPrice(value);
|
||||||
|
|
||||||
|
case 'recipient':
|
||||||
|
return this._onUpdateRecipient(value);
|
||||||
|
|
||||||
|
case 'tag':
|
||||||
|
return this._onUpdateTag(value);
|
||||||
|
|
||||||
|
case 'value':
|
||||||
|
return this._onUpdateValue(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@action getDefaults = () => {
|
||||||
|
Promise
|
||||||
|
.all([
|
||||||
|
this.api.parity.gasPriceHistogram(),
|
||||||
|
this.api.eth.gasPrice()
|
||||||
|
])
|
||||||
|
.then(([gasPriceHistogram, gasPrice]) => {
|
||||||
|
transaction(() => {
|
||||||
|
this.gasPrice = gasPrice.toString();
|
||||||
|
this.gasPriceDefault = gasPrice.toFormat();
|
||||||
|
this.gasPriceHistogram = gasPriceHistogram;
|
||||||
|
|
||||||
|
this.recalculate();
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.warn('getDefaults', error);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action onSend = () => {
|
||||||
|
this.onNext();
|
||||||
|
this.sending = true;
|
||||||
|
|
||||||
|
const promise = this.isEth ? this._sendEth() : this._sendToken();
|
||||||
|
|
||||||
|
promise
|
||||||
|
.then((requestId) => {
|
||||||
|
this.busyState = 'Waiting for authorization in the Parity Signer';
|
||||||
|
|
||||||
|
return this.api
|
||||||
|
.pollMethod('parity_checkRequest', requestId)
|
||||||
|
.catch((e) => {
|
||||||
|
if (e.code === ERROR_CODES.REQUEST_REJECTED) {
|
||||||
|
this.rejected = true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw e;
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.then((txhash) => {
|
||||||
|
transaction(() => {
|
||||||
|
this.onNext();
|
||||||
|
|
||||||
|
this.sending = false;
|
||||||
|
this.txhash = txhash;
|
||||||
|
this.busyState = 'Your transaction has been posted to the network';
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
this.sending = false;
|
||||||
|
this.newError(error);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateAll = (valueAll) => {
|
||||||
|
this.valueAll = valueAll;
|
||||||
|
this.recalculateGas();
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateExtras = (extras) => {
|
||||||
|
this.extras = extras;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateData = (data) => {
|
||||||
|
this.data = data;
|
||||||
|
this.recalculateGas();
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateGas = (gas) => {
|
||||||
|
const gasError = this._validatePositiveNumber(gas);
|
||||||
|
|
||||||
|
transaction(() => {
|
||||||
|
this.gas = gas;
|
||||||
|
this.gasError = gasError;
|
||||||
|
|
||||||
|
this.recalculate();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateGasPrice = (gasPrice) => {
|
||||||
|
const gasPriceError = this._validatePositiveNumber(gasPrice);
|
||||||
|
|
||||||
|
transaction(() => {
|
||||||
|
this.gasPrice = gasPrice;
|
||||||
|
this.gasPriceError = gasPriceError;
|
||||||
|
|
||||||
|
this.recalculate();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateRecipient = (recipient) => {
|
||||||
|
let recipientError = null;
|
||||||
|
|
||||||
|
if (!recipient || !recipient.length) {
|
||||||
|
recipientError = ERRORS.requireRecipient;
|
||||||
|
} else if (!this.api.util.isAddressValid(recipient)) {
|
||||||
|
recipientError = ERRORS.invalidAddress;
|
||||||
|
}
|
||||||
|
|
||||||
|
transaction(() => {
|
||||||
|
this.recipient = recipient;
|
||||||
|
this.recipientError = recipientError;
|
||||||
|
|
||||||
|
this.recalculateGas();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateTag = (tag) => {
|
||||||
|
transaction(() => {
|
||||||
|
this.tag = tag;
|
||||||
|
this.isEth = tag.toLowerCase().trim() === 'eth';
|
||||||
|
|
||||||
|
this.recalculateGas();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action _onUpdateValue = (value) => {
|
||||||
|
let valueError = this._validatePositiveNumber(value);
|
||||||
|
|
||||||
|
if (!valueError) {
|
||||||
|
valueError = this._validateDecimals(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
transaction(() => {
|
||||||
|
this.value = value;
|
||||||
|
this.valueError = valueError;
|
||||||
|
|
||||||
|
this.recalculateGas();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action recalculateGas = () => {
|
||||||
|
if (!this.isValid) {
|
||||||
|
this.gas = 0;
|
||||||
|
return this.recalculate();
|
||||||
|
}
|
||||||
|
|
||||||
|
const promise = this.isEth ? this._estimateGasEth() : this._estimateGasToken();
|
||||||
|
|
||||||
|
promise
|
||||||
|
.then((gasEst) => {
|
||||||
|
let gas = gasEst;
|
||||||
|
let gasLimitError = null;
|
||||||
|
|
||||||
|
if (gas.gt(DEFAULT_GAS)) {
|
||||||
|
gas = gas.mul(1.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gas.gte(MAX_GAS_ESTIMATION)) {
|
||||||
|
gasLimitError = ERRORS.gasException;
|
||||||
|
} else if (gas.gt(this.gasLimit)) {
|
||||||
|
gasLimitError = ERRORS.gasBlockLimit;
|
||||||
|
}
|
||||||
|
|
||||||
|
transaction(() => {
|
||||||
|
this.gas = gas.toFixed(0);
|
||||||
|
this.gasEst = gasEst.toFormat();
|
||||||
|
this.gasLimitError = gasLimitError;
|
||||||
|
|
||||||
|
this.recalculate();
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error('etimateGas', error);
|
||||||
|
this.recalculate();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@action recalculate = () => {
|
||||||
|
const { account, balance } = this;
|
||||||
|
|
||||||
|
if (!account || !balance) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { gas, gasPrice, tag, valueAll, isEth } = this;
|
||||||
|
|
||||||
|
const gasTotal = new BigNumber(gasPrice || 0).mul(new BigNumber(gas || 0));
|
||||||
|
const balance_ = balance.tokens.find((b) => tag === b.token.tag);
|
||||||
|
const availableEth = new BigNumber(balance.tokens[0].value);
|
||||||
|
const available = new BigNumber(balance_.value);
|
||||||
|
const format = new BigNumber(balance_.token.format || 1);
|
||||||
|
|
||||||
|
let { value, valueError } = this;
|
||||||
|
let totalEth = gasTotal;
|
||||||
|
let totalError = null;
|
||||||
|
|
||||||
|
if (valueAll) {
|
||||||
|
if (isEth) {
|
||||||
|
const bn = this.api.util.fromWei(availableEth.minus(gasTotal));
|
||||||
|
value = (bn.lt(0) ? new BigNumber(0.0) : bn).toString();
|
||||||
|
} else {
|
||||||
|
value = available.div(format).toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isEth) {
|
||||||
|
totalEth = totalEth.plus(this.api.util.toWei(value || 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (new BigNumber(value || 0).gt(available.div(format))) {
|
||||||
|
valueError = ERRORS.largeAmount;
|
||||||
|
} else if (valueError === ERRORS.largeAmount) {
|
||||||
|
valueError = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (totalEth.gt(availableEth)) {
|
||||||
|
totalError = ERRORS.largeAmount;
|
||||||
|
}
|
||||||
|
|
||||||
|
transaction(() => {
|
||||||
|
this.total = this.api.util.fromWei(totalEth).toString();
|
||||||
|
this.totalError = totalError;
|
||||||
|
this.value = value;
|
||||||
|
this.valueError = valueError;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_sendEth () {
|
||||||
|
const { account, data, gas, gasPrice, recipient, value } = this;
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
from: account.address,
|
||||||
|
to: recipient,
|
||||||
|
gas,
|
||||||
|
gasPrice,
|
||||||
|
value: this.api.util.toWei(value || 0)
|
||||||
|
};
|
||||||
|
|
||||||
|
if (data && data.length) {
|
||||||
|
options.data = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.api.parity.postTransaction(options);
|
||||||
|
}
|
||||||
|
|
||||||
|
_sendToken () {
|
||||||
|
const { account, balance } = this;
|
||||||
|
const { gas, gasPrice, recipient, value, tag } = this;
|
||||||
|
|
||||||
|
const token = balance.tokens.find((balance) => balance.token.tag === tag).token;
|
||||||
|
|
||||||
|
return token.contract.instance.transfer
|
||||||
|
.postTransaction({
|
||||||
|
from: account.address,
|
||||||
|
to: token.address,
|
||||||
|
gas,
|
||||||
|
gasPrice
|
||||||
|
}, [
|
||||||
|
recipient,
|
||||||
|
new BigNumber(value).mul(token.format).toFixed(0)
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
_estimateGasToken () {
|
||||||
|
const { account, balance } = this;
|
||||||
|
const { recipient, value, tag } = this;
|
||||||
|
|
||||||
|
const token = balance.tokens.find((balance) => balance.token.tag === tag).token;
|
||||||
|
|
||||||
|
return token.contract.instance.transfer
|
||||||
|
.estimateGas({
|
||||||
|
gas: MAX_GAS_ESTIMATION,
|
||||||
|
from: account.address,
|
||||||
|
to: token.address
|
||||||
|
}, [
|
||||||
|
recipient,
|
||||||
|
new BigNumber(value || 0).mul(token.format).toFixed(0)
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
_estimateGasEth () {
|
||||||
|
const { account, data, recipient, value } = this;
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
gas: MAX_GAS_ESTIMATION,
|
||||||
|
from: account.address,
|
||||||
|
to: recipient,
|
||||||
|
value: this.api.util.toWei(value || 0)
|
||||||
|
};
|
||||||
|
|
||||||
|
if (data && data.length) {
|
||||||
|
options.data = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.api.eth.estimateGas(options);
|
||||||
|
}
|
||||||
|
|
||||||
|
_validatePositiveNumber (num) {
|
||||||
|
try {
|
||||||
|
const v = new BigNumber(num);
|
||||||
|
if (v.lt(0)) {
|
||||||
|
return ERRORS.invalidAmount;
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
return ERRORS.invalidAmount;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
_validateDecimals (num) {
|
||||||
|
const { balance } = this;
|
||||||
|
|
||||||
|
if (this.tag === 'ETH') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const token = balance.tokens.find((balance) => balance.token.tag === this.tag).token;
|
||||||
|
const s = new BigNumber(num).mul(token.format || 1).toFixed();
|
||||||
|
|
||||||
|
if (s.indexOf('.') !== -1) {
|
||||||
|
return ERRORS.invalidDecimals;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
@ -14,88 +14,50 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import BigNumber from 'bignumber.js';
|
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { bindActionCreators } from 'redux';
|
import { bindActionCreators } from 'redux';
|
||||||
|
import { observer } from 'mobx-react';
|
||||||
|
|
||||||
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
import ActionDoneAll from 'material-ui/svg-icons/action/done-all';
|
||||||
import ContentClear from 'material-ui/svg-icons/content/clear';
|
import ContentClear from 'material-ui/svg-icons/content/clear';
|
||||||
import NavigationArrowBack from 'material-ui/svg-icons/navigation/arrow-back';
|
import NavigationArrowBack from 'material-ui/svg-icons/navigation/arrow-back';
|
||||||
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
import NavigationArrowForward from 'material-ui/svg-icons/navigation/arrow-forward';
|
||||||
|
|
||||||
import { BusyStep, CompletedStep, Button, IdentityIcon, Modal, TxHash } from '../../ui';
|
import { newError } from '~/ui/Errors/actions';
|
||||||
import { DEFAULT_GAS, DEFAULT_GASPRICE, MAX_GAS_ESTIMATION } from '../../util/constants';
|
import { BusyStep, CompletedStep, Button, IdentityIcon, Modal, TxHash } from '~/ui';
|
||||||
|
|
||||||
import Details from './Details';
|
import Details from './Details';
|
||||||
import Extras from './Extras';
|
import Extras from './Extras';
|
||||||
import ERRORS from './errors';
|
|
||||||
|
import TransferStore from './store';
|
||||||
import styles from './transfer.css';
|
import styles from './transfer.css';
|
||||||
|
|
||||||
import { ERROR_CODES } from '../../api/transport/error';
|
@observer
|
||||||
|
|
||||||
const TITLES = {
|
|
||||||
transfer: 'transfer details',
|
|
||||||
sending: 'sending',
|
|
||||||
complete: 'complete',
|
|
||||||
extras: 'extra information',
|
|
||||||
rejected: 'rejected'
|
|
||||||
};
|
|
||||||
const STAGES_BASIC = [TITLES.transfer, TITLES.sending, TITLES.complete];
|
|
||||||
const STAGES_EXTRA = [TITLES.transfer, TITLES.extras, TITLES.sending, TITLES.complete];
|
|
||||||
|
|
||||||
class Transfer extends Component {
|
class Transfer extends Component {
|
||||||
static contextTypes = {
|
static contextTypes = {
|
||||||
api: PropTypes.object.isRequired,
|
api: PropTypes.object.isRequired
|
||||||
store: PropTypes.object.isRequired
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static propTypes = {
|
static propTypes = {
|
||||||
|
newError: PropTypes.func.isRequired,
|
||||||
|
gasLimit: PropTypes.object.isRequired,
|
||||||
|
images: PropTypes.object.isRequired,
|
||||||
|
|
||||||
account: PropTypes.object,
|
account: PropTypes.object,
|
||||||
balance: PropTypes.object,
|
balance: PropTypes.object,
|
||||||
balances: PropTypes.object,
|
balances: PropTypes.object,
|
||||||
gasLimit: PropTypes.object.isRequired,
|
|
||||||
images: PropTypes.object.isRequired,
|
|
||||||
onClose: PropTypes.func
|
onClose: PropTypes.func
|
||||||
}
|
}
|
||||||
|
|
||||||
state = {
|
store = new TransferStore(this.context.api, this.props);
|
||||||
stage: 0,
|
|
||||||
data: '',
|
|
||||||
dataError: null,
|
|
||||||
extras: false,
|
|
||||||
gas: DEFAULT_GAS,
|
|
||||||
gasEst: '0',
|
|
||||||
gasError: null,
|
|
||||||
gasLimitError: null,
|
|
||||||
gasPrice: DEFAULT_GASPRICE,
|
|
||||||
gasPriceHistogram: {},
|
|
||||||
gasPriceError: null,
|
|
||||||
recipient: '',
|
|
||||||
recipientError: ERRORS.requireRecipient,
|
|
||||||
sending: false,
|
|
||||||
tag: 'ETH',
|
|
||||||
total: '0.0',
|
|
||||||
totalError: null,
|
|
||||||
value: '0.0',
|
|
||||||
valueAll: false,
|
|
||||||
valueError: null,
|
|
||||||
isEth: true,
|
|
||||||
busyState: null,
|
|
||||||
rejected: false
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount () {
|
componentDidMount () {
|
||||||
this.getDefaults();
|
this.store.getDefaults();
|
||||||
}
|
}
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
const { stage, extras, rejected } = this.state;
|
const { stage, extras, steps } = this.store;
|
||||||
|
|
||||||
const steps = [].concat(extras ? STAGES_EXTRA : STAGES_BASIC);
|
|
||||||
|
|
||||||
if (rejected) {
|
|
||||||
steps[steps.length - 1] = TITLES.rejected;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
@ -134,7 +96,7 @@ class Transfer extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
renderPage () {
|
renderPage () {
|
||||||
const { extras, stage } = this.state;
|
const { extras, stage } = this.store;
|
||||||
|
|
||||||
if (stage === 0) {
|
if (stage === 0) {
|
||||||
return this.renderDetailsPage();
|
return this.renderDetailsPage();
|
||||||
@ -146,7 +108,7 @@ class Transfer extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
renderCompletePage () {
|
renderCompletePage () {
|
||||||
const { sending, txhash, busyState, rejected } = this.state;
|
const { sending, txhash, busyState, rejected } = this.store;
|
||||||
|
|
||||||
if (rejected) {
|
if (rejected) {
|
||||||
return (
|
return (
|
||||||
@ -174,83 +136,88 @@ class Transfer extends Component {
|
|||||||
|
|
||||||
renderDetailsPage () {
|
renderDetailsPage () {
|
||||||
const { account, balance, images } = this.props;
|
const { account, balance, images } = this.props;
|
||||||
|
const { valueAll, extras, recipient, recipientError, tag } = this.store;
|
||||||
|
const { total, totalError, value, valueError } = this.store;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Details
|
<Details
|
||||||
address={ account.address }
|
address={ account.address }
|
||||||
all={ this.state.valueAll }
|
all={ valueAll }
|
||||||
balance={ balance }
|
balance={ balance }
|
||||||
extras={ this.state.extras }
|
extras={ extras }
|
||||||
images={ images }
|
images={ images }
|
||||||
recipient={ this.state.recipient }
|
recipient={ recipient }
|
||||||
recipientError={ this.state.recipientError }
|
recipientError={ recipientError }
|
||||||
tag={ this.state.tag }
|
tag={ tag }
|
||||||
total={ this.state.total }
|
total={ total }
|
||||||
totalError={ this.state.totalError }
|
totalError={ totalError }
|
||||||
value={ this.state.value }
|
value={ value }
|
||||||
valueError={ this.state.valueError }
|
valueError={ valueError }
|
||||||
onChange={ this.onUpdateDetails } />
|
onChange={ this.store.onUpdateDetails } />
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
renderExtrasPage () {
|
renderExtrasPage () {
|
||||||
if (!this.state.gasPriceHistogram) {
|
if (!this.store.gasPriceHistogram) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const { isEth, data, dataError, gas, gasEst, gasError, gasPrice } = this.store;
|
||||||
|
const { gasPriceDefault, gasPriceError, gasPriceHistogram, total, totalError } = this.store;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Extras
|
<Extras
|
||||||
isEth={ this.state.isEth }
|
isEth={ isEth }
|
||||||
data={ this.state.data }
|
data={ data }
|
||||||
dataError={ this.state.dataError }
|
dataError={ dataError }
|
||||||
gas={ this.state.gas }
|
gas={ gas }
|
||||||
gasEst={ this.state.gasEst }
|
gasEst={ gasEst }
|
||||||
gasError={ this.state.gasError }
|
gasError={ gasError }
|
||||||
gasPrice={ this.state.gasPrice }
|
gasPrice={ gasPrice }
|
||||||
gasPriceDefault={ this.state.gasPriceDefault }
|
gasPriceDefault={ gasPriceDefault }
|
||||||
gasPriceError={ this.state.gasPriceError }
|
gasPriceError={ gasPriceError }
|
||||||
gasPriceHistogram={ this.state.gasPriceHistogram }
|
gasPriceHistogram={ gasPriceHistogram }
|
||||||
total={ this.state.total }
|
total={ total }
|
||||||
totalError={ this.state.totalError }
|
totalError={ totalError }
|
||||||
onChange={ this.onUpdateDetails } />
|
onChange={ this.store.onUpdateDetails } />
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
renderDialogActions () {
|
renderDialogActions () {
|
||||||
const { account } = this.props;
|
const { account } = this.props;
|
||||||
const { extras, sending, stage } = this.state;
|
const { extras, sending, stage } = this.store;
|
||||||
|
|
||||||
const cancelBtn = (
|
const cancelBtn = (
|
||||||
<Button
|
<Button
|
||||||
icon={ <ContentClear /> }
|
icon={ <ContentClear /> }
|
||||||
label='Cancel'
|
label='Cancel'
|
||||||
onClick={ this.onClose } />
|
onClick={ this.store.onClose } />
|
||||||
);
|
);
|
||||||
const nextBtn = (
|
const nextBtn = (
|
||||||
<Button
|
<Button
|
||||||
disabled={ !this.isValid() }
|
disabled={ !this.store.isValid }
|
||||||
icon={ <NavigationArrowForward /> }
|
icon={ <NavigationArrowForward /> }
|
||||||
label='Next'
|
label='Next'
|
||||||
onClick={ this.onNext } />
|
onClick={ this.store.onNext } />
|
||||||
);
|
);
|
||||||
const prevBtn = (
|
const prevBtn = (
|
||||||
<Button
|
<Button
|
||||||
icon={ <NavigationArrowBack /> }
|
icon={ <NavigationArrowBack /> }
|
||||||
label='Back'
|
label='Back'
|
||||||
onClick={ this.onPrev } />
|
onClick={ this.store.onPrev } />
|
||||||
);
|
);
|
||||||
const sendBtn = (
|
const sendBtn = (
|
||||||
<Button
|
<Button
|
||||||
disabled={ !this.isValid() || sending }
|
disabled={ !this.store.isValid || sending }
|
||||||
icon={ <IdentityIcon address={ account.address } button /> }
|
icon={ <IdentityIcon address={ account.address } button /> }
|
||||||
label='Send'
|
label='Send'
|
||||||
onClick={ this.onSend } />
|
onClick={ this.store.onSend } />
|
||||||
);
|
);
|
||||||
const doneBtn = (
|
const doneBtn = (
|
||||||
<Button
|
<Button
|
||||||
icon={ <ActionDoneAll /> }
|
icon={ <ActionDoneAll /> }
|
||||||
label='Close'
|
label='Close'
|
||||||
onClick={ this.onClose } />
|
onClick={ this.store.onClose } />
|
||||||
);
|
);
|
||||||
|
|
||||||
switch (stage) {
|
switch (stage) {
|
||||||
@ -268,7 +235,7 @@ class Transfer extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
renderWarning () {
|
renderWarning () {
|
||||||
const { gasLimitError } = this.state;
|
const { gasLimitError } = this.store;
|
||||||
|
|
||||||
if (!gasLimitError) {
|
if (!gasLimitError) {
|
||||||
return null;
|
return null;
|
||||||
@ -280,413 +247,17 @@ class Transfer extends Component {
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
isValid () {
|
|
||||||
const detailsValid = !this.state.recipientError && !this.state.valueError && !this.state.totalError;
|
|
||||||
const extrasValid = !this.state.gasError && !this.state.gasPriceError && !this.state.totalError;
|
|
||||||
const verifyValid = !this.state.passwordError;
|
|
||||||
|
|
||||||
switch (this.state.stage) {
|
|
||||||
case 0:
|
|
||||||
return detailsValid;
|
|
||||||
|
|
||||||
case 1:
|
|
||||||
return this.state.extras ? extrasValid : verifyValid;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
return verifyValid;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
onNext = () => {
|
|
||||||
this.setState({
|
|
||||||
stage: this.state.stage + 1
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
onPrev = () => {
|
|
||||||
this.setState({
|
|
||||||
stage: this.state.stage - 1
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateAll (valueAll) {
|
|
||||||
this.setState({
|
|
||||||
valueAll
|
|
||||||
}, this.recalculateGas);
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateExtras (extras) {
|
|
||||||
this.setState({
|
|
||||||
extras
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateData (data) {
|
|
||||||
this.setState({
|
|
||||||
data
|
|
||||||
}, this.recalculateGas);
|
|
||||||
}
|
|
||||||
|
|
||||||
validatePositiveNumber (num) {
|
|
||||||
try {
|
|
||||||
const v = new BigNumber(num);
|
|
||||||
if (v.lt(0)) {
|
|
||||||
return ERRORS.invalidAmount;
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
return ERRORS.invalidAmount;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
validateDecimals (num) {
|
|
||||||
const { balance } = this.props;
|
|
||||||
const { tag } = this.state;
|
|
||||||
|
|
||||||
if (tag === 'ETH') {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const token = balance.tokens.find((balance) => balance.token.tag === tag).token;
|
|
||||||
const s = new BigNumber(num).mul(token.format || 1).toFixed();
|
|
||||||
|
|
||||||
if (s.indexOf('.') !== -1) {
|
|
||||||
return ERRORS.invalidDecimals;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateGas (gas) {
|
|
||||||
const gasError = this.validatePositiveNumber(gas);
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
gas,
|
|
||||||
gasError
|
|
||||||
}, this.recalculate);
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateGasPrice (gasPrice) {
|
|
||||||
const gasPriceError = this.validatePositiveNumber(gasPrice);
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
gasPrice,
|
|
||||||
gasPriceError
|
|
||||||
}, this.recalculate);
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateRecipient (recipient) {
|
|
||||||
const { api } = this.context;
|
|
||||||
let recipientError = null;
|
|
||||||
|
|
||||||
if (!recipient || !recipient.length) {
|
|
||||||
recipientError = ERRORS.requireRecipient;
|
|
||||||
} else if (!api.util.isAddressValid(recipient)) {
|
|
||||||
recipientError = ERRORS.invalidAddress;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
recipient,
|
|
||||||
recipientError
|
|
||||||
}, this.recalculateGas);
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateTag (tag) {
|
|
||||||
const { balance } = this.props;
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
tag,
|
|
||||||
isEth: tag === balance.tokens[0].token.tag
|
|
||||||
}, this.recalculateGas);
|
|
||||||
}
|
|
||||||
|
|
||||||
_onUpdateValue (value) {
|
|
||||||
let valueError = this.validatePositiveNumber(value);
|
|
||||||
|
|
||||||
if (!valueError) {
|
|
||||||
valueError = this.validateDecimals(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
value,
|
|
||||||
valueError
|
|
||||||
}, this.recalculateGas);
|
|
||||||
}
|
|
||||||
|
|
||||||
onUpdateDetails = (type, value) => {
|
|
||||||
switch (type) {
|
|
||||||
case 'all':
|
|
||||||
return this._onUpdateAll(value);
|
|
||||||
|
|
||||||
case 'extras':
|
|
||||||
return this._onUpdateExtras(value);
|
|
||||||
|
|
||||||
case 'data':
|
|
||||||
return this._onUpdateData(value);
|
|
||||||
|
|
||||||
case 'gas':
|
|
||||||
return this._onUpdateGas(value);
|
|
||||||
|
|
||||||
case 'gasPrice':
|
|
||||||
return this._onUpdateGasPrice(value);
|
|
||||||
|
|
||||||
case 'recipient':
|
|
||||||
return this._onUpdateRecipient(value);
|
|
||||||
|
|
||||||
case 'tag':
|
|
||||||
return this._onUpdateTag(value);
|
|
||||||
|
|
||||||
case 'value':
|
|
||||||
return this._onUpdateValue(value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_sendEth () {
|
|
||||||
const { api } = this.context;
|
|
||||||
const { account } = this.props;
|
|
||||||
const { data, gas, gasPrice, recipient, value } = this.state;
|
|
||||||
|
|
||||||
const options = {
|
|
||||||
from: account.address,
|
|
||||||
to: recipient,
|
|
||||||
gas,
|
|
||||||
gasPrice,
|
|
||||||
value: api.util.toWei(value || 0)
|
|
||||||
};
|
|
||||||
|
|
||||||
if (data && data.length) {
|
|
||||||
options.data = data;
|
|
||||||
}
|
|
||||||
|
|
||||||
return api.parity.postTransaction(options);
|
|
||||||
}
|
|
||||||
|
|
||||||
_sendToken () {
|
|
||||||
const { account, balance } = this.props;
|
|
||||||
const { gas, gasPrice, recipient, value, tag } = this.state;
|
|
||||||
const token = balance.tokens.find((balance) => balance.token.tag === tag).token;
|
|
||||||
|
|
||||||
return token.contract.instance.transfer
|
|
||||||
.postTransaction({
|
|
||||||
from: account.address,
|
|
||||||
to: token.address,
|
|
||||||
gas,
|
|
||||||
gasPrice
|
|
||||||
}, [
|
|
||||||
recipient,
|
|
||||||
new BigNumber(value).mul(token.format).toFixed(0)
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
onSend = () => {
|
|
||||||
const { api } = this.context;
|
|
||||||
|
|
||||||
this.onNext();
|
|
||||||
|
|
||||||
this.setState({ sending: true }, () => {
|
|
||||||
(this.state.isEth
|
|
||||||
? this._sendEth()
|
|
||||||
: this._sendToken()
|
|
||||||
).then((requestId) => {
|
|
||||||
this.setState({ busyState: 'Waiting for authorization in the Parity Signer' });
|
|
||||||
|
|
||||||
return api
|
|
||||||
.pollMethod('parity_checkRequest', requestId)
|
|
||||||
.catch((e) => {
|
|
||||||
if (e.code === ERROR_CODES.REQUEST_REJECTED) {
|
|
||||||
this.setState({ rejected: true });
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
throw e;
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.then((txhash) => {
|
|
||||||
this.onNext();
|
|
||||||
this.setState({
|
|
||||||
sending: false,
|
|
||||||
txhash,
|
|
||||||
busyState: 'Your transaction has been posted to the network'
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.log('send', error);
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
sending: false
|
|
||||||
});
|
|
||||||
|
|
||||||
this.newError(error);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
onClose = () => {
|
|
||||||
this.setState({ stage: 0 }, () => {
|
|
||||||
this.props.onClose && this.props.onClose();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_estimateGasToken () {
|
|
||||||
const { account, balance } = this.props;
|
|
||||||
const { recipient, value, tag } = this.state;
|
|
||||||
const token = balance.tokens.find((balance) => balance.token.tag === tag).token;
|
|
||||||
|
|
||||||
return token.contract.instance.transfer
|
|
||||||
.estimateGas({
|
|
||||||
gas: MAX_GAS_ESTIMATION,
|
|
||||||
from: account.address,
|
|
||||||
to: token.address
|
|
||||||
}, [
|
|
||||||
recipient,
|
|
||||||
new BigNumber(value || 0).mul(token.format).toFixed(0)
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
_estimateGasEth () {
|
|
||||||
const { api } = this.context;
|
|
||||||
const { account } = this.props;
|
|
||||||
const { data, recipient, value } = this.state;
|
|
||||||
const options = {
|
|
||||||
gas: MAX_GAS_ESTIMATION,
|
|
||||||
from: account.address,
|
|
||||||
to: recipient,
|
|
||||||
value: api.util.toWei(value || 0)
|
|
||||||
};
|
|
||||||
|
|
||||||
if (data && data.length) {
|
|
||||||
options.data = data;
|
|
||||||
}
|
|
||||||
|
|
||||||
return api.eth.estimateGas(options);
|
|
||||||
}
|
|
||||||
|
|
||||||
recalculateGas = () => {
|
|
||||||
if (!this.isValid()) {
|
|
||||||
this.setState({
|
|
||||||
gas: '0'
|
|
||||||
}, this.recalculate);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { gasLimit } = this.props;
|
|
||||||
|
|
||||||
(this.state.isEth
|
|
||||||
? this._estimateGasEth()
|
|
||||||
: this._estimateGasToken()
|
|
||||||
).then((gasEst) => {
|
|
||||||
let gas = gasEst;
|
|
||||||
let gasLimitError = null;
|
|
||||||
|
|
||||||
if (gas.gt(DEFAULT_GAS)) {
|
|
||||||
gas = gas.mul(1.2);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (gas.gte(MAX_GAS_ESTIMATION)) {
|
|
||||||
gasLimitError = ERRORS.gasException;
|
|
||||||
} else if (gas.gt(gasLimit)) {
|
|
||||||
gasLimitError = ERRORS.gasBlockLimit;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
gas: gas.toFixed(0),
|
|
||||||
gasEst: gasEst.toFormat(),
|
|
||||||
gasLimitError
|
|
||||||
}, this.recalculate);
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('etimateGas', error);
|
|
||||||
this.recalculate();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
recalculate = () => {
|
|
||||||
const { api } = this.context;
|
|
||||||
const { account, balance } = this.props;
|
|
||||||
|
|
||||||
if (!account || !balance) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { gas, gasPrice, tag, valueAll, isEth } = this.state;
|
|
||||||
const gasTotal = new BigNumber(gasPrice || 0).mul(new BigNumber(gas || 0));
|
|
||||||
const balance_ = balance.tokens.find((b) => tag === b.token.tag);
|
|
||||||
const availableEth = new BigNumber(balance.tokens[0].value);
|
|
||||||
const available = new BigNumber(balance_.value);
|
|
||||||
const format = new BigNumber(balance_.token.format || 1);
|
|
||||||
|
|
||||||
let { value, valueError } = this.state;
|
|
||||||
let totalEth = gasTotal;
|
|
||||||
let totalError = null;
|
|
||||||
|
|
||||||
if (valueAll) {
|
|
||||||
if (isEth) {
|
|
||||||
const bn = api.util.fromWei(availableEth.minus(gasTotal));
|
|
||||||
value = (bn.lt(0) ? new BigNumber(0.0) : bn).toString();
|
|
||||||
} else {
|
|
||||||
value = available.div(format).toString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isEth) {
|
|
||||||
totalEth = totalEth.plus(api.util.toWei(value || 0));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (new BigNumber(value || 0).gt(available.div(format))) {
|
|
||||||
valueError = ERRORS.largeAmount;
|
|
||||||
} else if (valueError === ERRORS.largeAmount) {
|
|
||||||
valueError = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (totalEth.gt(availableEth)) {
|
|
||||||
totalError = ERRORS.largeAmount;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
total: api.util.fromWei(totalEth).toString(),
|
|
||||||
totalError,
|
|
||||||
value,
|
|
||||||
valueError
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
getDefaults = () => {
|
|
||||||
const { api } = this.context;
|
|
||||||
|
|
||||||
Promise
|
|
||||||
.all([
|
|
||||||
api.parity.gasPriceHistogram(),
|
|
||||||
api.eth.gasPrice()
|
|
||||||
])
|
|
||||||
.then(([gasPriceHistogram, gasPrice]) => {
|
|
||||||
this.setState({
|
|
||||||
gasPrice: gasPrice.toString(),
|
|
||||||
gasPriceDefault: gasPrice.toFormat(),
|
|
||||||
gasPriceHistogram
|
|
||||||
}, this.recalculate);
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.warn('getDefaults', error);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
newError = (error) => {
|
|
||||||
const { store } = this.context;
|
|
||||||
|
|
||||||
store.dispatch({ type: 'newError', error });
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function mapStateToProps (state) {
|
function mapStateToProps (state) {
|
||||||
const { gasLimit } = state.nodeStatus;
|
const { gasLimit } = state.nodeStatus;
|
||||||
|
|
||||||
return { gasLimit };
|
return { gasLimit };
|
||||||
}
|
}
|
||||||
|
|
||||||
function mapDispatchToProps (dispatch) {
|
function mapDispatchToProps (dispatch) {
|
||||||
return bindActionCreators({}, dispatch);
|
return bindActionCreators({
|
||||||
|
newError
|
||||||
|
}, dispatch);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default connect(
|
export default connect(
|
||||||
|
@ -14,10 +14,10 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import { newError } from '../ui/Errors/actions';
|
import { newError } from '~/ui/Errors/actions';
|
||||||
import { setAddressImage } from './providers/imagesActions';
|
import { setAddressImage } from './providers/imagesActions';
|
||||||
import { clearStatusLogs, toggleStatusLogs, toggleStatusRefresh } from './providers/statusActions';
|
import { clearStatusLogs, toggleStatusLogs, toggleStatusRefresh } from './providers/statusActions';
|
||||||
import { toggleView } from '../views/Settings/actions';
|
import { toggleView } from '~/views/Settings/actions';
|
||||||
|
|
||||||
export {
|
export {
|
||||||
newError,
|
newError,
|
||||||
|
@ -15,11 +15,11 @@
|
|||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
import thunk from 'redux-thunk';
|
import thunk from 'redux-thunk';
|
||||||
|
|
||||||
import ErrorsMiddleware from '../ui/Errors/middleware';
|
import ErrorsMiddleware from '~/ui/Errors/middleware';
|
||||||
import SettingsMiddleware from '../views/Settings/middleware';
|
import SettingsMiddleware from '~/views/Settings/middleware';
|
||||||
import SignerMiddleware from './providers/signerMiddleware';
|
import SignerMiddleware from './providers/signerMiddleware';
|
||||||
|
|
||||||
import statusMiddleware from '../views/Status/middleware';
|
import statusMiddleware from '~/views/Status/middleware';
|
||||||
import CertificationsMiddleware from './providers/certifications/middleware';
|
import CertificationsMiddleware from './providers/certifications/middleware';
|
||||||
|
|
||||||
export default function (api) {
|
export default function (api) {
|
||||||
|
@ -17,9 +17,9 @@
|
|||||||
import { throttle } from 'lodash';
|
import { throttle } from 'lodash';
|
||||||
|
|
||||||
import { loadTokens, setTokenReg, fetchBalances, fetchTokens, fetchTokensBalances } from './balancesActions';
|
import { loadTokens, setTokenReg, fetchBalances, fetchTokens, fetchTokensBalances } from './balancesActions';
|
||||||
import { padRight } from '../../api/util/format';
|
import { padRight } from '~/api/util/format';
|
||||||
|
|
||||||
import Contracts from '../../contracts';
|
import Contracts from '~/contracts';
|
||||||
|
|
||||||
export default class Balances {
|
export default class Balances {
|
||||||
constructor (store, api) {
|
constructor (store, api) {
|
||||||
|
@ -19,7 +19,7 @@ import { range, uniq, isEqual } from 'lodash';
|
|||||||
import { hashToImageUrl } from './imagesReducer';
|
import { hashToImageUrl } from './imagesReducer';
|
||||||
import { setAddressImage } from './imagesActions';
|
import { setAddressImage } from './imagesActions';
|
||||||
|
|
||||||
import * as ABIS from '../../contracts/abi';
|
import * as ABIS from '~/contracts/abi';
|
||||||
import imagesEthereum from '../../../assets/images/contracts/ethereum-black-64x64.png';
|
import imagesEthereum from '../../../assets/images/contracts/ethereum-black-64x64.png';
|
||||||
|
|
||||||
const ETH = {
|
const ETH = {
|
||||||
@ -256,7 +256,8 @@ export function queryTokensFilter (tokensFilter) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const tokens = balances.tokens.filter((t) => tokenAddresses.includes(t.address));
|
const tokens = Object.values(balances.tokens)
|
||||||
|
.filter((t) => tokenAddresses.includes(t.address));
|
||||||
|
|
||||||
fetchTokensBalances(uniq(addresses), tokens)(dispatch, getState);
|
fetchTokensBalances(uniq(addresses), tokens)(dispatch, getState);
|
||||||
});
|
});
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import Contracts from '../../contracts';
|
import Contracts from '~/contracts';
|
||||||
|
|
||||||
export function setBlock (blockNumber, block) {
|
export function setBlock (blockNumber, block) {
|
||||||
return {
|
return {
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import Contracts from '../../../contracts';
|
import Contracts from '~/contracts';
|
||||||
import { addCertification } from './actions';
|
import { addCertification } from './actions';
|
||||||
|
|
||||||
const knownCertifiers = [ 'smsverification' ];
|
const knownCertifiers = [ 'smsverification' ];
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import { handleActions } from 'redux-actions';
|
import { handleActions } from 'redux-actions';
|
||||||
import { bytesToHex } from '../../api/util/format';
|
import { bytesToHex } from '~/api/util/format';
|
||||||
|
|
||||||
const ZERO = '0x0000000000000000000000000000000000000000000000000000000000000000';
|
const ZERO = '0x0000000000000000000000000000000000000000000000000000000000000000';
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
import * as actions from './signerActions';
|
import * as actions from './signerActions';
|
||||||
|
|
||||||
import { inHex } from '../../api/format/input';
|
import { inHex } from '~/api/format/input';
|
||||||
import { Wallet } from '../../util/wallet';
|
import { Wallet } from '../../util/wallet';
|
||||||
|
|
||||||
export default class SignerMiddleware {
|
export default class SignerMiddleware {
|
||||||
|
@ -20,9 +20,9 @@ import { routerReducer } from 'react-router-redux';
|
|||||||
import { apiReducer, balancesReducer, blockchainReducer, compilerReducer, imagesReducer, personalReducer, signerReducer, statusReducer as nodeStatusReducer, snackbarReducer } from './providers';
|
import { apiReducer, balancesReducer, blockchainReducer, compilerReducer, imagesReducer, personalReducer, signerReducer, statusReducer as nodeStatusReducer, snackbarReducer } from './providers';
|
||||||
import certificationsReducer from './providers/certifications/reducer';
|
import certificationsReducer from './providers/certifications/reducer';
|
||||||
|
|
||||||
import errorReducer from '../ui/Errors/reducers';
|
import errorReducer from '~/ui/Errors/reducers';
|
||||||
import settingsReducer from '../views/Settings/reducers';
|
import settingsReducer from '~/views/Settings/reducers';
|
||||||
import tooltipReducer from '../ui/Tooltips/reducers';
|
import tooltipReducer from '~/ui/Tooltips/reducers';
|
||||||
|
|
||||||
export default function () {
|
export default function () {
|
||||||
return combineReducers({
|
return combineReducers({
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import { hashToImageUrl } from './providers/imagesReducer';
|
import { hashToImageUrl } from './providers/imagesReducer';
|
||||||
import { withError } from '../ui/Errors/middleware';
|
import { withError } from '~/ui/Errors/middleware';
|
||||||
|
|
||||||
export {
|
export {
|
||||||
hashToImageUrl,
|
hashToImageUrl,
|
||||||
|
@ -18,8 +18,8 @@ import React, { Component, PropTypes } from 'react';
|
|||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { bindActionCreators } from 'redux';
|
import { bindActionCreators } from 'redux';
|
||||||
|
|
||||||
import { hashToImageUrl } from '../../redux/providers/imagesReducer';
|
import { hashToImageUrl } from '~/redux/providers/imagesReducer';
|
||||||
import { fetchCertifications } from '../../redux/providers/certifications/actions';
|
import { fetchCertifications } from '~/redux/providers/certifications/actions';
|
||||||
|
|
||||||
import defaultIcon from '../../../assets/images/certifications/unknown.svg';
|
import defaultIcon from '../../../assets/images/certifications/unknown.svg';
|
||||||
|
|
||||||
|
@ -23,7 +23,7 @@ import Clipboard from 'react-copy-to-clipboard';
|
|||||||
import CopyIcon from 'material-ui/svg-icons/content/content-copy';
|
import CopyIcon from 'material-ui/svg-icons/content/content-copy';
|
||||||
import Theme from '../Theme';
|
import Theme from '../Theme';
|
||||||
|
|
||||||
import { showSnackbar } from '../../redux/providers/snackbarActions';
|
import { showSnackbar } from '~/redux/providers/snackbarActions';
|
||||||
|
|
||||||
const { textColor, disabledTextColor } = Theme.flatButton;
|
const { textColor, disabledTextColor } = Theme.flatButton;
|
||||||
|
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
|
|
||||||
import React, { Component, PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { MenuItem } from 'material-ui';
|
import { MenuItem } from 'material-ui';
|
||||||
import { isEqual } from 'lodash';
|
|
||||||
|
|
||||||
import AutoComplete from '../AutoComplete';
|
import AutoComplete from '../AutoComplete';
|
||||||
import IdentityIcon from '../../IdentityIcon';
|
import IdentityIcon from '../../IdentityIcon';
|
||||||
@ -64,13 +63,6 @@ export default class AddressSelect extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
componentWillReceiveProps (newProps) {
|
componentWillReceiveProps (newProps) {
|
||||||
const entries = this.entriesFromProps();
|
|
||||||
const addresses = Object.keys(entries).sort();
|
|
||||||
|
|
||||||
if (!isEqual(addresses, this.state.addresses)) {
|
|
||||||
this.setState({ entries, addresses });
|
|
||||||
}
|
|
||||||
|
|
||||||
if (newProps.value !== this.props.value) {
|
if (newProps.value !== this.props.value) {
|
||||||
this.setState({ value: newProps.value });
|
this.setState({ value: newProps.value });
|
||||||
}
|
}
|
||||||
@ -127,31 +119,33 @@ export default class AddressSelect extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
renderItem = (entry) => {
|
renderItem = (entry) => {
|
||||||
|
const { address, name } = entry;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
text: entry.name && entry.name.toUpperCase() || entry.address,
|
text: name && name.toUpperCase() || address,
|
||||||
value: this.renderSelectEntry(entry),
|
value: this.renderMenuItem(address),
|
||||||
address: entry.address
|
address
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
renderSelectEntry = (entry) => {
|
renderMenuItem (address) {
|
||||||
const item = (
|
const item = (
|
||||||
<div className={ styles.account }>
|
<div className={ styles.account }>
|
||||||
<IdentityIcon
|
<IdentityIcon
|
||||||
className={ styles.image }
|
className={ styles.image }
|
||||||
inline center
|
inline center
|
||||||
address={ entry.address } />
|
address={ address } />
|
||||||
<IdentityName
|
<IdentityName
|
||||||
className={ styles.name }
|
className={ styles.name }
|
||||||
address={ entry.address } />
|
address={ address } />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<MenuItem
|
<MenuItem
|
||||||
className={ styles.menuItem }
|
className={ styles.menuItem }
|
||||||
key={ entry.address }
|
key={ address }
|
||||||
value={ entry.address }
|
value={ address }
|
||||||
label={ item }>
|
label={ item }>
|
||||||
{ item }
|
{ item }
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
|
@ -19,6 +19,8 @@ import keycode from 'keycode';
|
|||||||
import { MenuItem, AutoComplete as MUIAutoComplete } from 'material-ui';
|
import { MenuItem, AutoComplete as MUIAutoComplete } from 'material-ui';
|
||||||
import { PopoverAnimationVertical } from 'material-ui/Popover';
|
import { PopoverAnimationVertical } from 'material-ui/Popover';
|
||||||
|
|
||||||
|
import { isEqual } from 'lodash';
|
||||||
|
|
||||||
export default class AutoComplete extends Component {
|
export default class AutoComplete extends Component {
|
||||||
static propTypes = {
|
static propTypes = {
|
||||||
onChange: PropTypes.func.isRequired,
|
onChange: PropTypes.func.isRequired,
|
||||||
@ -42,12 +44,28 @@ export default class AutoComplete extends Component {
|
|||||||
lastChangedValue: undefined,
|
lastChangedValue: undefined,
|
||||||
entry: null,
|
entry: null,
|
||||||
open: false,
|
open: false,
|
||||||
fakeBlur: false
|
fakeBlur: false,
|
||||||
|
dataSource: []
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillMount () {
|
||||||
|
const dataSource = this.getDataSource();
|
||||||
|
this.setState({ dataSource });
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillReceiveProps (nextProps) {
|
||||||
|
const prevEntries = Object.keys(this.props.entries || {}).sort();
|
||||||
|
const nextEntries = Object.keys(nextProps.entries || {}).sort();
|
||||||
|
|
||||||
|
if (!isEqual(prevEntries, nextEntries)) {
|
||||||
|
const dataSource = this.getDataSource(nextProps);
|
||||||
|
this.setState({ dataSource });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
const { disabled, error, hint, label, value, className, filter, onUpdateInput } = this.props;
|
const { disabled, error, hint, label, value, className, filter, onUpdateInput } = this.props;
|
||||||
const { open } = this.state;
|
const { open, dataSource } = this.state;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<MUIAutoComplete
|
<MUIAutoComplete
|
||||||
@ -68,7 +86,7 @@ export default class AutoComplete extends Component {
|
|||||||
menuCloseDelay={ 0 }
|
menuCloseDelay={ 0 }
|
||||||
fullWidth
|
fullWidth
|
||||||
floatingLabelFixed
|
floatingLabelFixed
|
||||||
dataSource={ this.getDataSource() }
|
dataSource={ dataSource }
|
||||||
menuProps={ { maxHeight: 400 } }
|
menuProps={ { maxHeight: 400 } }
|
||||||
ref='muiAutocomplete'
|
ref='muiAutocomplete'
|
||||||
onKeyDown={ this.onKeyDown }
|
onKeyDown={ this.onKeyDown }
|
||||||
@ -76,8 +94,8 @@ export default class AutoComplete extends Component {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
getDataSource () {
|
getDataSource (props = this.props) {
|
||||||
const { renderItem, entries } = this.props;
|
const { renderItem, entries } = props;
|
||||||
const entriesArray = (entries instanceof Array)
|
const entriesArray = (entries instanceof Array)
|
||||||
? entries
|
? entries
|
||||||
: Object.values(entries);
|
: Object.values(entries);
|
||||||
|
@ -20,7 +20,7 @@ import { bindActionCreators } from 'redux';
|
|||||||
|
|
||||||
import Input from '../Input';
|
import Input from '../Input';
|
||||||
import IdentityIcon from '../../IdentityIcon';
|
import IdentityIcon from '../../IdentityIcon';
|
||||||
import util from '../../../api/util';
|
import util from '~/api/util';
|
||||||
|
|
||||||
import styles from './inputAddress.css';
|
import styles from './inputAddress.css';
|
||||||
|
|
||||||
|
@ -22,9 +22,9 @@ import IconButton from 'material-ui/IconButton';
|
|||||||
import AddIcon from 'material-ui/svg-icons/content/add';
|
import AddIcon from 'material-ui/svg-icons/content/add';
|
||||||
import RemoveIcon from 'material-ui/svg-icons/content/remove';
|
import RemoveIcon from 'material-ui/svg-icons/content/remove';
|
||||||
|
|
||||||
import Input from '../../../ui/Form/Input';
|
import Input from '~/ui/Form/Input';
|
||||||
import InputAddressSelect from '../../../ui/Form/InputAddressSelect';
|
import InputAddressSelect from '~/ui/Form/InputAddressSelect';
|
||||||
import Select from '../../../ui/Form/Select';
|
import Select from '~/ui/Form/Select';
|
||||||
|
|
||||||
import { ABI_TYPES } from '../../../util/abi';
|
import { ABI_TYPES } from '../../../util/abi';
|
||||||
|
|
||||||
|
@ -14,4 +14,4 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
export default from './requestFinished';
|
export default from './loading';
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user