mirror of
https://github.com/kaspanet/kaspad.git
synced 2025-09-14 05:20:11 +00:00
Compare commits
455 Commits
v0.8.8-rc2
...
master
Author | SHA1 | Date | |
---|---|---|---|
![]() |
4bb5bf25d3 | ||
![]() |
25c2dd8670 | ||
![]() |
c93100ccd0 | ||
![]() |
03cc7dfc19 | ||
![]() |
ed745a9acb | ||
![]() |
c23c1d141c | ||
![]() |
352d261fd6 | ||
![]() |
43b9523919 | ||
![]() |
6085d1fc84 | ||
![]() |
1e9ddc42d0 | ||
![]() |
48a142e12f | ||
![]() |
86b89065cf | ||
![]() |
f41dc7fa0b | ||
![]() |
6b38bf7069 | ||
![]() |
d2453f8e7b | ||
![]() |
629faa8436 | ||
![]() |
91e6c6b74b | ||
![]() |
0819244ba1 | ||
![]() |
a0149cd8d0 | ||
![]() |
5a3b8a0066 | ||
![]() |
8e71f79f98 | ||
![]() |
346341a709 | ||
![]() |
8c881aea39 | ||
![]() |
40ec440dcf | ||
![]() |
88bdcb43bc | ||
![]() |
9d1e44673f | ||
![]() |
387fade044 | ||
![]() |
c417c8b525 | ||
![]() |
bd1420220a | ||
![]() |
5640ec4020 | ||
![]() |
1c0887ca60 | ||
![]() |
7be3f41aa7 | ||
![]() |
26c4c73624 | ||
![]() |
880d917e58 | ||
![]() |
3c53c6d8cd | ||
![]() |
3c4b973090 | ||
![]() |
8aee8f81c5 | ||
![]() |
ec3441e63f | ||
![]() |
e3ba1ca07e | ||
![]() |
27fdbd9c88 | ||
![]() |
377d9aaaeb | ||
![]() |
beee947dda | ||
![]() |
d4a27bf1c1 | ||
![]() |
eec6eb9669 | ||
![]() |
d5c10832c2 | ||
![]() |
9fbfba17b6 | ||
![]() |
09d698dd0e | ||
![]() |
ec51c6926a | ||
![]() |
7d44275eb1 | ||
![]() |
a3387a56b3 | ||
![]() |
c2ae03fc89 | ||
![]() |
6c774c966b | ||
![]() |
2d54c9693b | ||
![]() |
d8350d62b0 | ||
![]() |
26c7db251f | ||
![]() |
4d435f2b3a | ||
![]() |
067688f549 | ||
![]() |
3a3fa0d3f0 | ||
![]() |
cf4073b773 | ||
![]() |
6a5e7c9e3f | ||
![]() |
7e9b5b9010 | ||
![]() |
953838e0d8 | ||
![]() |
a1dcb34c29 | ||
![]() |
23764e1b0b | ||
![]() |
0838cc8e32 | ||
![]() |
9f51330f38 | ||
![]() |
f6d46fd23f | ||
![]() |
2a7e03e232 | ||
![]() |
3286a7d010 | ||
![]() |
aabbc741d7 | ||
![]() |
20b7ab89f9 | ||
![]() |
10f1e7e3f4 | ||
![]() |
d941c73701 | ||
![]() |
3f80638c86 | ||
![]() |
266ec6c270 | ||
![]() |
9ee409afaa | ||
![]() |
715cb3b1ac | ||
![]() |
eb693c4a86 | ||
![]() |
7a61c637b0 | ||
![]() |
c7bd84ef9d | ||
![]() |
b26b9f6c4b | ||
![]() |
1c9bb54cc2 | ||
![]() |
b9093d59eb | ||
![]() |
18d000f625 | ||
![]() |
c5aade7e7f | ||
![]() |
d4b741fd7c | ||
![]() |
74a4f927e9 | ||
![]() |
847aafc91f | ||
![]() |
c87e541570 | ||
![]() |
2ea1c4f922 | ||
![]() |
5e9c28b77b | ||
![]() |
d957a6d93a | ||
![]() |
b2648aa5bd | ||
![]() |
3908f274ae | ||
![]() |
fa7ea121ff | ||
![]() |
24848da895 | ||
![]() |
b200b77541 | ||
![]() |
d50ad0667c | ||
![]() |
5cea285960 | ||
![]() |
7eb5085f6b | ||
![]() |
491e3569d2 | ||
![]() |
440aea19b0 | ||
![]() |
968d47c3e6 | ||
![]() |
052193865e | ||
![]() |
85febcb551 | ||
![]() |
a4d9fa10bf | ||
![]() |
cd5fd86ad3 | ||
![]() |
b84d6fed2c | ||
![]() |
24c94b38be | ||
![]() |
4dd7113dc5 | ||
![]() |
48c7fa0104 | ||
![]() |
4d0cf2169a | ||
![]() |
5f7cc079e9 | ||
![]() |
016ddfdfce | ||
![]() |
5d24e2afbc | ||
![]() |
8735da045f | ||
![]() |
c839337425 | ||
![]() |
7390651072 | ||
![]() |
52fbeedf20 | ||
![]() |
1660cf0cf1 | ||
![]() |
2b5202be7a | ||
![]() |
9ffbb15160 | ||
![]() |
540b0d3a22 | ||
![]() |
8d5faee53a | ||
![]() |
6e2fd0633b | ||
![]() |
beb038c815 | ||
![]() |
35a959b56f | ||
![]() |
57c6118be8 | ||
![]() |
723aebbec9 | ||
![]() |
2b395e34b1 | ||
![]() |
ada559f007 | ||
![]() |
357e8ce73c | ||
![]() |
6725902663 | ||
![]() |
99bb21c512 | ||
![]() |
a4669f3fb5 | ||
![]() |
e8f40bdff9 | ||
![]() |
68a407ea37 | ||
![]() |
80879cabe1 | ||
![]() |
71afc62298 | ||
![]() |
ca5c8549b9 | ||
![]() |
ab73def07a | ||
![]() |
3f840233d8 | ||
![]() |
90d9edb8e5 | ||
![]() |
b9b360bce4 | ||
![]() |
27654961f9 | ||
![]() |
d45af760d8 | ||
![]() |
95fa045297 | ||
![]() |
cb65dae63d | ||
![]() |
21b82d7efc | ||
![]() |
63c6d7443b | ||
![]() |
753f4a2ec1 | ||
![]() |
ed667f7e54 | ||
![]() |
c4a034eb43 | ||
![]() |
2eca0f0b5f | ||
![]() |
58d627e05a | ||
![]() |
639183ba0e | ||
![]() |
9fa08442cf | ||
![]() |
0dd50394ec | ||
![]() |
ac8d4e1341 | ||
![]() |
2488fbde78 | ||
![]() |
2ab8065142 | ||
![]() |
25410b86ae | ||
![]() |
4e44dd8510 | ||
![]() |
1e56a22b32 | ||
![]() |
7a95f0c7a4 | ||
![]() |
c81506220b | ||
![]() |
e5598c15a7 | ||
![]() |
433af5e0fe | ||
![]() |
b7be807167 | ||
![]() |
e687ceeae7 | ||
![]() |
04e35321aa | ||
![]() |
061e65be93 | ||
![]() |
190e725dd0 | ||
![]() |
6449b03034 | ||
![]() |
9f02a24e8b | ||
![]() |
9b23bbcdb5 | ||
![]() |
b30f7309a2 | ||
![]() |
1c18a49992 | ||
![]() |
28d0f1ea2e | ||
![]() |
3f7e482291 | ||
![]() |
ce4f5fcc33 | ||
![]() |
be3a6604d7 | ||
![]() |
f452531df0 | ||
![]() |
13a09da848 | ||
![]() |
f58aeb4f9f | ||
![]() |
82f0a4d74f | ||
![]() |
69d90fe827 | ||
![]() |
c85b5d70fd | ||
![]() |
1cd712a63e | ||
![]() |
27ba9d0374 | ||
![]() |
b1229f7908 | ||
![]() |
4a560f25a6 | ||
![]() |
dab1a881fe | ||
![]() |
598392d0cf | ||
![]() |
6d27637055 | ||
![]() |
4855d845b3 | ||
![]() |
b1b179c105 | ||
![]() |
dadacdc0f4 | ||
![]() |
d2379608ad | ||
![]() |
14b2bcbd81 | ||
![]() |
71b284f4d5 | ||
![]() |
0e1d247915 | ||
![]() |
504ec36612 | ||
![]() |
c80b113319 | ||
![]() |
0bdd19136f | ||
![]() |
7c1cddff11 | ||
![]() |
064b0454e8 | ||
![]() |
8282fb486e | ||
![]() |
428449bb7d | ||
![]() |
f54659ead0 | ||
![]() |
99f82eb80f | ||
![]() |
aa43c14fc5 | ||
![]() |
129e9119d2 | ||
![]() |
cae7faced2 | ||
![]() |
f036b18f9e | ||
![]() |
1d740e1eab | ||
![]() |
011871cda2 | ||
![]() |
7b4f761fb9 | ||
![]() |
5806fef35f | ||
![]() |
227ef392ba | ||
![]() |
f3d76d6565 | ||
![]() |
df573bba63 | ||
![]() |
2a97b7c9bb | ||
![]() |
70900c571b | ||
![]() |
7292438e4a | ||
![]() |
dced1a9376 | ||
![]() |
32e8e539ac | ||
![]() |
11103a36d3 | ||
![]() |
606b781ca0 | ||
![]() |
dbf18d8052 | ||
![]() |
2a1b38ce7a | ||
![]() |
29c410d123 | ||
![]() |
6e6fabf956 | ||
![]() |
b04292c97a | ||
![]() |
765dd170e4 | ||
![]() |
8e362845b3 | ||
![]() |
5c1ba9170e | ||
![]() |
9d8c555bdf | ||
![]() |
a2f574eab8 | ||
![]() |
7bed86dc1b | ||
![]() |
9b81f5145e | ||
![]() |
cd8341ef57 | ||
![]() |
ad8bdbed21 | ||
![]() |
7cdceb6df0 | ||
![]() |
cc5248106e | ||
![]() |
e3463b7268 | ||
![]() |
a2173ef80a | ||
![]() |
aeb4500b61 | ||
![]() |
0a1daae319 | ||
![]() |
131cd3357e | ||
![]() |
ff72568d6b | ||
![]() |
2dddb650b9 | ||
![]() |
99aaacd649 | ||
![]() |
77a344cc29 | ||
![]() |
3dbc42b4f7 | ||
![]() |
1b9be28613 | ||
![]() |
5dbb1da84b | ||
![]() |
afaac28da1 | ||
![]() |
0053ee788d | ||
![]() |
af7e7de247 | ||
![]() |
02a08902a7 | ||
![]() |
d9bc94a2a8 | ||
![]() |
837dac68b5 | ||
![]() |
ba5880fab1 | ||
![]() |
7b5720a155 | ||
![]() |
65b5a080e4 | ||
![]() |
ce17348175 | ||
![]() |
d922ee1be2 | ||
![]() |
4132891ac9 | ||
![]() |
2094f4facf | ||
![]() |
2de68f43f0 | ||
![]() |
d748089a14 | ||
![]() |
7d1071a9b1 | ||
![]() |
f26a7fdedf | ||
![]() |
d207888b67 | ||
![]() |
38e2ee1b43 | ||
![]() |
aba44e7bfb | ||
![]() |
c731d74bc0 | ||
![]() |
60e7a8ebed | ||
![]() |
369a3bac09 | ||
![]() |
8022e4cbea | ||
![]() |
28ac77b202 | ||
![]() |
28af7eb596 | ||
![]() |
a4d241c30a | ||
![]() |
487fab0e2b | ||
![]() |
2f272cd517 | ||
![]() |
e3a6d9e49a | ||
![]() |
069ee26e84 | ||
![]() |
61aa15fd61 | ||
![]() |
f7cce5cb39 | ||
![]() |
2f7a1395e7 | ||
![]() |
8b1ac86532 | ||
![]() |
ab721f3ad6 | ||
![]() |
798c5fab7d | ||
![]() |
c13a4d90ed | ||
![]() |
4ba8b14675 | ||
![]() |
319cbce768 | ||
![]() |
bdd42903b4 | ||
![]() |
9bedf84740 | ||
![]() |
f317f51cdd | ||
![]() |
4207c82f5a | ||
![]() |
70399dae2a | ||
![]() |
2ae1b7853f | ||
![]() |
d53d040bee | ||
![]() |
79c74c482b | ||
![]() |
3b0394eefe | ||
![]() |
43e6467ff1 | ||
![]() |
363494ef7a | ||
![]() |
d1df97c4c5 | ||
![]() |
4f52a6de51 | ||
![]() |
4f4a8934e7 | ||
![]() |
16ba2bd312 | ||
![]() |
6613faee2d | ||
![]() |
edc459ae1b | ||
![]() |
d7f2cf81c0 | ||
![]() |
4658f9d05c | ||
![]() |
010df3b0d3 | ||
![]() |
346598e67f | ||
![]() |
268906a7ce | ||
![]() |
befc60b185 | ||
![]() |
dd3e04e671 | ||
![]() |
9c743db4d6 | ||
![]() |
eb3dba5c88 | ||
![]() |
e46e2580b1 | ||
![]() |
414f58fb90 | ||
![]() |
9df80957b1 | ||
![]() |
268c9fa83c | ||
![]() |
2e3592e351 | ||
![]() |
19718ac102 | ||
![]() |
28a8e96e65 | ||
![]() |
4df283934a | ||
![]() |
ab89efe3dc | ||
![]() |
fa16c30cf3 | ||
![]() |
c28366eb50 | ||
![]() |
dc0bf56bf3 | ||
![]() |
91de1807ad | ||
![]() |
830684167c | ||
![]() |
1f56a68a28 | ||
![]() |
13a6b4cc51 | ||
![]() |
dfd8b3423d | ||
![]() |
28bfc0fb9c | ||
![]() |
83beae4463 | ||
![]() |
a6ebe83198 | ||
![]() |
acdc59b565 | ||
![]() |
15811b0bcb | ||
![]() |
a8a7e3dd9b | ||
![]() |
3f193e9219 | ||
![]() |
dfa24d8353 | ||
![]() |
3c3ad1425d | ||
![]() |
9bb8123391 | ||
![]() |
347dd8fc4b | ||
![]() |
d2cccd2829 | ||
![]() |
7186f83095 | ||
![]() |
5c394c2951 | ||
![]() |
a786cdc15e | ||
![]() |
6dd3d4a9e7 | ||
![]() |
73b36f12f0 | ||
![]() |
a795a9e619 | ||
![]() |
0be1bba408 | ||
![]() |
6afc06ce58 | ||
![]() |
d01a213f3d | ||
![]() |
7ad8ce521c | ||
![]() |
86ba80a091 | ||
![]() |
088e2114c2 | ||
![]() |
2854d91688 | ||
![]() |
af10b59181 | ||
![]() |
c5b0394bbc | ||
![]() |
9266d179a9 | ||
![]() |
321792778e | ||
![]() |
70f3fa9893 | ||
![]() |
4e18031483 | ||
![]() |
2abc284e3b | ||
![]() |
f1451406f7 | ||
![]() |
c12e180873 | ||
![]() |
3959bc1e7c | ||
![]() |
6ec0a8a559 | ||
![]() |
6824be9216 | ||
![]() |
d0511c1636 | ||
![]() |
7d69b66c7c | ||
![]() |
cebcab7f5c | ||
![]() |
caf251b7a8 | ||
![]() |
1a4161ffc0 | ||
![]() |
b84080f3d9 | ||
![]() |
cbd0bb6d14 | ||
![]() |
d9449a32b8 | ||
![]() |
0ee8f2b631 | ||
![]() |
ff1c96c149 | ||
![]() |
5e335be5ab | ||
![]() |
032eda4604 | ||
![]() |
e4e3541a30 | ||
![]() |
b5933bc4fe | ||
![]() |
ec446ac511 | ||
![]() |
3d668cc1bd | ||
![]() |
1486a6312c | ||
![]() |
cd27f2850e | ||
![]() |
14cf7f81f3 | ||
![]() |
74539f8f0b | ||
![]() |
a7299c1b87 | ||
![]() |
27c1e4611e | ||
![]() |
b8413fcecb | ||
![]() |
c084c69771 | ||
![]() |
53781eed4d | ||
![]() |
837fa65735 | ||
![]() |
dd3b2cf7d1 | ||
![]() |
3fd324ca28 | ||
![]() |
0271858f25 | ||
![]() |
7909480757 | ||
![]() |
18274c236b | ||
![]() |
7829a9fd76 | ||
![]() |
1548ed9629 | ||
![]() |
32cd643e8b | ||
![]() |
05df6e3e4e | ||
![]() |
ce326b3c7d | ||
![]() |
a4ae263ed5 | ||
![]() |
79be1edaa5 | ||
![]() |
c1ef5f0c56 | ||
![]() |
458e409654 | ||
![]() |
df19bdfaf3 | ||
![]() |
103edf97d0 | ||
![]() |
1f69f9eed9 | ||
![]() |
6a12428504 | ||
![]() |
63b1d2a05a | ||
![]() |
089115c389 | ||
![]() |
24a12cf2a1 | ||
![]() |
6597f24bab | ||
![]() |
dc84913214 | ||
![]() |
201646282b | ||
![]() |
3f08bf87a8 | ||
![]() |
581a12db96 | ||
![]() |
fb6c9c8f21 | ||
![]() |
2adb4f5d0f | ||
![]() |
9ffda2b1da | ||
![]() |
bee0893660 | ||
![]() |
a7bb1853f9 | ||
![]() |
35e555e959 | ||
![]() |
a250f697ee | ||
![]() |
f66708b3c6 | ||
![]() |
8fbea5d239 | ||
![]() |
5fa06fe7d7 | ||
![]() |
06fd6f1b95 | ||
![]() |
d2f4ed660c | ||
![]() |
19878aa062 | ||
![]() |
6415e525c3 | ||
![]() |
995e526dae | ||
![]() |
00a023620d | ||
![]() |
2908a46441 | ||
![]() |
e78cdff3d0 | ||
![]() |
2a31074fc4 | ||
![]() |
d835f72e74 | ||
![]() |
a581dea127 | ||
![]() |
7b4b5668e2 | ||
![]() |
0e2061d838 | ||
![]() |
0a579e7f78 | ||
![]() |
1ed6c4c086 | ||
![]() |
fea83e5c6c |
4
.github/workflows/SetPageFileSize.ps1
vendored
4
.github/workflows/SetPageFileSize.ps1
vendored
@ -11,8 +11,8 @@
|
|||||||
#>
|
#>
|
||||||
|
|
||||||
param(
|
param(
|
||||||
[System.UInt64] $MinimumSize = 8gb ,
|
[System.UInt64] $MinimumSize = 16gb ,
|
||||||
[System.UInt64] $MaximumSize = 8gb ,
|
[System.UInt64] $MaximumSize = 16gb ,
|
||||||
[System.String] $DiskRoot = "D:"
|
[System.String] $DiskRoot = "D:"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
69
.github/workflows/deploy.yaml
vendored
Normal file
69
.github/workflows/deploy.yaml
vendored
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
name: Build and upload assets
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
os: [ubuntu-latest, windows-latest, macos-latest]
|
||||||
|
name: Building, ${{ matrix.os }}
|
||||||
|
steps:
|
||||||
|
- name: Fix CRLF on Windows
|
||||||
|
if: runner.os == 'Windows'
|
||||||
|
run: git config --global core.autocrlf false
|
||||||
|
|
||||||
|
- name: Check out code into the Go module directory
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup Go
|
||||||
|
uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version: 1.21
|
||||||
|
|
||||||
|
- name: Build on Linux
|
||||||
|
if: runner.os == 'Linux'
|
||||||
|
# `-extldflags=-static` - means static link everything,
|
||||||
|
# `-tags netgo,osusergo` means use pure go replacements for "os/user" and "net"
|
||||||
|
# `-s -w` strips the binary to produce smaller size binaries
|
||||||
|
run: |
|
||||||
|
go build -v -ldflags="-s -w -extldflags=-static" -tags netgo,osusergo -o ./bin/ ./cmd/...
|
||||||
|
archive="bin/kaspad-${{ github.event.release.tag_name }}-linux.zip"
|
||||||
|
asset_name="kaspad-${{ github.event.release.tag_name }}-linux.zip"
|
||||||
|
zip -r "${archive}" ./bin/*
|
||||||
|
echo "archive=${archive}" >> $GITHUB_ENV
|
||||||
|
echo "asset_name=${asset_name}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Build on Windows
|
||||||
|
if: runner.os == 'Windows'
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
go build -v -ldflags="-s -w" -o bin/ ./cmd/...
|
||||||
|
archive="bin/kaspad-${{ github.event.release.tag_name }}-win64.zip"
|
||||||
|
asset_name="kaspad-${{ github.event.release.tag_name }}-win64.zip"
|
||||||
|
powershell "Compress-Archive bin/* \"${archive}\""
|
||||||
|
echo "archive=${archive}" >> $GITHUB_ENV
|
||||||
|
echo "asset_name=${asset_name}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Build on MacOS
|
||||||
|
if: runner.os == 'macOS'
|
||||||
|
run: |
|
||||||
|
go build -v -ldflags="-s -w" -o ./bin/ ./cmd/...
|
||||||
|
archive="bin/kaspad-${{ github.event.release.tag_name }}-osx.zip"
|
||||||
|
asset_name="kaspad-${{ github.event.release.tag_name }}-osx.zip"
|
||||||
|
zip -r "${archive}" ./bin/*
|
||||||
|
echo "archive=${archive}" >> $GITHUB_ENV
|
||||||
|
echo "asset_name=${asset_name}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Upload release asset
|
||||||
|
uses: actions/upload-release-asset@v1
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
upload_url: ${{ github.event.release.upload_url }}
|
||||||
|
asset_path: "./${{ env.archive }}"
|
||||||
|
asset_name: "${{ env.asset_name }}"
|
||||||
|
asset_content_type: application/zip
|
69
.github/workflows/go.yml
vendored
69
.github/workflows/go.yml
vendored
@ -1,69 +0,0 @@
|
|||||||
name: Go
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
pull_request:
|
|
||||||
# edtited - "title, body, or the base branch of the PR is modified"
|
|
||||||
# synchronize - "commit(s) pushed to the pull request"
|
|
||||||
types: [opened, synchronize, edited, reopened]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: ${{ matrix.os }}
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
os: [ ubuntu-16.04, macos-10.15 ]
|
|
||||||
name: Testing on on ${{ matrix.os }}
|
|
||||||
steps:
|
|
||||||
|
|
||||||
- name: Fix windows CRLF
|
|
||||||
run: git config --global core.autocrlf false
|
|
||||||
|
|
||||||
- name: Check out code into the Go module directory
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
# We need to increase the page size because the tests run out of memory on github CI windows.
|
|
||||||
# Use the powershell script from this github action: https://github.com/al-cheb/configure-pagefile-action/blob/master/scripts/SetPageFileSize.ps1
|
|
||||||
# MIT License (MIT) Copyright (c) 2020 Maxim Lobanov and contributors
|
|
||||||
- name: Increase page size on windows
|
|
||||||
if: runner.os == 'Windows'
|
|
||||||
shell: powershell
|
|
||||||
run: powershell -command .\.github\workflows\SetPageFileSize.ps1
|
|
||||||
|
|
||||||
|
|
||||||
- name: Set up Go 1.x
|
|
||||||
uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: 1.15
|
|
||||||
|
|
||||||
|
|
||||||
# Source: https://github.com/actions/cache/blob/main/examples.md#go---modules
|
|
||||||
- name: Go Cache
|
|
||||||
uses: actions/cache@v2
|
|
||||||
with:
|
|
||||||
path: ~/go/pkg/mod
|
|
||||||
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-go-
|
|
||||||
|
|
||||||
- name: Test
|
|
||||||
shell: bash
|
|
||||||
run: ./build_and_test.sh
|
|
||||||
|
|
||||||
coverage:
|
|
||||||
runs-on: ubuntu-20.04
|
|
||||||
name: Produce code coverage
|
|
||||||
steps:
|
|
||||||
- name: Check out code into the Go module directory
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Set up Go 1.x
|
|
||||||
uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: 1.15
|
|
||||||
|
|
||||||
- name: Create coverage file
|
|
||||||
run: go test -covermode=atomic -coverpkg=./... -coverprofile coverage.txt ./...
|
|
||||||
|
|
||||||
- name: Upload coverage file
|
|
||||||
run: bash <(curl -s https://codecov.io/bash)
|
|
49
.github/workflows/race.yaml
vendored
Normal file
49
.github/workflows/race.yaml
vendored
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
name: Race
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: "0 0 * * *"
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
race_test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
branch: [master, latest]
|
||||||
|
name: Race detection on ${{ matrix.branch }}
|
||||||
|
steps:
|
||||||
|
- name: Check out code into the Go module directory
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Setup Go
|
||||||
|
uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version: 1.23
|
||||||
|
|
||||||
|
- name: Set scheduled branch name
|
||||||
|
shell: bash
|
||||||
|
if: github.event_name == 'schedule'
|
||||||
|
run: |
|
||||||
|
if [ "${{ matrix.branch }}" == "master" ]; then
|
||||||
|
echo "run_on=master" >> $GITHUB_ENV
|
||||||
|
fi
|
||||||
|
if [ "${{ matrix.branch }}" == "latest" ]; then
|
||||||
|
branch=$(git branch -r | grep 'v\([0-9]\+\.\)\([0-9]\+\.\)\([0-9]\+\)-dev' | sort -Vr | head -1 | xargs)
|
||||||
|
echo "run_on=${branch}" >> $GITHUB_ENV
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Set manual branch name
|
||||||
|
shell: bash
|
||||||
|
if: github.event_name == 'workflow_dispatch'
|
||||||
|
run: echo "run_on=${{ github.ref }}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Test with race detector
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
git checkout "${{ env.run_on }}"
|
||||||
|
git status
|
||||||
|
go test -timeout 20m -race ./...
|
92
.github/workflows/tests.yaml
vendored
Normal file
92
.github/workflows/tests.yaml
vendored
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
name: Tests
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
pull_request:
|
||||||
|
# edtited - because base branch can be modified
|
||||||
|
# synchronize - update commits on PR
|
||||||
|
types: [opened, synchronize, edited]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
os: [ubuntu-latest, macos-latest]
|
||||||
|
name: Tests, ${{ matrix.os }}
|
||||||
|
steps:
|
||||||
|
- name: Fix CRLF on Windows
|
||||||
|
if: runner.os == 'Windows'
|
||||||
|
run: git config --global core.autocrlf false
|
||||||
|
|
||||||
|
- name: Check out code into the Go module directory
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
# Increase the pagefile size on Windows to aviod running out of memory
|
||||||
|
- name: Increase pagefile size on Windows
|
||||||
|
if: runner.os == 'Windows'
|
||||||
|
run: powershell -command .github\workflows\SetPageFileSize.ps1
|
||||||
|
|
||||||
|
- name: Setup Go
|
||||||
|
uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version: 1.23
|
||||||
|
|
||||||
|
# Source: https://github.com/actions/cache/blob/main/examples.md#go---modules
|
||||||
|
- name: Go Cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ~/go/pkg/mod
|
||||||
|
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-go-
|
||||||
|
|
||||||
|
- name: Test
|
||||||
|
shell: bash
|
||||||
|
run: ./build_and_test.sh -v
|
||||||
|
|
||||||
|
stability-test-fast:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
name: Fast stability tests, ${{ github.head_ref }}
|
||||||
|
steps:
|
||||||
|
- name: Setup Go
|
||||||
|
uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version: 1.23
|
||||||
|
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Install kaspad
|
||||||
|
run: go install ./...
|
||||||
|
|
||||||
|
- name: Install golint
|
||||||
|
run: go get -u golang.org/x/lint/golint
|
||||||
|
|
||||||
|
- name: Run fast stability tests
|
||||||
|
working-directory: stability-tests
|
||||||
|
run: ./install_and_test.sh
|
||||||
|
|
||||||
|
coverage:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
name: Produce code coverage
|
||||||
|
steps:
|
||||||
|
- name: Check out code into the Go module directory
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup Go
|
||||||
|
uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version: 1.23
|
||||||
|
|
||||||
|
- name: Delete the stability tests from coverage
|
||||||
|
run: rm -r stability-tests
|
||||||
|
|
||||||
|
- name: Create coverage file
|
||||||
|
run: go test -v -covermode=atomic -coverpkg=./... -coverprofile coverage.txt ./...
|
||||||
|
|
||||||
|
- name: Upload coverage file
|
||||||
|
run: bash <(curl -s https://codecov.io/bash)
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -53,6 +53,7 @@ _testmain.go
|
|||||||
debug
|
debug
|
||||||
debug.test
|
debug.test
|
||||||
__debug_bin
|
__debug_bin
|
||||||
|
*__debug_*
|
||||||
|
|
||||||
# CI
|
# CI
|
||||||
version.txt
|
version.txt
|
||||||
|
43
CODE_OF_CONDUCT.md
Normal file
43
CODE_OF_CONDUCT.md
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
# Contributor Covenant Code of Conduct
|
||||||
|
|
||||||
|
## Our Pledge
|
||||||
|
|
||||||
|
In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, gender identity and expression, level of experience, nationality, personal appearance, race, religion, or sexual identity and orientation.
|
||||||
|
|
||||||
|
## Our Standards
|
||||||
|
|
||||||
|
Examples of behavior that contributes to creating a positive environment include:
|
||||||
|
|
||||||
|
* Using welcoming and inclusive language
|
||||||
|
* Being respectful of differing viewpoints and experiences
|
||||||
|
* Gracefully accepting constructive criticism
|
||||||
|
* Focusing on what is best for the community
|
||||||
|
* Showing empathy towards other community members
|
||||||
|
|
||||||
|
Examples of unacceptable behavior by participants include:
|
||||||
|
|
||||||
|
* The use of sexualized language or imagery and unwelcome sexual attention or advances
|
||||||
|
* Trolling, insulting/derogatory comments, and personal or political attacks
|
||||||
|
* Public or private harassment
|
||||||
|
* Publishing others' private information, such as a physical or electronic address, without explicit permission
|
||||||
|
* Other conduct which could reasonably be considered inappropriate in a professional setting
|
||||||
|
|
||||||
|
## Our Responsibilities
|
||||||
|
|
||||||
|
Project maintainers are responsible for clarifying the standards of acceptable behavior and are expected to take appropriate and fair corrective action in response to any instances of unacceptable behavior.
|
||||||
|
|
||||||
|
Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, or to ban temporarily or permanently any contributor for other behaviors that they deem inappropriate, threatening, offensive, or harmful.
|
||||||
|
|
||||||
|
## Enforcement
|
||||||
|
|
||||||
|
Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting the project maintainers on this [Google form][gform]. The project maintainers will review and investigate all complaints, and will respond in a way that it deems appropriate to the circumstances. The project maintainers are obligated to maintain confidentiality with regard to the reporter of an incident. Further details of specific enforcement policies may be posted separately.
|
||||||
|
|
||||||
|
Project maintainers who do not follow or enforce the Code of Conduct in good faith may face temporary or permanent repercussions as determined by other members of the project.
|
||||||
|
|
||||||
|
## Attribution
|
||||||
|
|
||||||
|
This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, available at [http://contributor-covenant.org/version/1/4][version]
|
||||||
|
|
||||||
|
[gform]: https://forms.gle/dnKXMJL7VxdUjt3x5
|
||||||
|
[homepage]: http://contributor-covenant.org
|
||||||
|
[version]: http://contributor-covenant.org/version/1/4/
|
@ -12,8 +12,7 @@ If you want to make a big change it's better to discuss it first by opening an i
|
|||||||
|
|
||||||
## Pull Request process
|
## Pull Request process
|
||||||
|
|
||||||
Any pull request should be opened against the development branch of the target version. The development branch format is
|
Any pull request should be opened against the development branch `dev`.
|
||||||
as follows: `vx.y.z-dev`, for example: `v0.8.5-dev`.
|
|
||||||
|
|
||||||
All pull requests should pass the checks written in `build_and_test.sh`, so it's recommended to run this script before
|
All pull requests should pass the checks written in `build_and_test.sh`, so it's recommended to run this script before
|
||||||
submitting your PR.
|
submitting your PR.
|
25
README.md
25
README.md
@ -1,16 +1,15 @@
|
|||||||
|
# DEPRECATED
|
||||||
|
|
||||||
Kaspad
|
The full node reference implementation was [rewritten in Rust](https://github.com/kaspanet/rusty-kaspa), as a result, the Go implementation is now deprecated.
|
||||||
====
|
|
||||||
Warning: This is pre-alpha software. There's no guarantee anything works.
|
PLEASE NOTE: Any pull requests or issues that will be opened in this repository will be closed without treatment, except for issues or pull requests related to the kaspawallet, which remains maintained. In any other case, please use the [Rust implementation](https://github.com/kaspanet/rusty-kaspa) instead.
|
||||||
====
|
|
||||||
|
# Kaspad
|
||||||
|
|
||||||
[](https://choosealicense.com/licenses/isc/)
|
[](https://choosealicense.com/licenses/isc/)
|
||||||
[](http://godoc.org/github.com/kaspanet/kaspad)
|
[](http://godoc.org/github.com/kaspanet/kaspad)
|
||||||
|
|
||||||
Kaspad is the reference full node Kaspa implementation written in Go (golang).
|
Kaspad was the reference full node Kaspa implementation written in Go (golang).
|
||||||
|
|
||||||
This project is currently under active development and is in a pre-Alpha state.
|
|
||||||
Some things still don't work and APIs are far from finalized. The code is provided for reference only.
|
|
||||||
|
|
||||||
## What is kaspa
|
## What is kaspa
|
||||||
|
|
||||||
@ -18,7 +17,7 @@ Kaspa is an attempt at a proof-of-work cryptocurrency with instant confirmations
|
|||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
Go 1.15 or later.
|
Go 1.23 or later.
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
@ -45,7 +44,6 @@ $ go install . ./cmd/...
|
|||||||
not already add the bin directory to your system path during Go installation,
|
not already add the bin directory to your system path during Go installation,
|
||||||
you are encouraged to do so now.
|
you are encouraged to do so now.
|
||||||
|
|
||||||
|
|
||||||
## Getting Started
|
## Getting Started
|
||||||
|
|
||||||
Kaspad has several configuration options available to tweak how it runs, but all
|
Kaspad has several configuration options available to tweak how it runs, but all
|
||||||
@ -56,16 +54,19 @@ $ kaspad
|
|||||||
```
|
```
|
||||||
|
|
||||||
## Discord
|
## Discord
|
||||||
Join our discord server using the following link: https://discord.gg/WmGhhzk
|
|
||||||
|
Join our discord server using the following link: https://discord.gg/YNYnNN5Pf2
|
||||||
|
|
||||||
## Issue Tracker
|
## Issue Tracker
|
||||||
|
|
||||||
The [integrated github issue tracker](https://github.com/kaspanet/kaspad/issues)
|
The [integrated github issue tracker](https://github.com/kaspanet/kaspad/issues)
|
||||||
is used for this project.
|
is used for this project.
|
||||||
|
|
||||||
|
Issue priorities may be seen at https://github.com/orgs/kaspanet/projects/4
|
||||||
|
|
||||||
## Documentation
|
## Documentation
|
||||||
|
|
||||||
The documentation is a work-in-progress.
|
The [documentation](https://github.com/kaspanet/docs) is a work-in-progress
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
50
app/app.go
50
app/app.go
@ -7,22 +7,23 @@ import (
|
|||||||
"runtime"
|
"runtime"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/kaspanet/kaspad/infrastructure/config"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/db/database"
|
"github.com/kaspanet/kaspad/infrastructure/db/database"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/db/database/ldb"
|
"github.com/kaspanet/kaspad/infrastructure/db/database/ldb"
|
||||||
|
"github.com/kaspanet/kaspad/infrastructure/logger"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/os/signal"
|
|
||||||
"github.com/kaspanet/kaspad/util/profiling"
|
|
||||||
"github.com/kaspanet/kaspad/version"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/util/panics"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/config"
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/os/execenv"
|
"github.com/kaspanet/kaspad/infrastructure/os/execenv"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/os/limits"
|
"github.com/kaspanet/kaspad/infrastructure/os/limits"
|
||||||
|
"github.com/kaspanet/kaspad/infrastructure/os/signal"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/os/winservice"
|
"github.com/kaspanet/kaspad/infrastructure/os/winservice"
|
||||||
|
"github.com/kaspanet/kaspad/util/panics"
|
||||||
|
"github.com/kaspanet/kaspad/util/profiling"
|
||||||
|
"github.com/kaspanet/kaspad/version"
|
||||||
)
|
)
|
||||||
|
|
||||||
const leveldbCacheSizeMiB = 256
|
const (
|
||||||
|
leveldbCacheSizeMiB = 256
|
||||||
|
defaultDataDirname = "datadir2"
|
||||||
|
)
|
||||||
|
|
||||||
var desiredLimits = &limits.DesiredLimits{
|
var desiredLimits = &limits.DesiredLimits{
|
||||||
FileLimitWant: 2048,
|
FileLimitWant: 2048,
|
||||||
@ -51,6 +52,7 @@ func StartApp() error {
|
|||||||
fmt.Fprintln(os.Stderr, err)
|
fmt.Fprintln(os.Stderr, err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
defer logger.BackendLog.Close()
|
||||||
defer panics.HandlePanic(log, "MAIN", nil)
|
defer panics.HandlePanic(log, "MAIN", nil)
|
||||||
|
|
||||||
app := &kaspadApp{cfg: cfg}
|
app := &kaspadApp{cfg: cfg}
|
||||||
@ -85,12 +87,7 @@ func (app *kaspadApp) main(startedChan chan<- struct{}) error {
|
|||||||
if app.cfg.Profile != "" {
|
if app.cfg.Profile != "" {
|
||||||
profiling.Start(app.cfg.Profile, log)
|
profiling.Start(app.cfg.Profile, log)
|
||||||
}
|
}
|
||||||
|
profiling.TrackHeap(app.cfg.AppDir, log)
|
||||||
// Perform upgrades to kaspad as new versions require it.
|
|
||||||
if err := doUpgrades(); err != nil {
|
|
||||||
log.Error(err)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return now if an interrupt signal was triggered.
|
// Return now if an interrupt signal was triggered.
|
||||||
if signal.InterruptRequested(interrupt) {
|
if signal.InterruptRequested(interrupt) {
|
||||||
@ -108,7 +105,7 @@ func (app *kaspadApp) main(startedChan chan<- struct{}) error {
|
|||||||
// Open the database
|
// Open the database
|
||||||
databaseContext, err := openDB(app.cfg)
|
databaseContext, err := openDB(app.cfg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error(err)
|
log.Errorf("Loading database failed: %+v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -164,15 +161,9 @@ func (app *kaspadApp) main(startedChan chan<- struct{}) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// doUpgrades performs upgrades to kaspad as new versions require it.
|
|
||||||
// currently it's a placeholder we got from kaspad upstream, that does nothing
|
|
||||||
func doUpgrades() error {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// dbPath returns the path to the block database given a database type.
|
// dbPath returns the path to the block database given a database type.
|
||||||
func databasePath(cfg *config.Config) string {
|
func databasePath(cfg *config.Config) string {
|
||||||
return filepath.Join(cfg.DataDir, "db")
|
return filepath.Join(cfg.AppDir, defaultDataDirname)
|
||||||
}
|
}
|
||||||
|
|
||||||
func removeDatabase(cfg *config.Config) error {
|
func removeDatabase(cfg *config.Config) error {
|
||||||
@ -182,6 +173,17 @@ func removeDatabase(cfg *config.Config) error {
|
|||||||
|
|
||||||
func openDB(cfg *config.Config) (database.Database, error) {
|
func openDB(cfg *config.Config) (database.Database, error) {
|
||||||
dbPath := databasePath(cfg)
|
dbPath := databasePath(cfg)
|
||||||
|
|
||||||
|
err := checkDatabaseVersion(dbPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
log.Infof("Loading database from '%s'", dbPath)
|
log.Infof("Loading database from '%s'", dbPath)
|
||||||
return ldb.NewLevelDB(dbPath, leveldbCacheSizeMiB)
|
db, err := ldb.NewLevelDB(dbPath, leveldbCacheSizeMiB)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return db, nil
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,7 @@ supported kaspa messages to and from the appmessage. This package does not deal
|
|||||||
with the specifics of message handling such as what to do when a message is
|
with the specifics of message handling such as what to do when a message is
|
||||||
received. This provides the caller with a high level of flexibility.
|
received. This provides the caller with a high level of flexibility.
|
||||||
|
|
||||||
Kaspa Message Overview
|
# Kaspa Message Overview
|
||||||
|
|
||||||
The kaspa protocol consists of exchanging messages between peers. Each
|
The kaspa protocol consists of exchanging messages between peers. Each
|
||||||
message is preceded by a header which identifies information about it such as
|
message is preceded by a header which identifies information about it such as
|
||||||
@ -22,7 +22,7 @@ messages, all of the details of marshalling and unmarshalling to and from the
|
|||||||
appmessage using kaspa encoding are handled so the caller doesn't have to concern
|
appmessage using kaspa encoding are handled so the caller doesn't have to concern
|
||||||
themselves with the specifics.
|
themselves with the specifics.
|
||||||
|
|
||||||
Message Interaction
|
# Message Interaction
|
||||||
|
|
||||||
The following provides a quick summary of how the kaspa messages are intended
|
The following provides a quick summary of how the kaspa messages are intended
|
||||||
to interact with one another. As stated above, these interactions are not
|
to interact with one another. As stated above, these interactions are not
|
||||||
@ -45,13 +45,13 @@ interactions in no particular order.
|
|||||||
notfound message (MsgNotFound)
|
notfound message (MsgNotFound)
|
||||||
ping message (MsgPing) pong message (MsgPong)
|
ping message (MsgPing) pong message (MsgPong)
|
||||||
|
|
||||||
Common Parameters
|
# Common Parameters
|
||||||
|
|
||||||
There are several common parameters that arise when using this package to read
|
There are several common parameters that arise when using this package to read
|
||||||
and write kaspa messages. The following sections provide a quick overview of
|
and write kaspa messages. The following sections provide a quick overview of
|
||||||
these parameters so the next sections can build on them.
|
these parameters so the next sections can build on them.
|
||||||
|
|
||||||
Protocol Version
|
# Protocol Version
|
||||||
|
|
||||||
The protocol version should be negotiated with the remote peer at a higher
|
The protocol version should be negotiated with the remote peer at a higher
|
||||||
level than this package via the version (MsgVersion) message exchange, however,
|
level than this package via the version (MsgVersion) message exchange, however,
|
||||||
@ -60,7 +60,7 @@ latest protocol version this package supports and is typically the value to use
|
|||||||
for all outbound connections before a potentially lower protocol version is
|
for all outbound connections before a potentially lower protocol version is
|
||||||
negotiated.
|
negotiated.
|
||||||
|
|
||||||
Kaspa Network
|
# Kaspa Network
|
||||||
|
|
||||||
The kaspa network is a magic number which is used to identify the start of a
|
The kaspa network is a magic number which is used to identify the start of a
|
||||||
message and which kaspa network the message applies to. This package provides
|
message and which kaspa network the message applies to. This package provides
|
||||||
@ -71,7 +71,7 @@ the following constants:
|
|||||||
appmessage.Simnet (Simulation test network)
|
appmessage.Simnet (Simulation test network)
|
||||||
appmessage.Devnet (Development network)
|
appmessage.Devnet (Development network)
|
||||||
|
|
||||||
Determining Message Type
|
# Determining Message Type
|
||||||
|
|
||||||
As discussed in the kaspa message overview section, this package reads
|
As discussed in the kaspa message overview section, this package reads
|
||||||
and writes kaspa messages using a generic interface named Message. In
|
and writes kaspa messages using a generic interface named Message. In
|
||||||
@ -89,7 +89,7 @@ switch or type assertion. An example of a type switch follows:
|
|||||||
fmt.Printf("Number of tx in block: %d", msg.Header.TxnCount)
|
fmt.Printf("Number of tx in block: %d", msg.Header.TxnCount)
|
||||||
}
|
}
|
||||||
|
|
||||||
Reading Messages
|
# Reading Messages
|
||||||
|
|
||||||
In order to unmarshall kaspa messages from the appmessage, use the ReadMessage
|
In order to unmarshall kaspa messages from the appmessage, use the ReadMessage
|
||||||
function. It accepts any io.Reader, but typically this will be a net.Conn to
|
function. It accepts any io.Reader, but typically this will be a net.Conn to
|
||||||
@ -104,7 +104,7 @@ a remote node running a kaspa peer. Example syntax is:
|
|||||||
// Log and handle the error
|
// Log and handle the error
|
||||||
}
|
}
|
||||||
|
|
||||||
Writing Messages
|
# Writing Messages
|
||||||
|
|
||||||
In order to marshall kaspa messages to the appmessage, use the WriteMessage
|
In order to marshall kaspa messages to the appmessage, use the WriteMessage
|
||||||
function. It accepts any io.Writer, but typically this will be a net.Conn to
|
function. It accepts any io.Writer, but typically this will be a net.Conn to
|
||||||
@ -122,7 +122,7 @@ from a remote peer is:
|
|||||||
// Log and handle the error
|
// Log and handle the error
|
||||||
}
|
}
|
||||||
|
|
||||||
Errors
|
# Errors
|
||||||
|
|
||||||
Errors returned by this package are either the raw errors provided by underlying
|
Errors returned by this package are either the raw errors provided by underlying
|
||||||
calls to read/write from streams such as io.EOF, io.ErrUnexpectedEOF, and
|
calls to read/write from streams such as io.EOF, io.ErrUnexpectedEOF, and
|
||||||
|
@ -2,7 +2,12 @@ package appmessage
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
|
"math/big"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/blockheader"
|
"github.com/kaspanet/kaspad/domain/consensus/utils/blockheader"
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/utils/hashes"
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/utxo"
|
"github.com/kaspanet/kaspad/domain/consensus/utils/utxo"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
@ -27,13 +32,17 @@ func DomainBlockToMsgBlock(domainBlock *externalapi.DomainBlock) *MsgBlock {
|
|||||||
func DomainBlockHeaderToBlockHeader(domainBlockHeader externalapi.BlockHeader) *MsgBlockHeader {
|
func DomainBlockHeaderToBlockHeader(domainBlockHeader externalapi.BlockHeader) *MsgBlockHeader {
|
||||||
return &MsgBlockHeader{
|
return &MsgBlockHeader{
|
||||||
Version: domainBlockHeader.Version(),
|
Version: domainBlockHeader.Version(),
|
||||||
ParentHashes: domainBlockHeader.ParentHashes(),
|
Parents: domainBlockHeader.Parents(),
|
||||||
HashMerkleRoot: domainBlockHeader.HashMerkleRoot(),
|
HashMerkleRoot: domainBlockHeader.HashMerkleRoot(),
|
||||||
AcceptedIDMerkleRoot: domainBlockHeader.AcceptedIDMerkleRoot(),
|
AcceptedIDMerkleRoot: domainBlockHeader.AcceptedIDMerkleRoot(),
|
||||||
UTXOCommitment: domainBlockHeader.UTXOCommitment(),
|
UTXOCommitment: domainBlockHeader.UTXOCommitment(),
|
||||||
Timestamp: mstime.UnixMilliseconds(domainBlockHeader.TimeInMilliseconds()),
|
Timestamp: mstime.UnixMilliseconds(domainBlockHeader.TimeInMilliseconds()),
|
||||||
Bits: domainBlockHeader.Bits(),
|
Bits: domainBlockHeader.Bits(),
|
||||||
Nonce: domainBlockHeader.Nonce(),
|
Nonce: domainBlockHeader.Nonce(),
|
||||||
|
BlueScore: domainBlockHeader.BlueScore(),
|
||||||
|
DAAScore: domainBlockHeader.DAAScore(),
|
||||||
|
BlueWork: domainBlockHeader.BlueWork(),
|
||||||
|
PruningPoint: domainBlockHeader.PruningPoint(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -54,13 +63,17 @@ func MsgBlockToDomainBlock(msgBlock *MsgBlock) *externalapi.DomainBlock {
|
|||||||
func BlockHeaderToDomainBlockHeader(blockHeader *MsgBlockHeader) externalapi.BlockHeader {
|
func BlockHeaderToDomainBlockHeader(blockHeader *MsgBlockHeader) externalapi.BlockHeader {
|
||||||
return blockheader.NewImmutableBlockHeader(
|
return blockheader.NewImmutableBlockHeader(
|
||||||
blockHeader.Version,
|
blockHeader.Version,
|
||||||
blockHeader.ParentHashes,
|
blockHeader.Parents,
|
||||||
blockHeader.HashMerkleRoot,
|
blockHeader.HashMerkleRoot,
|
||||||
blockHeader.AcceptedIDMerkleRoot,
|
blockHeader.AcceptedIDMerkleRoot,
|
||||||
blockHeader.UTXOCommitment,
|
blockHeader.UTXOCommitment,
|
||||||
blockHeader.Timestamp.UnixMilliseconds(),
|
blockHeader.Timestamp.UnixMilliseconds(),
|
||||||
blockHeader.Bits,
|
blockHeader.Bits,
|
||||||
blockHeader.Nonce,
|
blockHeader.Nonce,
|
||||||
|
blockHeader.DAAScore,
|
||||||
|
blockHeader.BlueScore,
|
||||||
|
blockHeader.BlueWork,
|
||||||
|
blockHeader.PruningPoint,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -83,7 +96,6 @@ func DomainTransactionToMsgTx(domainTransaction *externalapi.DomainTransaction)
|
|||||||
LockTime: domainTransaction.LockTime,
|
LockTime: domainTransaction.LockTime,
|
||||||
SubnetworkID: domainTransaction.SubnetworkID,
|
SubnetworkID: domainTransaction.SubnetworkID,
|
||||||
Gas: domainTransaction.Gas,
|
Gas: domainTransaction.Gas,
|
||||||
PayloadHash: domainTransaction.PayloadHash,
|
|
||||||
Payload: domainTransaction.Payload,
|
Payload: domainTransaction.Payload,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -100,6 +112,7 @@ func domainTransactionInputToTxIn(domainTransactionInput *externalapi.DomainTran
|
|||||||
PreviousOutpoint: *domainOutpointToOutpoint(domainTransactionInput.PreviousOutpoint),
|
PreviousOutpoint: *domainOutpointToOutpoint(domainTransactionInput.PreviousOutpoint),
|
||||||
SignatureScript: domainTransactionInput.SignatureScript,
|
SignatureScript: domainTransactionInput.SignatureScript,
|
||||||
Sequence: domainTransactionInput.Sequence,
|
Sequence: domainTransactionInput.Sequence,
|
||||||
|
SigOpCount: domainTransactionInput.SigOpCount,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -133,7 +146,6 @@ func MsgTxToDomainTransaction(msgTx *MsgTx) *externalapi.DomainTransaction {
|
|||||||
LockTime: msgTx.LockTime,
|
LockTime: msgTx.LockTime,
|
||||||
SubnetworkID: msgTx.SubnetworkID,
|
SubnetworkID: msgTx.SubnetworkID,
|
||||||
Gas: msgTx.Gas,
|
Gas: msgTx.Gas,
|
||||||
PayloadHash: msgTx.PayloadHash,
|
|
||||||
Payload: payload,
|
Payload: payload,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -149,6 +161,7 @@ func txInToDomainTransactionInput(txIn *TxIn) *externalapi.DomainTransactionInpu
|
|||||||
return &externalapi.DomainTransactionInput{
|
return &externalapi.DomainTransactionInput{
|
||||||
PreviousOutpoint: *outpointToDomainOutpoint(&txIn.PreviousOutpoint), //TODO
|
PreviousOutpoint: *outpointToDomainOutpoint(&txIn.PreviousOutpoint), //TODO
|
||||||
SignatureScript: txIn.SignatureScript,
|
SignatureScript: txIn.SignatureScript,
|
||||||
|
SigOpCount: txIn.SigOpCount,
|
||||||
Sequence: txIn.Sequence,
|
Sequence: txIn.Sequence,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -164,18 +177,10 @@ func outpointToDomainOutpoint(outpoint *Outpoint) *externalapi.DomainOutpoint {
|
|||||||
func RPCTransactionToDomainTransaction(rpcTransaction *RPCTransaction) (*externalapi.DomainTransaction, error) {
|
func RPCTransactionToDomainTransaction(rpcTransaction *RPCTransaction) (*externalapi.DomainTransaction, error) {
|
||||||
inputs := make([]*externalapi.DomainTransactionInput, len(rpcTransaction.Inputs))
|
inputs := make([]*externalapi.DomainTransactionInput, len(rpcTransaction.Inputs))
|
||||||
for i, input := range rpcTransaction.Inputs {
|
for i, input := range rpcTransaction.Inputs {
|
||||||
transactionIDBytes, err := hex.DecodeString(input.PreviousOutpoint.TransactionID)
|
previousOutpoint, err := RPCOutpointToDomainOutpoint(input.PreviousOutpoint)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
transactionID, err := transactionid.FromBytes(transactionIDBytes)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
previousOutpoint := &externalapi.DomainOutpoint{
|
|
||||||
TransactionID: *transactionID,
|
|
||||||
Index: input.PreviousOutpoint.Index,
|
|
||||||
}
|
|
||||||
signatureScript, err := hex.DecodeString(input.SignatureScript)
|
signatureScript, err := hex.DecodeString(input.SignatureScript)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -184,6 +189,7 @@ func RPCTransactionToDomainTransaction(rpcTransaction *RPCTransaction) (*externa
|
|||||||
PreviousOutpoint: *previousOutpoint,
|
PreviousOutpoint: *previousOutpoint,
|
||||||
SignatureScript: signatureScript,
|
SignatureScript: signatureScript,
|
||||||
Sequence: input.Sequence,
|
Sequence: input.Sequence,
|
||||||
|
SigOpCount: input.SigOpCount,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
outputs := make([]*externalapi.DomainTransactionOutput, len(rpcTransaction.Outputs))
|
outputs := make([]*externalapi.DomainTransactionOutput, len(rpcTransaction.Outputs))
|
||||||
@ -198,19 +204,7 @@ func RPCTransactionToDomainTransaction(rpcTransaction *RPCTransaction) (*externa
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
subnetworkIDBytes, err := hex.DecodeString(rpcTransaction.SubnetworkID)
|
subnetworkID, err := subnetworks.FromString(rpcTransaction.SubnetworkID)
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
subnetworkID, err := subnetworks.FromBytes(subnetworkIDBytes)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
payloadHashBytes, err := hex.DecodeString(rpcTransaction.PayloadHash)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
payloadHash, err := externalapi.NewDomainHashFromByteSlice(payloadHashBytes)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -225,12 +219,42 @@ func RPCTransactionToDomainTransaction(rpcTransaction *RPCTransaction) (*externa
|
|||||||
Outputs: outputs,
|
Outputs: outputs,
|
||||||
LockTime: rpcTransaction.LockTime,
|
LockTime: rpcTransaction.LockTime,
|
||||||
SubnetworkID: *subnetworkID,
|
SubnetworkID: *subnetworkID,
|
||||||
Gas: rpcTransaction.LockTime,
|
Gas: rpcTransaction.Gas,
|
||||||
PayloadHash: *payloadHash,
|
MassCommitment: rpcTransaction.Mass,
|
||||||
Payload: payload,
|
Payload: payload,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RPCOutpointToDomainOutpoint converts RPCOutpoint to DomainOutpoint
|
||||||
|
func RPCOutpointToDomainOutpoint(outpoint *RPCOutpoint) (*externalapi.DomainOutpoint, error) {
|
||||||
|
transactionID, err := transactionid.FromString(outpoint.TransactionID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &externalapi.DomainOutpoint{
|
||||||
|
TransactionID: *transactionID,
|
||||||
|
Index: outpoint.Index,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RPCUTXOEntryToUTXOEntry converts RPCUTXOEntry to UTXOEntry
|
||||||
|
func RPCUTXOEntryToUTXOEntry(entry *RPCUTXOEntry) (externalapi.UTXOEntry, error) {
|
||||||
|
script, err := hex.DecodeString(entry.ScriptPublicKey.Script)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return utxo.NewUTXOEntry(
|
||||||
|
entry.Amount,
|
||||||
|
&externalapi.ScriptPublicKey{
|
||||||
|
Script: script,
|
||||||
|
Version: entry.ScriptPublicKey.Version,
|
||||||
|
},
|
||||||
|
entry.IsCoinbase,
|
||||||
|
entry.BlockDAAScore,
|
||||||
|
), nil
|
||||||
|
}
|
||||||
|
|
||||||
// DomainTransactionToRPCTransaction converts DomainTransactions to RPCTransactions
|
// DomainTransactionToRPCTransaction converts DomainTransactions to RPCTransactions
|
||||||
func DomainTransactionToRPCTransaction(transaction *externalapi.DomainTransaction) *RPCTransaction {
|
func DomainTransactionToRPCTransaction(transaction *externalapi.DomainTransaction) *RPCTransaction {
|
||||||
inputs := make([]*RPCTransactionInput, len(transaction.Inputs))
|
inputs := make([]*RPCTransactionInput, len(transaction.Inputs))
|
||||||
@ -245,6 +269,7 @@ func DomainTransactionToRPCTransaction(transaction *externalapi.DomainTransactio
|
|||||||
PreviousOutpoint: previousOutpoint,
|
PreviousOutpoint: previousOutpoint,
|
||||||
SignatureScript: signatureScript,
|
SignatureScript: signatureScript,
|
||||||
Sequence: input.Sequence,
|
Sequence: input.Sequence,
|
||||||
|
SigOpCount: input.SigOpCount,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
outputs := make([]*RPCTransactionOutput, len(transaction.Outputs))
|
outputs := make([]*RPCTransactionOutput, len(transaction.Outputs))
|
||||||
@ -255,8 +280,7 @@ func DomainTransactionToRPCTransaction(transaction *externalapi.DomainTransactio
|
|||||||
ScriptPublicKey: &RPCScriptPublicKey{Script: scriptPublicKey, Version: output.ScriptPublicKey.Version},
|
ScriptPublicKey: &RPCScriptPublicKey{Script: scriptPublicKey, Version: output.ScriptPublicKey.Version},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
subnetworkID := hex.EncodeToString(transaction.SubnetworkID[:])
|
subnetworkID := transaction.SubnetworkID.String()
|
||||||
payloadHash := transaction.PayloadHash.String()
|
|
||||||
payload := hex.EncodeToString(transaction.Payload)
|
payload := hex.EncodeToString(transaction.Payload)
|
||||||
return &RPCTransaction{
|
return &RPCTransaction{
|
||||||
Version: transaction.Version,
|
Version: transaction.Version,
|
||||||
@ -264,8 +288,8 @@ func DomainTransactionToRPCTransaction(transaction *externalapi.DomainTransactio
|
|||||||
Outputs: outputs,
|
Outputs: outputs,
|
||||||
LockTime: transaction.LockTime,
|
LockTime: transaction.LockTime,
|
||||||
SubnetworkID: subnetworkID,
|
SubnetworkID: subnetworkID,
|
||||||
Gas: transaction.LockTime,
|
Gas: transaction.Gas,
|
||||||
PayloadHash: payloadHash,
|
Mass: transaction.MassCommitment,
|
||||||
Payload: payload,
|
Payload: payload,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -277,7 +301,14 @@ func OutpointAndUTXOEntryPairsToDomainOutpointAndUTXOEntryPairs(
|
|||||||
|
|
||||||
domainOutpointAndUTXOEntryPairs := make([]*externalapi.OutpointAndUTXOEntryPair, len(outpointAndUTXOEntryPairs))
|
domainOutpointAndUTXOEntryPairs := make([]*externalapi.OutpointAndUTXOEntryPair, len(outpointAndUTXOEntryPairs))
|
||||||
for i, outpointAndUTXOEntryPair := range outpointAndUTXOEntryPairs {
|
for i, outpointAndUTXOEntryPair := range outpointAndUTXOEntryPairs {
|
||||||
domainOutpointAndUTXOEntryPairs[i] = &externalapi.OutpointAndUTXOEntryPair{
|
domainOutpointAndUTXOEntryPairs[i] = outpointAndUTXOEntryPairToDomainOutpointAndUTXOEntryPair(outpointAndUTXOEntryPair)
|
||||||
|
}
|
||||||
|
return domainOutpointAndUTXOEntryPairs
|
||||||
|
}
|
||||||
|
|
||||||
|
func outpointAndUTXOEntryPairToDomainOutpointAndUTXOEntryPair(
|
||||||
|
outpointAndUTXOEntryPair *OutpointAndUTXOEntryPair) *externalapi.OutpointAndUTXOEntryPair {
|
||||||
|
return &externalapi.OutpointAndUTXOEntryPair{
|
||||||
Outpoint: &externalapi.DomainOutpoint{
|
Outpoint: &externalapi.DomainOutpoint{
|
||||||
TransactionID: outpointAndUTXOEntryPair.Outpoint.TxID,
|
TransactionID: outpointAndUTXOEntryPair.Outpoint.TxID,
|
||||||
Index: outpointAndUTXOEntryPair.Outpoint.Index,
|
Index: outpointAndUTXOEntryPair.Outpoint.Index,
|
||||||
@ -286,11 +317,9 @@ func OutpointAndUTXOEntryPairsToDomainOutpointAndUTXOEntryPairs(
|
|||||||
outpointAndUTXOEntryPair.UTXOEntry.Amount,
|
outpointAndUTXOEntryPair.UTXOEntry.Amount,
|
||||||
outpointAndUTXOEntryPair.UTXOEntry.ScriptPublicKey,
|
outpointAndUTXOEntryPair.UTXOEntry.ScriptPublicKey,
|
||||||
outpointAndUTXOEntryPair.UTXOEntry.IsCoinbase,
|
outpointAndUTXOEntryPair.UTXOEntry.IsCoinbase,
|
||||||
outpointAndUTXOEntryPair.UTXOEntry.BlockBlueScore,
|
outpointAndUTXOEntryPair.UTXOEntry.BlockDAAScore,
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
return domainOutpointAndUTXOEntryPairs
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// DomainOutpointAndUTXOEntryPairsToOutpointAndUTXOEntryPairs converts
|
// DomainOutpointAndUTXOEntryPairsToOutpointAndUTXOEntryPairs converts
|
||||||
@ -309,9 +338,267 @@ func DomainOutpointAndUTXOEntryPairsToOutpointAndUTXOEntryPairs(
|
|||||||
Amount: outpointAndUTXOEntryPair.UTXOEntry.Amount(),
|
Amount: outpointAndUTXOEntryPair.UTXOEntry.Amount(),
|
||||||
ScriptPublicKey: outpointAndUTXOEntryPair.UTXOEntry.ScriptPublicKey(),
|
ScriptPublicKey: outpointAndUTXOEntryPair.UTXOEntry.ScriptPublicKey(),
|
||||||
IsCoinbase: outpointAndUTXOEntryPair.UTXOEntry.IsCoinbase(),
|
IsCoinbase: outpointAndUTXOEntryPair.UTXOEntry.IsCoinbase(),
|
||||||
BlockBlueScore: outpointAndUTXOEntryPair.UTXOEntry.BlockBlueScore(),
|
BlockDAAScore: outpointAndUTXOEntryPair.UTXOEntry.BlockDAAScore(),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return domainOutpointAndUTXOEntryPairs
|
return domainOutpointAndUTXOEntryPairs
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DomainBlockToRPCBlock converts DomainBlocks to RPCBlocks
|
||||||
|
func DomainBlockToRPCBlock(block *externalapi.DomainBlock) *RPCBlock {
|
||||||
|
parents := make([]*RPCBlockLevelParents, len(block.Header.Parents()))
|
||||||
|
for i, blockLevelParents := range block.Header.Parents() {
|
||||||
|
parents[i] = &RPCBlockLevelParents{
|
||||||
|
ParentHashes: hashes.ToStrings(blockLevelParents),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
header := &RPCBlockHeader{
|
||||||
|
Version: uint32(block.Header.Version()),
|
||||||
|
Parents: parents,
|
||||||
|
HashMerkleRoot: block.Header.HashMerkleRoot().String(),
|
||||||
|
AcceptedIDMerkleRoot: block.Header.AcceptedIDMerkleRoot().String(),
|
||||||
|
UTXOCommitment: block.Header.UTXOCommitment().String(),
|
||||||
|
Timestamp: block.Header.TimeInMilliseconds(),
|
||||||
|
Bits: block.Header.Bits(),
|
||||||
|
Nonce: block.Header.Nonce(),
|
||||||
|
DAAScore: block.Header.DAAScore(),
|
||||||
|
BlueScore: block.Header.BlueScore(),
|
||||||
|
BlueWork: block.Header.BlueWork().Text(16),
|
||||||
|
PruningPoint: block.Header.PruningPoint().String(),
|
||||||
|
}
|
||||||
|
transactions := make([]*RPCTransaction, len(block.Transactions))
|
||||||
|
for i, transaction := range block.Transactions {
|
||||||
|
transactions[i] = DomainTransactionToRPCTransaction(transaction)
|
||||||
|
}
|
||||||
|
return &RPCBlock{
|
||||||
|
Header: header,
|
||||||
|
Transactions: transactions,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// RPCBlockToDomainBlock converts `block` into a DomainBlock
|
||||||
|
func RPCBlockToDomainBlock(block *RPCBlock) (*externalapi.DomainBlock, error) {
|
||||||
|
parents := make([]externalapi.BlockLevelParents, len(block.Header.Parents))
|
||||||
|
for i, blockLevelParents := range block.Header.Parents {
|
||||||
|
parents[i] = make(externalapi.BlockLevelParents, len(blockLevelParents.ParentHashes))
|
||||||
|
for j, parentHash := range blockLevelParents.ParentHashes {
|
||||||
|
var err error
|
||||||
|
parents[i][j], err = externalapi.NewDomainHashFromString(parentHash)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
hashMerkleRoot, err := externalapi.NewDomainHashFromString(block.Header.HashMerkleRoot)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
acceptedIDMerkleRoot, err := externalapi.NewDomainHashFromString(block.Header.AcceptedIDMerkleRoot)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
utxoCommitment, err := externalapi.NewDomainHashFromString(block.Header.UTXOCommitment)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
blueWork, success := new(big.Int).SetString(block.Header.BlueWork, 16)
|
||||||
|
if !success {
|
||||||
|
return nil, errors.Errorf("failed to parse blue work: %s", block.Header.BlueWork)
|
||||||
|
}
|
||||||
|
pruningPoint, err := externalapi.NewDomainHashFromString(block.Header.PruningPoint)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
header := blockheader.NewImmutableBlockHeader(
|
||||||
|
uint16(block.Header.Version),
|
||||||
|
parents,
|
||||||
|
hashMerkleRoot,
|
||||||
|
acceptedIDMerkleRoot,
|
||||||
|
utxoCommitment,
|
||||||
|
block.Header.Timestamp,
|
||||||
|
block.Header.Bits,
|
||||||
|
block.Header.Nonce,
|
||||||
|
block.Header.DAAScore,
|
||||||
|
block.Header.BlueScore,
|
||||||
|
blueWork,
|
||||||
|
pruningPoint)
|
||||||
|
transactions := make([]*externalapi.DomainTransaction, len(block.Transactions))
|
||||||
|
for i, transaction := range block.Transactions {
|
||||||
|
domainTransaction, err := RPCTransactionToDomainTransaction(transaction)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
transactions[i] = domainTransaction
|
||||||
|
}
|
||||||
|
return &externalapi.DomainBlock{
|
||||||
|
Header: header,
|
||||||
|
Transactions: transactions,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// BlockWithTrustedDataToDomainBlockWithTrustedData converts *MsgBlockWithTrustedData to *externalapi.BlockWithTrustedData
|
||||||
|
func BlockWithTrustedDataToDomainBlockWithTrustedData(block *MsgBlockWithTrustedData) *externalapi.BlockWithTrustedData {
|
||||||
|
daaWindow := make([]*externalapi.TrustedDataDataDAAHeader, len(block.DAAWindow))
|
||||||
|
for i, daaBlock := range block.DAAWindow {
|
||||||
|
daaWindow[i] = &externalapi.TrustedDataDataDAAHeader{
|
||||||
|
Header: BlockHeaderToDomainBlockHeader(&daaBlock.Block.Header),
|
||||||
|
GHOSTDAGData: ghostdagDataToDomainGHOSTDAGData(daaBlock.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ghostdagData := make([]*externalapi.BlockGHOSTDAGDataHashPair, len(block.GHOSTDAGData))
|
||||||
|
for i, datum := range block.GHOSTDAGData {
|
||||||
|
ghostdagData[i] = &externalapi.BlockGHOSTDAGDataHashPair{
|
||||||
|
Hash: datum.Hash,
|
||||||
|
GHOSTDAGData: ghostdagDataToDomainGHOSTDAGData(datum.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &externalapi.BlockWithTrustedData{
|
||||||
|
Block: MsgBlockToDomainBlock(block.Block),
|
||||||
|
DAAWindow: daaWindow,
|
||||||
|
GHOSTDAGData: ghostdagData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TrustedDataDataDAABlockV4ToTrustedDataDataDAAHeader converts *TrustedDataDAAHeader to *externalapi.TrustedDataDataDAAHeader
|
||||||
|
func TrustedDataDataDAABlockV4ToTrustedDataDataDAAHeader(daaBlock *TrustedDataDAAHeader) *externalapi.TrustedDataDataDAAHeader {
|
||||||
|
return &externalapi.TrustedDataDataDAAHeader{
|
||||||
|
Header: BlockHeaderToDomainBlockHeader(daaBlock.Header),
|
||||||
|
GHOSTDAGData: ghostdagDataToDomainGHOSTDAGData(daaBlock.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GHOSTDAGHashPairToDomainGHOSTDAGHashPair converts *BlockGHOSTDAGDataHashPair to *externalapi.BlockGHOSTDAGDataHashPair
|
||||||
|
func GHOSTDAGHashPairToDomainGHOSTDAGHashPair(datum *BlockGHOSTDAGDataHashPair) *externalapi.BlockGHOSTDAGDataHashPair {
|
||||||
|
return &externalapi.BlockGHOSTDAGDataHashPair{
|
||||||
|
Hash: datum.Hash,
|
||||||
|
GHOSTDAGData: ghostdagDataToDomainGHOSTDAGData(datum.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ghostdagDataToDomainGHOSTDAGData(data *BlockGHOSTDAGData) *externalapi.BlockGHOSTDAGData {
|
||||||
|
bluesAnticoneSizes := make(map[externalapi.DomainHash]externalapi.KType, len(data.BluesAnticoneSizes))
|
||||||
|
for _, pair := range data.BluesAnticoneSizes {
|
||||||
|
bluesAnticoneSizes[*pair.BlueHash] = pair.AnticoneSize
|
||||||
|
}
|
||||||
|
return externalapi.NewBlockGHOSTDAGData(
|
||||||
|
data.BlueScore,
|
||||||
|
data.BlueWork,
|
||||||
|
data.SelectedParent,
|
||||||
|
data.MergeSetBlues,
|
||||||
|
data.MergeSetReds,
|
||||||
|
bluesAnticoneSizes,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func domainGHOSTDAGDataGHOSTDAGData(data *externalapi.BlockGHOSTDAGData) *BlockGHOSTDAGData {
|
||||||
|
bluesAnticoneSizes := make([]*BluesAnticoneSizes, 0, len(data.BluesAnticoneSizes()))
|
||||||
|
for blueHash, anticoneSize := range data.BluesAnticoneSizes() {
|
||||||
|
blueHashCopy := blueHash
|
||||||
|
bluesAnticoneSizes = append(bluesAnticoneSizes, &BluesAnticoneSizes{
|
||||||
|
BlueHash: &blueHashCopy,
|
||||||
|
AnticoneSize: anticoneSize,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return &BlockGHOSTDAGData{
|
||||||
|
BlueScore: data.BlueScore(),
|
||||||
|
BlueWork: data.BlueWork(),
|
||||||
|
SelectedParent: data.SelectedParent(),
|
||||||
|
MergeSetBlues: data.MergeSetBlues(),
|
||||||
|
MergeSetReds: data.MergeSetReds(),
|
||||||
|
BluesAnticoneSizes: bluesAnticoneSizes,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DomainBlockWithTrustedDataToBlockWithTrustedData converts *externalapi.BlockWithTrustedData to *MsgBlockWithTrustedData
|
||||||
|
func DomainBlockWithTrustedDataToBlockWithTrustedData(block *externalapi.BlockWithTrustedData) *MsgBlockWithTrustedData {
|
||||||
|
daaWindow := make([]*TrustedDataDataDAABlock, len(block.DAAWindow))
|
||||||
|
for i, daaBlock := range block.DAAWindow {
|
||||||
|
daaWindow[i] = &TrustedDataDataDAABlock{
|
||||||
|
Block: &MsgBlock{
|
||||||
|
Header: *DomainBlockHeaderToBlockHeader(daaBlock.Header),
|
||||||
|
},
|
||||||
|
GHOSTDAGData: domainGHOSTDAGDataGHOSTDAGData(daaBlock.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ghostdagData := make([]*BlockGHOSTDAGDataHashPair, len(block.GHOSTDAGData))
|
||||||
|
for i, datum := range block.GHOSTDAGData {
|
||||||
|
ghostdagData[i] = &BlockGHOSTDAGDataHashPair{
|
||||||
|
Hash: datum.Hash,
|
||||||
|
GHOSTDAGData: domainGHOSTDAGDataGHOSTDAGData(datum.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &MsgBlockWithTrustedData{
|
||||||
|
Block: DomainBlockToMsgBlock(block.Block),
|
||||||
|
DAAScore: block.Block.Header.DAAScore(),
|
||||||
|
DAAWindow: daaWindow,
|
||||||
|
GHOSTDAGData: ghostdagData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DomainBlockWithTrustedDataToBlockWithTrustedDataV4 converts a set of *externalapi.DomainBlock, daa window indices and ghostdag data indices
|
||||||
|
// to *MsgBlockWithTrustedDataV4
|
||||||
|
func DomainBlockWithTrustedDataToBlockWithTrustedDataV4(block *externalapi.DomainBlock, daaWindowIndices, ghostdagDataIndices []uint64) *MsgBlockWithTrustedDataV4 {
|
||||||
|
return &MsgBlockWithTrustedDataV4{
|
||||||
|
Block: DomainBlockToMsgBlock(block),
|
||||||
|
DAAWindowIndices: daaWindowIndices,
|
||||||
|
GHOSTDAGDataIndices: ghostdagDataIndices,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DomainTrustedDataToTrustedData converts *externalapi.BlockWithTrustedData to *MsgBlockWithTrustedData
|
||||||
|
func DomainTrustedDataToTrustedData(domainDAAWindow []*externalapi.TrustedDataDataDAAHeader, domainGHOSTDAGData []*externalapi.BlockGHOSTDAGDataHashPair) *MsgTrustedData {
|
||||||
|
daaWindow := make([]*TrustedDataDAAHeader, len(domainDAAWindow))
|
||||||
|
for i, daaBlock := range domainDAAWindow {
|
||||||
|
daaWindow[i] = &TrustedDataDAAHeader{
|
||||||
|
Header: DomainBlockHeaderToBlockHeader(daaBlock.Header),
|
||||||
|
GHOSTDAGData: domainGHOSTDAGDataGHOSTDAGData(daaBlock.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ghostdagData := make([]*BlockGHOSTDAGDataHashPair, len(domainGHOSTDAGData))
|
||||||
|
for i, datum := range domainGHOSTDAGData {
|
||||||
|
ghostdagData[i] = &BlockGHOSTDAGDataHashPair{
|
||||||
|
Hash: datum.Hash,
|
||||||
|
GHOSTDAGData: domainGHOSTDAGDataGHOSTDAGData(datum.GHOSTDAGData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &MsgTrustedData{
|
||||||
|
DAAWindow: daaWindow,
|
||||||
|
GHOSTDAGData: ghostdagData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MsgPruningPointProofToDomainPruningPointProof converts *MsgPruningPointProof to *externalapi.PruningPointProof
|
||||||
|
func MsgPruningPointProofToDomainPruningPointProof(pruningPointProofMessage *MsgPruningPointProof) *externalapi.PruningPointProof {
|
||||||
|
headers := make([][]externalapi.BlockHeader, len(pruningPointProofMessage.Headers))
|
||||||
|
for blockLevel, blockLevelParents := range pruningPointProofMessage.Headers {
|
||||||
|
headers[blockLevel] = make([]externalapi.BlockHeader, len(blockLevelParents))
|
||||||
|
for i, header := range blockLevelParents {
|
||||||
|
headers[blockLevel][i] = BlockHeaderToDomainBlockHeader(header)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return &externalapi.PruningPointProof{
|
||||||
|
Headers: headers,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DomainPruningPointProofToMsgPruningPointProof converts *externalapi.PruningPointProof to *MsgPruningPointProof
|
||||||
|
func DomainPruningPointProofToMsgPruningPointProof(pruningPointProof *externalapi.PruningPointProof) *MsgPruningPointProof {
|
||||||
|
headers := make([][]*MsgBlockHeader, len(pruningPointProof.Headers))
|
||||||
|
for blockLevel, blockLevelParents := range pruningPointProof.Headers {
|
||||||
|
headers[blockLevel] = make([]*MsgBlockHeader, len(blockLevelParents))
|
||||||
|
for i, header := range blockLevelParents {
|
||||||
|
headers[blockLevel][i] = DomainBlockHeaderToBlockHeader(header)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return &MsgPruningPointProof{
|
||||||
|
Headers: headers,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -38,6 +38,10 @@ type RPCError struct {
|
|||||||
Message string
|
Message string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (err RPCError) Error() string {
|
||||||
|
return err.Message
|
||||||
|
}
|
||||||
|
|
||||||
// RPCErrorf formats according to a format specifier and returns the string
|
// RPCErrorf formats according to a format specifier and returns the string
|
||||||
// as an RPCError.
|
// as an RPCError.
|
||||||
func RPCErrorf(format string, args ...interface{}) *RPCError {
|
func RPCErrorf(format string, args ...interface{}) *RPCError {
|
||||||
|
@ -45,24 +45,34 @@ const (
|
|||||||
CmdRequestRelayBlocks
|
CmdRequestRelayBlocks
|
||||||
CmdInvTransaction
|
CmdInvTransaction
|
||||||
CmdRequestTransactions
|
CmdRequestTransactions
|
||||||
CmdIBDBlock
|
|
||||||
CmdDoneHeaders
|
CmdDoneHeaders
|
||||||
CmdTransactionNotFound
|
CmdTransactionNotFound
|
||||||
CmdReject
|
CmdReject
|
||||||
CmdHeader
|
|
||||||
CmdRequestNextHeaders
|
CmdRequestNextHeaders
|
||||||
CmdRequestPruningPointUTXOSetAndBlock
|
CmdRequestPruningPointUTXOSet
|
||||||
CmdPruningPointUTXOSetChunk
|
CmdPruningPointUTXOSetChunk
|
||||||
CmdRequestIBDBlocks
|
|
||||||
CmdUnexpectedPruningPoint
|
CmdUnexpectedPruningPoint
|
||||||
CmdRequestPruningPointHash
|
|
||||||
CmdPruningPointHash
|
|
||||||
CmdIBDBlockLocator
|
CmdIBDBlockLocator
|
||||||
CmdIBDBlockLocatorHighestHash
|
CmdIBDBlockLocatorHighestHash
|
||||||
CmdIBDBlockLocatorHighestHashNotFound
|
CmdIBDBlockLocatorHighestHashNotFound
|
||||||
CmdBlockHeaders
|
CmdBlockHeaders
|
||||||
CmdRequestNextPruningPointUTXOSetChunk
|
CmdRequestNextPruningPointUTXOSetChunk
|
||||||
CmdDonePruningPointUTXOSetChunks
|
CmdDonePruningPointUTXOSetChunks
|
||||||
|
CmdBlockWithTrustedData
|
||||||
|
CmdDoneBlocksWithTrustedData
|
||||||
|
CmdRequestPruningPointAndItsAnticone
|
||||||
|
CmdIBDBlock
|
||||||
|
CmdRequestIBDBlocks
|
||||||
|
CmdPruningPoints
|
||||||
|
CmdRequestPruningPointProof
|
||||||
|
CmdPruningPointProof
|
||||||
|
CmdReady
|
||||||
|
CmdTrustedData
|
||||||
|
CmdBlockWithTrustedDataV4
|
||||||
|
CmdRequestNextPruningPointAndItsAnticoneBlocks
|
||||||
|
CmdRequestIBDChainBlockLocator
|
||||||
|
CmdIBDChainBlockLocator
|
||||||
|
CmdRequestAnticone
|
||||||
|
|
||||||
// rpc
|
// rpc
|
||||||
CmdGetCurrentNetworkRequestMessage
|
CmdGetCurrentNetworkRequestMessage
|
||||||
@ -117,8 +127,12 @@ const (
|
|||||||
CmdNotifyUTXOsChangedRequestMessage
|
CmdNotifyUTXOsChangedRequestMessage
|
||||||
CmdNotifyUTXOsChangedResponseMessage
|
CmdNotifyUTXOsChangedResponseMessage
|
||||||
CmdUTXOsChangedNotificationMessage
|
CmdUTXOsChangedNotificationMessage
|
||||||
|
CmdStopNotifyingUTXOsChangedRequestMessage
|
||||||
|
CmdStopNotifyingUTXOsChangedResponseMessage
|
||||||
CmdGetUTXOsByAddressesRequestMessage
|
CmdGetUTXOsByAddressesRequestMessage
|
||||||
CmdGetUTXOsByAddressesResponseMessage
|
CmdGetUTXOsByAddressesResponseMessage
|
||||||
|
CmdGetBalanceByAddressRequestMessage
|
||||||
|
CmdGetBalanceByAddressResponseMessage
|
||||||
CmdGetVirtualSelectedParentBlueScoreRequestMessage
|
CmdGetVirtualSelectedParentBlueScoreRequestMessage
|
||||||
CmdGetVirtualSelectedParentBlueScoreResponseMessage
|
CmdGetVirtualSelectedParentBlueScoreResponseMessage
|
||||||
CmdNotifyVirtualSelectedParentBlueScoreChangedRequestMessage
|
CmdNotifyVirtualSelectedParentBlueScoreChangedRequestMessage
|
||||||
@ -130,6 +144,29 @@ const (
|
|||||||
CmdUnbanResponseMessage
|
CmdUnbanResponseMessage
|
||||||
CmdGetInfoRequestMessage
|
CmdGetInfoRequestMessage
|
||||||
CmdGetInfoResponseMessage
|
CmdGetInfoResponseMessage
|
||||||
|
CmdNotifyPruningPointUTXOSetOverrideRequestMessage
|
||||||
|
CmdNotifyPruningPointUTXOSetOverrideResponseMessage
|
||||||
|
CmdPruningPointUTXOSetOverrideNotificationMessage
|
||||||
|
CmdStopNotifyingPruningPointUTXOSetOverrideRequestMessage
|
||||||
|
CmdStopNotifyingPruningPointUTXOSetOverrideResponseMessage
|
||||||
|
CmdEstimateNetworkHashesPerSecondRequestMessage
|
||||||
|
CmdEstimateNetworkHashesPerSecondResponseMessage
|
||||||
|
CmdNotifyVirtualDaaScoreChangedRequestMessage
|
||||||
|
CmdNotifyVirtualDaaScoreChangedResponseMessage
|
||||||
|
CmdVirtualDaaScoreChangedNotificationMessage
|
||||||
|
CmdGetBalancesByAddressesRequestMessage
|
||||||
|
CmdGetBalancesByAddressesResponseMessage
|
||||||
|
CmdNotifyNewBlockTemplateRequestMessage
|
||||||
|
CmdNotifyNewBlockTemplateResponseMessage
|
||||||
|
CmdNewBlockTemplateNotificationMessage
|
||||||
|
CmdGetMempoolEntriesByAddressesRequestMessage
|
||||||
|
CmdGetMempoolEntriesByAddressesResponseMessage
|
||||||
|
CmdGetCoinSupplyRequestMessage
|
||||||
|
CmdGetCoinSupplyResponseMessage
|
||||||
|
CmdGetFeeEstimateRequestMessage
|
||||||
|
CmdGetFeeEstimateResponseMessage
|
||||||
|
CmdSubmitTransactionReplacementRequestMessage
|
||||||
|
CmdSubmitTransactionReplacementResponseMessage
|
||||||
)
|
)
|
||||||
|
|
||||||
// ProtocolMessageCommandToString maps all MessageCommands to their string representation
|
// ProtocolMessageCommandToString maps all MessageCommands to their string representation
|
||||||
@ -138,7 +175,7 @@ var ProtocolMessageCommandToString = map[MessageCommand]string{
|
|||||||
CmdVerAck: "VerAck",
|
CmdVerAck: "VerAck",
|
||||||
CmdRequestAddresses: "RequestAddresses",
|
CmdRequestAddresses: "RequestAddresses",
|
||||||
CmdAddresses: "Addresses",
|
CmdAddresses: "Addresses",
|
||||||
CmdRequestHeaders: "RequestHeaders",
|
CmdRequestHeaders: "CmdRequestHeaders",
|
||||||
CmdBlock: "Block",
|
CmdBlock: "Block",
|
||||||
CmdTx: "Tx",
|
CmdTx: "Tx",
|
||||||
CmdPing: "Ping",
|
CmdPing: "Ping",
|
||||||
@ -149,24 +186,34 @@ var ProtocolMessageCommandToString = map[MessageCommand]string{
|
|||||||
CmdRequestRelayBlocks: "RequestRelayBlocks",
|
CmdRequestRelayBlocks: "RequestRelayBlocks",
|
||||||
CmdInvTransaction: "InvTransaction",
|
CmdInvTransaction: "InvTransaction",
|
||||||
CmdRequestTransactions: "RequestTransactions",
|
CmdRequestTransactions: "RequestTransactions",
|
||||||
CmdIBDBlock: "IBDBlock",
|
|
||||||
CmdDoneHeaders: "DoneHeaders",
|
CmdDoneHeaders: "DoneHeaders",
|
||||||
CmdTransactionNotFound: "TransactionNotFound",
|
CmdTransactionNotFound: "TransactionNotFound",
|
||||||
CmdReject: "Reject",
|
CmdReject: "Reject",
|
||||||
CmdHeader: "Header",
|
|
||||||
CmdRequestNextHeaders: "RequestNextHeaders",
|
CmdRequestNextHeaders: "RequestNextHeaders",
|
||||||
CmdRequestPruningPointUTXOSetAndBlock: "RequestPruningPointUTXOSetAndBlock",
|
CmdRequestPruningPointUTXOSet: "RequestPruningPointUTXOSet",
|
||||||
CmdPruningPointUTXOSetChunk: "PruningPointUTXOSetChunk",
|
CmdPruningPointUTXOSetChunk: "PruningPointUTXOSetChunk",
|
||||||
CmdRequestIBDBlocks: "RequestIBDBlocks",
|
|
||||||
CmdUnexpectedPruningPoint: "UnexpectedPruningPoint",
|
CmdUnexpectedPruningPoint: "UnexpectedPruningPoint",
|
||||||
CmdRequestPruningPointHash: "RequestPruningPointHashHash",
|
|
||||||
CmdPruningPointHash: "PruningPointHash",
|
|
||||||
CmdIBDBlockLocator: "IBDBlockLocator",
|
CmdIBDBlockLocator: "IBDBlockLocator",
|
||||||
CmdIBDBlockLocatorHighestHash: "IBDBlockLocatorHighestHash",
|
CmdIBDBlockLocatorHighestHash: "IBDBlockLocatorHighestHash",
|
||||||
CmdIBDBlockLocatorHighestHashNotFound: "IBDBlockLocatorHighestHashNotFound",
|
CmdIBDBlockLocatorHighestHashNotFound: "IBDBlockLocatorHighestHashNotFound",
|
||||||
CmdBlockHeaders: "BlockHeaders",
|
CmdBlockHeaders: "BlockHeaders",
|
||||||
CmdRequestNextPruningPointUTXOSetChunk: "RequestNextPruningPointUTXOSetChunk",
|
CmdRequestNextPruningPointUTXOSetChunk: "RequestNextPruningPointUTXOSetChunk",
|
||||||
CmdDonePruningPointUTXOSetChunks: "DonePruningPointUTXOSetChunks",
|
CmdDonePruningPointUTXOSetChunks: "DonePruningPointUTXOSetChunks",
|
||||||
|
CmdBlockWithTrustedData: "BlockWithTrustedData",
|
||||||
|
CmdDoneBlocksWithTrustedData: "DoneBlocksWithTrustedData",
|
||||||
|
CmdRequestPruningPointAndItsAnticone: "RequestPruningPointAndItsAnticoneHeaders",
|
||||||
|
CmdIBDBlock: "IBDBlock",
|
||||||
|
CmdRequestIBDBlocks: "RequestIBDBlocks",
|
||||||
|
CmdPruningPoints: "PruningPoints",
|
||||||
|
CmdRequestPruningPointProof: "RequestPruningPointProof",
|
||||||
|
CmdPruningPointProof: "PruningPointProof",
|
||||||
|
CmdReady: "Ready",
|
||||||
|
CmdTrustedData: "TrustedData",
|
||||||
|
CmdBlockWithTrustedDataV4: "BlockWithTrustedDataV4",
|
||||||
|
CmdRequestNextPruningPointAndItsAnticoneBlocks: "RequestNextPruningPointAndItsAnticoneBlocks",
|
||||||
|
CmdRequestIBDChainBlockLocator: "RequestIBDChainBlockLocator",
|
||||||
|
CmdIBDChainBlockLocator: "IBDChainBlockLocator",
|
||||||
|
CmdRequestAnticone: "RequestAnticone",
|
||||||
}
|
}
|
||||||
|
|
||||||
// RPCMessageCommandToString maps all MessageCommands to their string representation
|
// RPCMessageCommandToString maps all MessageCommands to their string representation
|
||||||
@ -221,8 +268,12 @@ var RPCMessageCommandToString = map[MessageCommand]string{
|
|||||||
CmdNotifyUTXOsChangedRequestMessage: "NotifyUTXOsChangedRequest",
|
CmdNotifyUTXOsChangedRequestMessage: "NotifyUTXOsChangedRequest",
|
||||||
CmdNotifyUTXOsChangedResponseMessage: "NotifyUTXOsChangedResponse",
|
CmdNotifyUTXOsChangedResponseMessage: "NotifyUTXOsChangedResponse",
|
||||||
CmdUTXOsChangedNotificationMessage: "UTXOsChangedNotification",
|
CmdUTXOsChangedNotificationMessage: "UTXOsChangedNotification",
|
||||||
|
CmdStopNotifyingUTXOsChangedRequestMessage: "StopNotifyingUTXOsChangedRequest",
|
||||||
|
CmdStopNotifyingUTXOsChangedResponseMessage: "StopNotifyingUTXOsChangedResponse",
|
||||||
CmdGetUTXOsByAddressesRequestMessage: "GetUTXOsByAddressesRequest",
|
CmdGetUTXOsByAddressesRequestMessage: "GetUTXOsByAddressesRequest",
|
||||||
CmdGetUTXOsByAddressesResponseMessage: "GetUTXOsByAddressesResponse",
|
CmdGetUTXOsByAddressesResponseMessage: "GetUTXOsByAddressesResponse",
|
||||||
|
CmdGetBalanceByAddressRequestMessage: "GetBalanceByAddressRequest",
|
||||||
|
CmdGetBalanceByAddressResponseMessage: "GetBalancesByAddressResponse",
|
||||||
CmdGetVirtualSelectedParentBlueScoreRequestMessage: "GetVirtualSelectedParentBlueScoreRequest",
|
CmdGetVirtualSelectedParentBlueScoreRequestMessage: "GetVirtualSelectedParentBlueScoreRequest",
|
||||||
CmdGetVirtualSelectedParentBlueScoreResponseMessage: "GetVirtualSelectedParentBlueScoreResponse",
|
CmdGetVirtualSelectedParentBlueScoreResponseMessage: "GetVirtualSelectedParentBlueScoreResponse",
|
||||||
CmdNotifyVirtualSelectedParentBlueScoreChangedRequestMessage: "NotifyVirtualSelectedParentBlueScoreChangedRequest",
|
CmdNotifyVirtualSelectedParentBlueScoreChangedRequestMessage: "NotifyVirtualSelectedParentBlueScoreChangedRequest",
|
||||||
@ -232,8 +283,31 @@ var RPCMessageCommandToString = map[MessageCommand]string{
|
|||||||
CmdBanResponseMessage: "BanResponse",
|
CmdBanResponseMessage: "BanResponse",
|
||||||
CmdUnbanRequestMessage: "UnbanRequest",
|
CmdUnbanRequestMessage: "UnbanRequest",
|
||||||
CmdUnbanResponseMessage: "UnbanResponse",
|
CmdUnbanResponseMessage: "UnbanResponse",
|
||||||
CmdGetInfoRequestMessage: "GetInfoRequestMessage",
|
CmdGetInfoRequestMessage: "GetInfoRequest",
|
||||||
CmdGetInfoResponseMessage: "GeInfoResponseMessage",
|
CmdGetInfoResponseMessage: "GeInfoResponse",
|
||||||
|
CmdNotifyPruningPointUTXOSetOverrideRequestMessage: "NotifyPruningPointUTXOSetOverrideRequest",
|
||||||
|
CmdNotifyPruningPointUTXOSetOverrideResponseMessage: "NotifyPruningPointUTXOSetOverrideResponse",
|
||||||
|
CmdPruningPointUTXOSetOverrideNotificationMessage: "PruningPointUTXOSetOverrideNotification",
|
||||||
|
CmdStopNotifyingPruningPointUTXOSetOverrideRequestMessage: "StopNotifyingPruningPointUTXOSetOverrideRequest",
|
||||||
|
CmdStopNotifyingPruningPointUTXOSetOverrideResponseMessage: "StopNotifyingPruningPointUTXOSetOverrideResponse",
|
||||||
|
CmdEstimateNetworkHashesPerSecondRequestMessage: "EstimateNetworkHashesPerSecondRequest",
|
||||||
|
CmdEstimateNetworkHashesPerSecondResponseMessage: "EstimateNetworkHashesPerSecondResponse",
|
||||||
|
CmdNotifyVirtualDaaScoreChangedRequestMessage: "NotifyVirtualDaaScoreChangedRequest",
|
||||||
|
CmdNotifyVirtualDaaScoreChangedResponseMessage: "NotifyVirtualDaaScoreChangedResponse",
|
||||||
|
CmdVirtualDaaScoreChangedNotificationMessage: "VirtualDaaScoreChangedNotification",
|
||||||
|
CmdGetBalancesByAddressesRequestMessage: "GetBalancesByAddressesRequest",
|
||||||
|
CmdGetBalancesByAddressesResponseMessage: "GetBalancesByAddressesResponse",
|
||||||
|
CmdNotifyNewBlockTemplateRequestMessage: "NotifyNewBlockTemplateRequest",
|
||||||
|
CmdNotifyNewBlockTemplateResponseMessage: "NotifyNewBlockTemplateResponse",
|
||||||
|
CmdNewBlockTemplateNotificationMessage: "NewBlockTemplateNotification",
|
||||||
|
CmdGetMempoolEntriesByAddressesRequestMessage: "GetMempoolEntriesByAddressesRequest",
|
||||||
|
CmdGetMempoolEntriesByAddressesResponseMessage: "GetMempoolEntriesByAddressesResponse",
|
||||||
|
CmdGetCoinSupplyRequestMessage: "GetCoinSupplyRequest",
|
||||||
|
CmdGetCoinSupplyResponseMessage: "GetCoinSupplyResponse",
|
||||||
|
CmdGetFeeEstimateRequestMessage: "GetFeeEstimateRequest",
|
||||||
|
CmdGetFeeEstimateResponseMessage: "GetFeeEstimateResponse",
|
||||||
|
CmdSubmitTransactionReplacementRequestMessage: "SubmitTransactionReplacementRequest",
|
||||||
|
CmdSubmitTransactionReplacementResponseMessage: "SubmitTransactionReplacementResponse",
|
||||||
}
|
}
|
||||||
|
|
||||||
// Message is an interface that describes a kaspa message. A type that
|
// Message is an interface that describes a kaspa message. A type that
|
||||||
|
@ -15,19 +15,6 @@ import (
|
|||||||
// backing array multiple times.
|
// backing array multiple times.
|
||||||
const defaultTransactionAlloc = 2048
|
const defaultTransactionAlloc = 2048
|
||||||
|
|
||||||
// MaxMassAcceptedByBlock is the maximum total transaction mass a block may accept.
|
|
||||||
const MaxMassAcceptedByBlock = 10000000
|
|
||||||
|
|
||||||
// MaxMassPerTx is the maximum total mass a transaction may have.
|
|
||||||
const MaxMassPerTx = MaxMassAcceptedByBlock / 2
|
|
||||||
|
|
||||||
// MaxTxPerBlock is the maximum number of transactions that could
|
|
||||||
// possibly fit into a block.
|
|
||||||
const MaxTxPerBlock = (MaxMassAcceptedByBlock / minTxPayload) + 1
|
|
||||||
|
|
||||||
// MaxBlockParents is the maximum allowed number of parents for block.
|
|
||||||
const MaxBlockParents = 10
|
|
||||||
|
|
||||||
// TxLoc holds locator data for the offset and length of where a transaction is
|
// TxLoc holds locator data for the offset and length of where a transaction is
|
||||||
// located within a MsgBlock data buffer.
|
// located within a MsgBlock data buffer.
|
||||||
type TxLoc struct {
|
type TxLoc struct {
|
||||||
|
@ -18,16 +18,21 @@ import (
|
|||||||
|
|
||||||
// TestBlock tests the MsgBlock API.
|
// TestBlock tests the MsgBlock API.
|
||||||
func TestBlock(t *testing.T) {
|
func TestBlock(t *testing.T) {
|
||||||
pver := ProtocolVersion
|
pver := uint32(4)
|
||||||
|
|
||||||
// Block 1 header.
|
// Block 1 header.
|
||||||
parentHashes := blockOne.Header.ParentHashes
|
parents := blockOne.Header.Parents
|
||||||
hashMerkleRoot := blockOne.Header.HashMerkleRoot
|
hashMerkleRoot := blockOne.Header.HashMerkleRoot
|
||||||
acceptedIDMerkleRoot := blockOne.Header.AcceptedIDMerkleRoot
|
acceptedIDMerkleRoot := blockOne.Header.AcceptedIDMerkleRoot
|
||||||
utxoCommitment := blockOne.Header.UTXOCommitment
|
utxoCommitment := blockOne.Header.UTXOCommitment
|
||||||
bits := blockOne.Header.Bits
|
bits := blockOne.Header.Bits
|
||||||
nonce := blockOne.Header.Nonce
|
nonce := blockOne.Header.Nonce
|
||||||
bh := NewBlockHeader(1, parentHashes, hashMerkleRoot, acceptedIDMerkleRoot, utxoCommitment, bits, nonce)
|
daaScore := blockOne.Header.DAAScore
|
||||||
|
blueScore := blockOne.Header.BlueScore
|
||||||
|
blueWork := blockOne.Header.BlueWork
|
||||||
|
pruningPoint := blockOne.Header.PruningPoint
|
||||||
|
bh := NewBlockHeader(1, parents, hashMerkleRoot, acceptedIDMerkleRoot, utxoCommitment, bits, nonce,
|
||||||
|
daaScore, blueScore, blueWork, pruningPoint)
|
||||||
|
|
||||||
// Ensure the command is expected value.
|
// Ensure the command is expected value.
|
||||||
wantCmd := MessageCommand(5)
|
wantCmd := MessageCommand(5)
|
||||||
@ -127,11 +132,11 @@ func TestConvertToPartial(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//blockOne is the first block in the mainnet block DAG.
|
// blockOne is the first block in the mainnet block DAG.
|
||||||
var blockOne = MsgBlock{
|
var blockOne = MsgBlock{
|
||||||
Header: MsgBlockHeader{
|
Header: MsgBlockHeader{
|
||||||
Version: 0,
|
Version: 0,
|
||||||
ParentHashes: []*externalapi.DomainHash{mainnetGenesisHash, simnetGenesisHash},
|
Parents: []externalapi.BlockLevelParents{[]*externalapi.DomainHash{mainnetGenesisHash, simnetGenesisHash}},
|
||||||
HashMerkleRoot: mainnetGenesisMerkleRoot,
|
HashMerkleRoot: mainnetGenesisMerkleRoot,
|
||||||
AcceptedIDMerkleRoot: exampleAcceptedIDMerkleRoot,
|
AcceptedIDMerkleRoot: exampleAcceptedIDMerkleRoot,
|
||||||
UTXOCommitment: exampleUTXOCommitment,
|
UTXOCommitment: exampleUTXOCommitment,
|
||||||
|
@ -5,13 +5,12 @@
|
|||||||
package appmessage
|
package appmessage
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"math"
|
"math/big"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
"github.com/kaspanet/kaspad/util/mstime"
|
"github.com/kaspanet/kaspad/util/mstime"
|
||||||
"github.com/pkg/errors"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// BaseBlockHeaderPayload is the base number of bytes a block header can be,
|
// BaseBlockHeaderPayload is the base number of bytes a block header can be,
|
||||||
@ -39,8 +38,8 @@ type MsgBlockHeader struct {
|
|||||||
// Version of the block. This is not the same as the protocol version.
|
// Version of the block. This is not the same as the protocol version.
|
||||||
Version uint16
|
Version uint16
|
||||||
|
|
||||||
// Hashes of the parent block headers in the blockDAG.
|
// Parents are the parent block hashes of the block in the DAG per superblock level.
|
||||||
ParentHashes []*externalapi.DomainHash
|
Parents []externalapi.BlockLevelParents
|
||||||
|
|
||||||
// HashMerkleRoot is the merkle tree reference to hash of all transactions for the block.
|
// HashMerkleRoot is the merkle tree reference to hash of all transactions for the block.
|
||||||
HashMerkleRoot *externalapi.DomainHash
|
HashMerkleRoot *externalapi.DomainHash
|
||||||
@ -60,15 +59,16 @@ type MsgBlockHeader struct {
|
|||||||
|
|
||||||
// Nonce used to generate the block.
|
// Nonce used to generate the block.
|
||||||
Nonce uint64
|
Nonce uint64
|
||||||
}
|
|
||||||
|
|
||||||
// NumParentBlocks return the number of entries in ParentHashes
|
// DAASCore is the DAA score of the block.
|
||||||
func (h *MsgBlockHeader) NumParentBlocks() byte {
|
DAAScore uint64
|
||||||
numParents := len(h.ParentHashes)
|
|
||||||
if numParents > math.MaxUint8 {
|
BlueScore uint64
|
||||||
panic(errors.Errorf("number of parents is %d, which is more than one byte can fit", numParents))
|
|
||||||
}
|
// BlueWork is the blue work of the block.
|
||||||
return byte(numParents)
|
BlueWork *big.Int
|
||||||
|
|
||||||
|
PruningPoint *externalapi.DomainHash
|
||||||
}
|
}
|
||||||
|
|
||||||
// BlockHash computes the block identifier hash for the given block header.
|
// BlockHash computes the block identifier hash for the given block header.
|
||||||
@ -76,33 +76,27 @@ func (h *MsgBlockHeader) BlockHash() *externalapi.DomainHash {
|
|||||||
return consensushashing.HeaderHash(BlockHeaderToDomainBlockHeader(h))
|
return consensushashing.HeaderHash(BlockHeaderToDomainBlockHeader(h))
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsGenesis returns true iff this block is a genesis block
|
|
||||||
func (h *MsgBlockHeader) IsGenesis() bool {
|
|
||||||
return h.NumParentBlocks() == 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// Command returns the protocol command string for the message. This is part
|
|
||||||
// of the Message interface implementation.
|
|
||||||
func (h *MsgBlockHeader) Command() MessageCommand {
|
|
||||||
return CmdHeader
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewBlockHeader returns a new MsgBlockHeader using the provided version, previous
|
// NewBlockHeader returns a new MsgBlockHeader using the provided version, previous
|
||||||
// block hash, hash merkle root, accepted ID merkle root, difficulty bits, and nonce used to generate the
|
// block hash, hash merkle root, accepted ID merkle root, difficulty bits, and nonce used to generate the
|
||||||
// block with defaults or calclulated values for the remaining fields.
|
// block with defaults or calclulated values for the remaining fields.
|
||||||
func NewBlockHeader(version uint16, parentHashes []*externalapi.DomainHash, hashMerkleRoot *externalapi.DomainHash,
|
func NewBlockHeader(version uint16, parents []externalapi.BlockLevelParents, hashMerkleRoot *externalapi.DomainHash,
|
||||||
acceptedIDMerkleRoot *externalapi.DomainHash, utxoCommitment *externalapi.DomainHash, bits uint32, nonce uint64) *MsgBlockHeader {
|
acceptedIDMerkleRoot *externalapi.DomainHash, utxoCommitment *externalapi.DomainHash, bits uint32, nonce,
|
||||||
|
daaScore, blueScore uint64, blueWork *big.Int, pruningPoint *externalapi.DomainHash) *MsgBlockHeader {
|
||||||
|
|
||||||
// Limit the timestamp to one millisecond precision since the protocol
|
// Limit the timestamp to one millisecond precision since the protocol
|
||||||
// doesn't support better.
|
// doesn't support better.
|
||||||
return &MsgBlockHeader{
|
return &MsgBlockHeader{
|
||||||
Version: version,
|
Version: version,
|
||||||
ParentHashes: parentHashes,
|
Parents: parents,
|
||||||
HashMerkleRoot: hashMerkleRoot,
|
HashMerkleRoot: hashMerkleRoot,
|
||||||
AcceptedIDMerkleRoot: acceptedIDMerkleRoot,
|
AcceptedIDMerkleRoot: acceptedIDMerkleRoot,
|
||||||
UTXOCommitment: utxoCommitment,
|
UTXOCommitment: utxoCommitment,
|
||||||
Timestamp: mstime.Now(),
|
Timestamp: mstime.Now(),
|
||||||
Bits: bits,
|
Bits: bits,
|
||||||
Nonce: nonce,
|
Nonce: nonce,
|
||||||
|
DAAScore: daaScore,
|
||||||
|
BlueScore: blueScore,
|
||||||
|
BlueWork: blueWork,
|
||||||
|
PruningPoint: pruningPoint,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,33 +5,34 @@
|
|||||||
package appmessage
|
package appmessage
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"math/big"
|
||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/davecgh/go-spew/spew"
|
"github.com/davecgh/go-spew/spew"
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
"github.com/kaspanet/kaspad/util/mstime"
|
|
||||||
"github.com/kaspanet/kaspad/util/random"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// TestBlockHeader tests the MsgBlockHeader API.
|
// TestBlockHeader tests the MsgBlockHeader API.
|
||||||
func TestBlockHeader(t *testing.T) {
|
func TestBlockHeader(t *testing.T) {
|
||||||
nonce, err := random.Uint64()
|
nonce := uint64(0xba4d87a69924a93d)
|
||||||
if err != nil {
|
|
||||||
t.Errorf("random.Uint64: Error generating nonce: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
hashes := []*externalapi.DomainHash{mainnetGenesisHash, simnetGenesisHash}
|
parents := []externalapi.BlockLevelParents{[]*externalapi.DomainHash{mainnetGenesisHash, simnetGenesisHash}}
|
||||||
|
|
||||||
merkleHash := mainnetGenesisMerkleRoot
|
merkleHash := mainnetGenesisMerkleRoot
|
||||||
acceptedIDMerkleRoot := exampleAcceptedIDMerkleRoot
|
acceptedIDMerkleRoot := exampleAcceptedIDMerkleRoot
|
||||||
bits := uint32(0x1d00ffff)
|
bits := uint32(0x1d00ffff)
|
||||||
bh := NewBlockHeader(1, hashes, merkleHash, acceptedIDMerkleRoot, exampleUTXOCommitment, bits, nonce)
|
daaScore := uint64(123)
|
||||||
|
blueScore := uint64(456)
|
||||||
|
blueWork := big.NewInt(789)
|
||||||
|
pruningPoint := simnetGenesisHash
|
||||||
|
bh := NewBlockHeader(1, parents, merkleHash, acceptedIDMerkleRoot, exampleUTXOCommitment, bits, nonce,
|
||||||
|
daaScore, blueScore, blueWork, pruningPoint)
|
||||||
|
|
||||||
// Ensure we get the same data back out.
|
// Ensure we get the same data back out.
|
||||||
if !reflect.DeepEqual(bh.ParentHashes, hashes) {
|
if !reflect.DeepEqual(bh.Parents, parents) {
|
||||||
t.Errorf("NewBlockHeader: wrong prev hashes - got %v, want %v",
|
t.Errorf("NewBlockHeader: wrong parents - got %v, want %v",
|
||||||
spew.Sprint(bh.ParentHashes), spew.Sprint(hashes))
|
spew.Sprint(bh.Parents), spew.Sprint(parents))
|
||||||
}
|
}
|
||||||
if bh.HashMerkleRoot != merkleHash {
|
if bh.HashMerkleRoot != merkleHash {
|
||||||
t.Errorf("NewBlockHeader: wrong merkle root - got %v, want %v",
|
t.Errorf("NewBlockHeader: wrong merkle root - got %v, want %v",
|
||||||
@ -45,44 +46,20 @@ func TestBlockHeader(t *testing.T) {
|
|||||||
t.Errorf("NewBlockHeader: wrong nonce - got %v, want %v",
|
t.Errorf("NewBlockHeader: wrong nonce - got %v, want %v",
|
||||||
bh.Nonce, nonce)
|
bh.Nonce, nonce)
|
||||||
}
|
}
|
||||||
}
|
if bh.DAAScore != daaScore {
|
||||||
|
t.Errorf("NewBlockHeader: wrong daaScore - got %v, want %v",
|
||||||
func TestIsGenesis(t *testing.T) {
|
bh.DAAScore, daaScore)
|
||||||
nonce := uint64(123123) // 0x1e0f3
|
|
||||||
bits := uint32(0x1d00ffff)
|
|
||||||
timestamp := mstime.UnixMilliseconds(0x495fab29000)
|
|
||||||
|
|
||||||
baseBlockHdr := &MsgBlockHeader{
|
|
||||||
Version: 1,
|
|
||||||
ParentHashes: []*externalapi.DomainHash{mainnetGenesisHash, simnetGenesisHash},
|
|
||||||
HashMerkleRoot: mainnetGenesisMerkleRoot,
|
|
||||||
Timestamp: timestamp,
|
|
||||||
Bits: bits,
|
|
||||||
Nonce: nonce,
|
|
||||||
}
|
}
|
||||||
genesisBlockHdr := &MsgBlockHeader{
|
if bh.BlueScore != blueScore {
|
||||||
Version: 1,
|
t.Errorf("NewBlockHeader: wrong blueScore - got %v, want %v",
|
||||||
ParentHashes: []*externalapi.DomainHash{},
|
bh.BlueScore, blueScore)
|
||||||
HashMerkleRoot: mainnetGenesisMerkleRoot,
|
|
||||||
Timestamp: timestamp,
|
|
||||||
Bits: bits,
|
|
||||||
Nonce: nonce,
|
|
||||||
}
|
}
|
||||||
|
if bh.BlueWork != blueWork {
|
||||||
tests := []struct {
|
t.Errorf("NewBlockHeader: wrong blueWork - got %v, want %v",
|
||||||
in *MsgBlockHeader // Block header to encode
|
bh.BlueWork, blueWork)
|
||||||
isGenesis bool // Expected result for call of .IsGenesis
|
|
||||||
}{
|
|
||||||
{genesisBlockHdr, true},
|
|
||||||
{baseBlockHdr, false},
|
|
||||||
}
|
|
||||||
|
|
||||||
t.Logf("Running %d tests", len(tests))
|
|
||||||
for i, test := range tests {
|
|
||||||
isGenesis := test.in.IsGenesis()
|
|
||||||
if isGenesis != test.isGenesis {
|
|
||||||
t.Errorf("MsgBlockHeader.IsGenesis: #%d got: %t, want: %t",
|
|
||||||
i, isGenesis, test.isGenesis)
|
|
||||||
}
|
}
|
||||||
|
if !bh.PruningPoint.Equal(pruningPoint) {
|
||||||
|
t.Errorf("NewBlockHeader: wrong pruningPoint - got %v, want %v",
|
||||||
|
bh.PruningPoint, pruningPoint)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
54
app/appmessage/p2p_msgblockwithtrusteddata.go
Normal file
54
app/appmessage/p2p_msgblockwithtrusteddata.go
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
|
"math/big"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MsgBlockWithTrustedData represents a kaspa BlockWithTrustedData message
|
||||||
|
type MsgBlockWithTrustedData struct {
|
||||||
|
baseMessage
|
||||||
|
|
||||||
|
Block *MsgBlock
|
||||||
|
DAAScore uint64
|
||||||
|
DAAWindow []*TrustedDataDataDAABlock
|
||||||
|
GHOSTDAGData []*BlockGHOSTDAGDataHashPair
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *MsgBlockWithTrustedData) Command() MessageCommand {
|
||||||
|
return CmdBlockWithTrustedData
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgBlockWithTrustedData returns a new MsgBlockWithTrustedData.
|
||||||
|
func NewMsgBlockWithTrustedData() *MsgBlockWithTrustedData {
|
||||||
|
return &MsgBlockWithTrustedData{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TrustedDataDataDAABlock is an appmessage representation of externalapi.TrustedDataDataDAABlock
|
||||||
|
type TrustedDataDataDAABlock struct {
|
||||||
|
Block *MsgBlock
|
||||||
|
GHOSTDAGData *BlockGHOSTDAGData
|
||||||
|
}
|
||||||
|
|
||||||
|
// BlockGHOSTDAGData is an appmessage representation of externalapi.BlockGHOSTDAGData
|
||||||
|
type BlockGHOSTDAGData struct {
|
||||||
|
BlueScore uint64
|
||||||
|
BlueWork *big.Int
|
||||||
|
SelectedParent *externalapi.DomainHash
|
||||||
|
MergeSetBlues []*externalapi.DomainHash
|
||||||
|
MergeSetReds []*externalapi.DomainHash
|
||||||
|
BluesAnticoneSizes []*BluesAnticoneSizes
|
||||||
|
}
|
||||||
|
|
||||||
|
// BluesAnticoneSizes is an appmessage representation of the BluesAnticoneSizes part of GHOSTDAG data.
|
||||||
|
type BluesAnticoneSizes struct {
|
||||||
|
BlueHash *externalapi.DomainHash
|
||||||
|
AnticoneSize externalapi.KType
|
||||||
|
}
|
||||||
|
|
||||||
|
// BlockGHOSTDAGDataHashPair is an appmessage representation of externalapi.BlockGHOSTDAGDataHashPair
|
||||||
|
type BlockGHOSTDAGDataHashPair struct {
|
||||||
|
Hash *externalapi.DomainHash
|
||||||
|
GHOSTDAGData *BlockGHOSTDAGData
|
||||||
|
}
|
20
app/appmessage/p2p_msgblockwithtrusteddatav4.go
Normal file
20
app/appmessage/p2p_msgblockwithtrusteddatav4.go
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgBlockWithTrustedDataV4 represents a kaspa BlockWithTrustedDataV4 message
|
||||||
|
type MsgBlockWithTrustedDataV4 struct {
|
||||||
|
baseMessage
|
||||||
|
|
||||||
|
Block *MsgBlock
|
||||||
|
DAAWindowIndices []uint64
|
||||||
|
GHOSTDAGDataIndices []uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *MsgBlockWithTrustedDataV4) Command() MessageCommand {
|
||||||
|
return CmdBlockWithTrustedDataV4
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgBlockWithTrustedDataV4 returns a new MsgBlockWithTrustedDataV4.
|
||||||
|
func NewMsgBlockWithTrustedDataV4() *MsgBlockWithTrustedDataV4 {
|
||||||
|
return &MsgBlockWithTrustedDataV4{}
|
||||||
|
}
|
21
app/appmessage/p2p_msgdoneblockswithmetadata.go
Normal file
21
app/appmessage/p2p_msgdoneblockswithmetadata.go
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgDoneBlocksWithTrustedData implements the Message interface and represents a kaspa
|
||||||
|
// DoneBlocksWithTrustedData message
|
||||||
|
//
|
||||||
|
// This message has no payload.
|
||||||
|
type MsgDoneBlocksWithTrustedData struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message. This is part
|
||||||
|
// of the Message interface implementation.
|
||||||
|
func (msg *MsgDoneBlocksWithTrustedData) Command() MessageCommand {
|
||||||
|
return CmdDoneBlocksWithTrustedData
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgDoneBlocksWithTrustedData returns a new kaspa DoneBlocksWithTrustedData message that conforms to the
|
||||||
|
// Message interface.
|
||||||
|
func NewMsgDoneBlocksWithTrustedData() *MsgDoneBlocksWithTrustedData {
|
||||||
|
return &MsgDoneBlocksWithTrustedData{}
|
||||||
|
}
|
@ -0,0 +1,16 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgRequestPruningPointAndItsAnticone represents a kaspa RequestPruningPointAndItsAnticone message
|
||||||
|
type MsgRequestPruningPointAndItsAnticone struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *MsgRequestPruningPointAndItsAnticone) Command() MessageCommand {
|
||||||
|
return CmdRequestPruningPointAndItsAnticone
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgRequestPruningPointAndItsAnticone returns a new MsgRequestPruningPointAndItsAnticone.
|
||||||
|
func NewMsgRequestPruningPointAndItsAnticone() *MsgRequestPruningPointAndItsAnticone {
|
||||||
|
return &MsgRequestPruningPointAndItsAnticone{}
|
||||||
|
}
|
@ -1,65 +0,0 @@
|
|||||||
// Copyright (c) 2013-2016 The btcsuite developers
|
|
||||||
// Use of this source code is governed by an ISC
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package appmessage
|
|
||||||
|
|
||||||
import (
|
|
||||||
"reflect"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/davecgh/go-spew/spew"
|
|
||||||
)
|
|
||||||
|
|
||||||
// TestIBDBlock tests the MsgIBDBlock API.
|
|
||||||
func TestIBDBlock(t *testing.T) {
|
|
||||||
pver := ProtocolVersion
|
|
||||||
|
|
||||||
// Block 1 header.
|
|
||||||
parentHashes := blockOne.Header.ParentHashes
|
|
||||||
hashMerkleRoot := blockOne.Header.HashMerkleRoot
|
|
||||||
acceptedIDMerkleRoot := blockOne.Header.AcceptedIDMerkleRoot
|
|
||||||
utxoCommitment := blockOne.Header.UTXOCommitment
|
|
||||||
bits := blockOne.Header.Bits
|
|
||||||
nonce := blockOne.Header.Nonce
|
|
||||||
bh := NewBlockHeader(1, parentHashes, hashMerkleRoot, acceptedIDMerkleRoot, utxoCommitment, bits, nonce)
|
|
||||||
|
|
||||||
// Ensure the command is expected value.
|
|
||||||
wantCmd := MessageCommand(15)
|
|
||||||
msg := NewMsgIBDBlock(NewMsgBlock(bh))
|
|
||||||
if cmd := msg.Command(); cmd != wantCmd {
|
|
||||||
t.Errorf("NewMsgIBDBlock: wrong command - got %v want %v",
|
|
||||||
cmd, wantCmd)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure max payload is expected value for latest protocol version.
|
|
||||||
wantPayload := uint32(1024 * 1024 * 32)
|
|
||||||
maxPayload := msg.MaxPayloadLength(pver)
|
|
||||||
if maxPayload != wantPayload {
|
|
||||||
t.Errorf("MaxPayloadLength: wrong max payload length for "+
|
|
||||||
"protocol version %d - got %v, want %v", pver,
|
|
||||||
maxPayload, wantPayload)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure we get the same block header data back out.
|
|
||||||
if !reflect.DeepEqual(&msg.Header, bh) {
|
|
||||||
t.Errorf("NewMsgIBDBlock: wrong block header - got %v, want %v",
|
|
||||||
spew.Sdump(&msg.Header), spew.Sdump(bh))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure transactions are added properly.
|
|
||||||
tx := blockOne.Transactions[0].Copy()
|
|
||||||
msg.AddTransaction(tx)
|
|
||||||
if !reflect.DeepEqual(msg.Transactions, blockOne.Transactions) {
|
|
||||||
t.Errorf("AddTransaction: wrong transactions - got %v, want %v",
|
|
||||||
spew.Sdump(msg.Transactions),
|
|
||||||
spew.Sdump(blockOne.Transactions))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure transactions are properly cleared.
|
|
||||||
msg.ClearTransactions()
|
|
||||||
if len(msg.Transactions) != 0 {
|
|
||||||
t.Errorf("ClearTransactions: wrong transactions - got %v, want %v",
|
|
||||||
len(msg.Transactions), 0)
|
|
||||||
}
|
|
||||||
}
|
|
27
app/appmessage/p2p_msgibdchainblocklocator.go
Normal file
27
app/appmessage/p2p_msgibdchainblocklocator.go
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MsgIBDChainBlockLocator implements the Message interface and represents a kaspa
|
||||||
|
// locator message. It is used to find the blockLocator of a peer that is
|
||||||
|
// syncing with you.
|
||||||
|
type MsgIBDChainBlockLocator struct {
|
||||||
|
baseMessage
|
||||||
|
BlockLocatorHashes []*externalapi.DomainHash
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message. This is part
|
||||||
|
// of the Message interface implementation.
|
||||||
|
func (msg *MsgIBDChainBlockLocator) Command() MessageCommand {
|
||||||
|
return CmdIBDChainBlockLocator
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgIBDChainBlockLocator returns a new kaspa locator message that conforms to
|
||||||
|
// the Message interface. See MsgBlockLocator for details.
|
||||||
|
func NewMsgIBDChainBlockLocator(locatorHashes []*externalapi.DomainHash) *MsgIBDChainBlockLocator {
|
||||||
|
return &MsgIBDChainBlockLocator{
|
||||||
|
BlockLocatorHashes: locatorHashes,
|
||||||
|
}
|
||||||
|
}
|
@ -6,17 +6,12 @@ package appmessage
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/util/random"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// TestPing tests the MsgPing API against the latest protocol version.
|
// TestPing tests the MsgPing API against the latest protocol version.
|
||||||
func TestPing(t *testing.T) {
|
func TestPing(t *testing.T) {
|
||||||
// Ensure we get the same nonce back out.
|
// Ensure we get the same nonce back out.
|
||||||
nonce, err := random.Uint64()
|
nonce := uint64(0x61c2c5535902862)
|
||||||
if err != nil {
|
|
||||||
t.Errorf("random.Uint64: Error generating nonce: %v", err)
|
|
||||||
}
|
|
||||||
msg := NewMsgPing(nonce)
|
msg := NewMsgPing(nonce)
|
||||||
if msg.Nonce != nonce {
|
if msg.Nonce != nonce {
|
||||||
t.Errorf("NewMsgPing: wrong nonce - got %v, want %v",
|
t.Errorf("NewMsgPing: wrong nonce - got %v, want %v",
|
||||||
|
@ -6,16 +6,11 @@ package appmessage
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/util/random"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// TestPongLatest tests the MsgPong API against the latest protocol version.
|
// TestPongLatest tests the MsgPong API against the latest protocol version.
|
||||||
func TestPongLatest(t *testing.T) {
|
func TestPongLatest(t *testing.T) {
|
||||||
nonce, err := random.Uint64()
|
nonce := uint64(0x1a05b581a5182c)
|
||||||
if err != nil {
|
|
||||||
t.Errorf("random.Uint64: error generating nonce: %v", err)
|
|
||||||
}
|
|
||||||
msg := NewMsgPong(nonce)
|
msg := NewMsgPong(nonce)
|
||||||
if msg.Nonce != nonce {
|
if msg.Nonce != nonce {
|
||||||
t.Errorf("NewMsgPong: wrong nonce - got %v, want %v",
|
t.Errorf("NewMsgPong: wrong nonce - got %v, want %v",
|
||||||
|
@ -1,23 +0,0 @@
|
|||||||
package appmessage
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
|
||||||
)
|
|
||||||
|
|
||||||
// MsgPruningPointHashMessage represents a kaspa PruningPointHash message
|
|
||||||
type MsgPruningPointHashMessage struct {
|
|
||||||
baseMessage
|
|
||||||
Hash *externalapi.DomainHash
|
|
||||||
}
|
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
|
||||||
func (msg *MsgPruningPointHashMessage) Command() MessageCommand {
|
|
||||||
return CmdPruningPointHash
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewPruningPointHashMessage returns a new kaspa PruningPointHash message
|
|
||||||
func NewPruningPointHashMessage(hash *externalapi.DomainHash) *MsgPruningPointHashMessage {
|
|
||||||
return &MsgPruningPointHashMessage{
|
|
||||||
Hash: hash,
|
|
||||||
}
|
|
||||||
}
|
|
20
app/appmessage/p2p_msgpruningpointproof.go
Normal file
20
app/appmessage/p2p_msgpruningpointproof.go
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgPruningPointProof represents a kaspa PruningPointProof message
|
||||||
|
type MsgPruningPointProof struct {
|
||||||
|
baseMessage
|
||||||
|
|
||||||
|
Headers [][]*MsgBlockHeader
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *MsgPruningPointProof) Command() MessageCommand {
|
||||||
|
return CmdPruningPointProof
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgPruningPointProof returns a new MsgPruningPointProof.
|
||||||
|
func NewMsgPruningPointProof(headers [][]*MsgBlockHeader) *MsgPruningPointProof {
|
||||||
|
return &MsgPruningPointProof{
|
||||||
|
Headers: headers,
|
||||||
|
}
|
||||||
|
}
|
20
app/appmessage/p2p_msgpruningpoints.go
Normal file
20
app/appmessage/p2p_msgpruningpoints.go
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgPruningPoints represents a kaspa PruningPoints message
|
||||||
|
type MsgPruningPoints struct {
|
||||||
|
baseMessage
|
||||||
|
|
||||||
|
Headers []*MsgBlockHeader
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *MsgPruningPoints) Command() MessageCommand {
|
||||||
|
return CmdPruningPoints
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgPruningPoints returns a new MsgPruningPoints.
|
||||||
|
func NewMsgPruningPoints(headers []*MsgBlockHeader) *MsgPruningPoints {
|
||||||
|
return &MsgPruningPoints{
|
||||||
|
Headers: headers,
|
||||||
|
}
|
||||||
|
}
|
@ -31,6 +31,6 @@ type OutpointAndUTXOEntryPair struct {
|
|||||||
type UTXOEntry struct {
|
type UTXOEntry struct {
|
||||||
Amount uint64
|
Amount uint64
|
||||||
ScriptPublicKey *externalapi.ScriptPublicKey
|
ScriptPublicKey *externalapi.ScriptPublicKey
|
||||||
BlockBlueScore uint64
|
BlockDAAScore uint64
|
||||||
IsCoinbase bool
|
IsCoinbase bool
|
||||||
}
|
}
|
||||||
|
33
app/appmessage/p2p_msgrequestanticone.go
Normal file
33
app/appmessage/p2p_msgrequestanticone.go
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
// Copyright (c) 2013-2016 The btcsuite developers
|
||||||
|
// Use of this source code is governed by an ISC
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package appmessage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MsgRequestAnticone implements the Message interface and represents a kaspa
|
||||||
|
// RequestHeaders message. It is used to request the set past(ContextHash) \cap anticone(BlockHash)
|
||||||
|
type MsgRequestAnticone struct {
|
||||||
|
baseMessage
|
||||||
|
BlockHash *externalapi.DomainHash
|
||||||
|
ContextHash *externalapi.DomainHash
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message. This is part
|
||||||
|
// of the Message interface implementation.
|
||||||
|
func (msg *MsgRequestAnticone) Command() MessageCommand {
|
||||||
|
return CmdRequestAnticone
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgRequestAnticone returns a new kaspa RequestPastDiff message that conforms to the
|
||||||
|
// Message interface using the passed parameters and defaults for the remaining
|
||||||
|
// fields.
|
||||||
|
func NewMsgRequestAnticone(blockHash, contextHash *externalapi.DomainHash) *MsgRequestAnticone {
|
||||||
|
return &MsgRequestAnticone{
|
||||||
|
BlockHash: blockHash,
|
||||||
|
ContextHash: contextHash,
|
||||||
|
}
|
||||||
|
}
|
@ -10,7 +10,6 @@ import (
|
|||||||
// The locator is returned via a locator message (MsgBlockLocator).
|
// The locator is returned via a locator message (MsgBlockLocator).
|
||||||
type MsgRequestBlockLocator struct {
|
type MsgRequestBlockLocator struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
LowHash *externalapi.DomainHash
|
|
||||||
HighHash *externalapi.DomainHash
|
HighHash *externalapi.DomainHash
|
||||||
Limit uint32
|
Limit uint32
|
||||||
}
|
}
|
||||||
@ -24,9 +23,8 @@ func (msg *MsgRequestBlockLocator) Command() MessageCommand {
|
|||||||
// NewMsgRequestBlockLocator returns a new RequestBlockLocator message that conforms to the
|
// NewMsgRequestBlockLocator returns a new RequestBlockLocator message that conforms to the
|
||||||
// Message interface using the passed parameters and defaults for the remaining
|
// Message interface using the passed parameters and defaults for the remaining
|
||||||
// fields.
|
// fields.
|
||||||
func NewMsgRequestBlockLocator(lowHash, highHash *externalapi.DomainHash, limit uint32) *MsgRequestBlockLocator {
|
func NewMsgRequestBlockLocator(highHash *externalapi.DomainHash, limit uint32) *MsgRequestBlockLocator {
|
||||||
return &MsgRequestBlockLocator{
|
return &MsgRequestBlockLocator{
|
||||||
LowHash: lowHash,
|
|
||||||
HighHash: highHash,
|
HighHash: highHash,
|
||||||
Limit: limit,
|
Limit: limit,
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,7 @@ func TestRequestBlockLocator(t *testing.T) {
|
|||||||
|
|
||||||
// Ensure the command is expected value.
|
// Ensure the command is expected value.
|
||||||
wantCmd := MessageCommand(9)
|
wantCmd := MessageCommand(9)
|
||||||
msg := NewMsgRequestBlockLocator(highHash, &externalapi.DomainHash{}, 0)
|
msg := NewMsgRequestBlockLocator(highHash, 0)
|
||||||
if cmd := msg.Command(); cmd != wantCmd {
|
if cmd := msg.Command(); cmd != wantCmd {
|
||||||
t.Errorf("NewMsgRequestBlockLocator: wrong command - got %v want %v",
|
t.Errorf("NewMsgRequestBlockLocator: wrong command - got %v want %v",
|
||||||
cmd, wantCmd)
|
cmd, wantCmd)
|
||||||
|
@ -10,7 +10,7 @@ import (
|
|||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
)
|
)
|
||||||
|
|
||||||
// TestRequstIBDBlocks tests the MsgRequestHeaders API.
|
// TestRequstIBDBlocks tests the MsgRequestIBDBlocks API.
|
||||||
func TestRequstIBDBlocks(t *testing.T) {
|
func TestRequstIBDBlocks(t *testing.T) {
|
||||||
hashStr := "000000000002e7ad7b9eef9479e4aabc65cb831269cc20d2632c13684406dee0"
|
hashStr := "000000000002e7ad7b9eef9479e4aabc65cb831269cc20d2632c13684406dee0"
|
||||||
lowHash, err := externalapi.NewDomainHashFromString(hashStr)
|
lowHash, err := externalapi.NewDomainHashFromString(hashStr)
|
||||||
@ -27,14 +27,14 @@ func TestRequstIBDBlocks(t *testing.T) {
|
|||||||
// Ensure we get the same data back out.
|
// Ensure we get the same data back out.
|
||||||
msg := NewMsgRequstHeaders(lowHash, highHash)
|
msg := NewMsgRequstHeaders(lowHash, highHash)
|
||||||
if !msg.HighHash.Equal(highHash) {
|
if !msg.HighHash.Equal(highHash) {
|
||||||
t.Errorf("NewMsgRequstHeaders: wrong high hash - got %v, want %v",
|
t.Errorf("NewMsgRequstIBDBlocks: wrong high hash - got %v, want %v",
|
||||||
msg.HighHash, highHash)
|
msg.HighHash, highHash)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure the command is expected value.
|
// Ensure the command is expected value.
|
||||||
wantCmd := MessageCommand(4)
|
wantCmd := MessageCommand(4)
|
||||||
if cmd := msg.Command(); cmd != wantCmd {
|
if cmd := msg.Command(); cmd != wantCmd {
|
||||||
t.Errorf("NewMsgRequstHeaders: wrong command - got %v want %v",
|
t.Errorf("NewMsgRequstIBDBlocks: wrong command - got %v want %v",
|
||||||
cmd, wantCmd)
|
cmd, wantCmd)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
31
app/appmessage/p2p_msgrequestibdchainblocklocator.go
Normal file
31
app/appmessage/p2p_msgrequestibdchainblocklocator.go
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MsgRequestIBDChainBlockLocator implements the Message interface and represents a kaspa
|
||||||
|
// IBDRequestChainBlockLocator message. It is used to request a block locator between low
|
||||||
|
// and high hash.
|
||||||
|
// The locator is returned via a locator message (MsgIBDChainBlockLocator).
|
||||||
|
type MsgRequestIBDChainBlockLocator struct {
|
||||||
|
baseMessage
|
||||||
|
HighHash *externalapi.DomainHash
|
||||||
|
LowHash *externalapi.DomainHash
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message. This is part
|
||||||
|
// of the Message interface implementation.
|
||||||
|
func (msg *MsgRequestIBDChainBlockLocator) Command() MessageCommand {
|
||||||
|
return CmdRequestIBDChainBlockLocator
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgIBDRequestChainBlockLocator returns a new IBDRequestChainBlockLocator message that conforms to the
|
||||||
|
// Message interface using the passed parameters and defaults for the remaining
|
||||||
|
// fields.
|
||||||
|
func NewMsgIBDRequestChainBlockLocator(highHash, lowHash *externalapi.DomainHash) *MsgRequestIBDChainBlockLocator {
|
||||||
|
return &MsgRequestIBDChainBlockLocator{
|
||||||
|
HighHash: highHash,
|
||||||
|
LowHash: lowHash,
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,22 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgRequestNextPruningPointAndItsAnticoneBlocks implements the Message interface and represents a kaspa
|
||||||
|
// RequestNextPruningPointAndItsAnticoneBlocks message. It is used to notify the IBD syncer peer to send
|
||||||
|
// more blocks from the pruning anticone.
|
||||||
|
//
|
||||||
|
// This message has no payload.
|
||||||
|
type MsgRequestNextPruningPointAndItsAnticoneBlocks struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message. This is part
|
||||||
|
// of the Message interface implementation.
|
||||||
|
func (msg *MsgRequestNextPruningPointAndItsAnticoneBlocks) Command() MessageCommand {
|
||||||
|
return CmdRequestNextPruningPointAndItsAnticoneBlocks
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgRequestNextPruningPointAndItsAnticoneBlocks returns a new kaspa RequestNextPruningPointAndItsAnticoneBlocks message that conforms to the
|
||||||
|
// Message interface.
|
||||||
|
func NewMsgRequestNextPruningPointAndItsAnticoneBlocks() *MsgRequestNextPruningPointAndItsAnticoneBlocks {
|
||||||
|
return &MsgRequestNextPruningPointAndItsAnticoneBlocks{}
|
||||||
|
}
|
16
app/appmessage/p2p_msgrequestpruningpointproof.go
Normal file
16
app/appmessage/p2p_msgrequestpruningpointproof.go
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgRequestPruningPointProof represents a kaspa RequestPruningPointProof message
|
||||||
|
type MsgRequestPruningPointProof struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *MsgRequestPruningPointProof) Command() MessageCommand {
|
||||||
|
return CmdRequestPruningPointProof
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgRequestPruningPointProof returns a new MsgRequestPruningPointProof.
|
||||||
|
func NewMsgRequestPruningPointProof() *MsgRequestPruningPointProof {
|
||||||
|
return &MsgRequestPruningPointProof{}
|
||||||
|
}
|
@ -4,20 +4,20 @@ import (
|
|||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
)
|
)
|
||||||
|
|
||||||
// MsgRequestPruningPointUTXOSetAndBlock represents a kaspa RequestPruningPointUTXOSetAndBlock message
|
// MsgRequestPruningPointUTXOSet represents a kaspa RequestPruningPointUTXOSet message
|
||||||
type MsgRequestPruningPointUTXOSetAndBlock struct {
|
type MsgRequestPruningPointUTXOSet struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
PruningPointHash *externalapi.DomainHash
|
PruningPointHash *externalapi.DomainHash
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
func (msg *MsgRequestPruningPointUTXOSetAndBlock) Command() MessageCommand {
|
func (msg *MsgRequestPruningPointUTXOSet) Command() MessageCommand {
|
||||||
return CmdRequestPruningPointUTXOSetAndBlock
|
return CmdRequestPruningPointUTXOSet
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewMsgRequestPruningPointUTXOSetAndBlock returns a new MsgRequestPruningPointUTXOSetAndBlock
|
// NewMsgRequestPruningPointUTXOSet returns a new MsgRequestPruningPointUTXOSet
|
||||||
func NewMsgRequestPruningPointUTXOSetAndBlock(pruningPointHash *externalapi.DomainHash) *MsgRequestPruningPointUTXOSetAndBlock {
|
func NewMsgRequestPruningPointUTXOSet(pruningPointHash *externalapi.DomainHash) *MsgRequestPruningPointUTXOSet {
|
||||||
return &MsgRequestPruningPointUTXOSetAndBlock{
|
return &MsgRequestPruningPointUTXOSet{
|
||||||
PruningPointHash: pruningPointHash,
|
PruningPointHash: pruningPointHash,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
25
app/appmessage/p2p_msgtrusteddata.go
Normal file
25
app/appmessage/p2p_msgtrusteddata.go
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgTrustedData represents a kaspa TrustedData message
|
||||||
|
type MsgTrustedData struct {
|
||||||
|
baseMessage
|
||||||
|
|
||||||
|
DAAWindow []*TrustedDataDAAHeader
|
||||||
|
GHOSTDAGData []*BlockGHOSTDAGDataHashPair
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *MsgTrustedData) Command() MessageCommand {
|
||||||
|
return CmdTrustedData
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgTrustedData returns a new MsgTrustedData.
|
||||||
|
func NewMsgTrustedData() *MsgTrustedData {
|
||||||
|
return &MsgTrustedData{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TrustedDataDAAHeader is an appmessage representation of externalapi.TrustedDataDataDAAHeader
|
||||||
|
type TrustedDataDAAHeader struct {
|
||||||
|
Header *MsgBlockHeader
|
||||||
|
GHOSTDAGData *BlockGHOSTDAGData
|
||||||
|
}
|
@ -6,7 +6,6 @@ package appmessage
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/hashes"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
||||||
@ -91,16 +90,18 @@ type TxIn struct {
|
|||||||
PreviousOutpoint Outpoint
|
PreviousOutpoint Outpoint
|
||||||
SignatureScript []byte
|
SignatureScript []byte
|
||||||
Sequence uint64
|
Sequence uint64
|
||||||
|
SigOpCount byte
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewTxIn returns a new kaspa transaction input with the provided
|
// NewTxIn returns a new kaspa transaction input with the provided
|
||||||
// previous outpoint point and signature script with a default sequence of
|
// previous outpoint point and signature script with a default sequence of
|
||||||
// MaxTxInSequenceNum.
|
// MaxTxInSequenceNum.
|
||||||
func NewTxIn(prevOut *Outpoint, signatureScript []byte, sequence uint64) *TxIn {
|
func NewTxIn(prevOut *Outpoint, signatureScript []byte, sequence uint64, sigOpCount byte) *TxIn {
|
||||||
return &TxIn{
|
return &TxIn{
|
||||||
PreviousOutpoint: *prevOut,
|
PreviousOutpoint: *prevOut,
|
||||||
SignatureScript: signatureScript,
|
SignatureScript: signatureScript,
|
||||||
Sequence: sequence,
|
Sequence: sequence,
|
||||||
|
SigOpCount: sigOpCount,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -133,7 +134,6 @@ type MsgTx struct {
|
|||||||
LockTime uint64
|
LockTime uint64
|
||||||
SubnetworkID externalapi.DomainSubnetworkID
|
SubnetworkID externalapi.DomainSubnetworkID
|
||||||
Gas uint64
|
Gas uint64
|
||||||
PayloadHash externalapi.DomainHash
|
|
||||||
Payload []byte
|
Payload []byte
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -179,7 +179,6 @@ func (msg *MsgTx) Copy() *MsgTx {
|
|||||||
LockTime: msg.LockTime,
|
LockTime: msg.LockTime,
|
||||||
SubnetworkID: msg.SubnetworkID,
|
SubnetworkID: msg.SubnetworkID,
|
||||||
Gas: msg.Gas,
|
Gas: msg.Gas,
|
||||||
PayloadHash: msg.PayloadHash,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if msg.Payload != nil {
|
if msg.Payload != nil {
|
||||||
@ -209,6 +208,7 @@ func (msg *MsgTx) Copy() *MsgTx {
|
|||||||
PreviousOutpoint: newOutpoint,
|
PreviousOutpoint: newOutpoint,
|
||||||
SignatureScript: newScript,
|
SignatureScript: newScript,
|
||||||
Sequence: oldTxIn.Sequence,
|
Sequence: oldTxIn.Sequence,
|
||||||
|
SigOpCount: oldTxIn.SigOpCount,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Finally, append this fully copied txin.
|
// Finally, append this fully copied txin.
|
||||||
@ -280,18 +280,12 @@ func newMsgTx(version uint16, txIn []*TxIn, txOut []*TxOut, subnetworkID *extern
|
|||||||
txOut = make([]*TxOut, 0, defaultTxInOutAlloc)
|
txOut = make([]*TxOut, 0, defaultTxInOutAlloc)
|
||||||
}
|
}
|
||||||
|
|
||||||
var payloadHash externalapi.DomainHash
|
|
||||||
if *subnetworkID != subnetworks.SubnetworkIDNative {
|
|
||||||
payloadHash = *hashes.PayloadHash(payload)
|
|
||||||
}
|
|
||||||
|
|
||||||
return &MsgTx{
|
return &MsgTx{
|
||||||
Version: version,
|
Version: version,
|
||||||
TxIn: txIn,
|
TxIn: txIn,
|
||||||
TxOut: txOut,
|
TxOut: txOut,
|
||||||
SubnetworkID: *subnetworkID,
|
SubnetworkID: *subnetworkID,
|
||||||
Gas: gas,
|
Gas: gas,
|
||||||
PayloadHash: payloadHash,
|
|
||||||
Payload: payload,
|
Payload: payload,
|
||||||
LockTime: lockTime,
|
LockTime: lockTime,
|
||||||
}
|
}
|
||||||
|
@ -22,7 +22,7 @@ import (
|
|||||||
|
|
||||||
// TestTx tests the MsgTx API.
|
// TestTx tests the MsgTx API.
|
||||||
func TestTx(t *testing.T) {
|
func TestTx(t *testing.T) {
|
||||||
pver := ProtocolVersion
|
pver := uint32(4)
|
||||||
|
|
||||||
txIDStr := "000000000003ba27aa200b1cecaad478d2b00432346c3f1f3986da1afd33e506"
|
txIDStr := "000000000003ba27aa200b1cecaad478d2b00432346c3f1f3986da1afd33e506"
|
||||||
txID, err := transactionid.FromString(txIDStr)
|
txID, err := transactionid.FromString(txIDStr)
|
||||||
@ -68,7 +68,7 @@ func TestTx(t *testing.T) {
|
|||||||
|
|
||||||
// Ensure we get the same transaction input back out.
|
// Ensure we get the same transaction input back out.
|
||||||
sigScript := []byte{0x04, 0x31, 0xdc, 0x00, 0x1b, 0x01, 0x62}
|
sigScript := []byte{0x04, 0x31, 0xdc, 0x00, 0x1b, 0x01, 0x62}
|
||||||
txIn := NewTxIn(prevOut, sigScript, constants.MaxTxInSequenceNum)
|
txIn := NewTxIn(prevOut, sigScript, constants.MaxTxInSequenceNum, 1)
|
||||||
if !reflect.DeepEqual(&txIn.PreviousOutpoint, prevOut) {
|
if !reflect.DeepEqual(&txIn.PreviousOutpoint, prevOut) {
|
||||||
t.Errorf("NewTxIn: wrong prev outpoint - got %v, want %v",
|
t.Errorf("NewTxIn: wrong prev outpoint - got %v, want %v",
|
||||||
spew.Sprint(&txIn.PreviousOutpoint),
|
spew.Sprint(&txIn.PreviousOutpoint),
|
||||||
@ -133,8 +133,8 @@ func TestTx(t *testing.T) {
|
|||||||
|
|
||||||
// TestTxHash tests the ability to generate the hash of a transaction accurately.
|
// TestTxHash tests the ability to generate the hash of a transaction accurately.
|
||||||
func TestTxHashAndID(t *testing.T) {
|
func TestTxHashAndID(t *testing.T) {
|
||||||
txHash1Str := "4bee9ee495bd93a755de428376bd582a2bb6ec37c041753b711c0606d5745c13"
|
txHash1Str := "b06f8b650115b5cf4d59499e10764a9312742930cb43c9b4ff6495d76f332ed7"
|
||||||
txID1Str := "f868bd20e816256b80eac976821be4589d24d21141bd1cec6e8005d0c16c6881"
|
txID1Str := "e20225c3d065ee41743607ee627db44d01ef396dc9779b05b2caf55bac50e12d"
|
||||||
wantTxID1, err := transactionid.FromString(txID1Str)
|
wantTxID1, err := transactionid.FromString(txID1Str)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("NewTxIDFromStr: %v", err)
|
t.Fatalf("NewTxIDFromStr: %v", err)
|
||||||
@ -185,14 +185,14 @@ func TestTxHashAndID(t *testing.T) {
|
|||||||
spew.Sprint(tx1ID), spew.Sprint(wantTxID1))
|
spew.Sprint(tx1ID), spew.Sprint(wantTxID1))
|
||||||
}
|
}
|
||||||
|
|
||||||
hash2Str := "cb1bdb4a83d4885535fb3cceb5c96597b7df903db83f0ffcd779d703affd8efd"
|
hash2Str := "fa16a8ce88d52ca1ff45187bbba0d33044e9f5fe309e8d0b22d4812dcf1782b7"
|
||||||
wantHash2, err := externalapi.NewDomainHashFromString(hash2Str)
|
wantHash2, err := externalapi.NewDomainHashFromString(hash2Str)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("NewTxIDFromStr: %v", err)
|
t.Errorf("NewTxIDFromStr: %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
id2Str := "ca080073d4ddf5b84443a0964af633f3c70a5b290fd3bc35a7e6f93fd33f9330"
|
id2Str := "89ffb49474637502d9059af38b8a95fc2f0d3baef5c801d7a9b9c8830671b711"
|
||||||
wantID2, err := transactionid.FromString(id2Str)
|
wantID2, err := transactionid.FromString(id2Str)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("NewTxIDFromStr: %v", err)
|
t.Errorf("NewTxIDFromStr: %v", err)
|
||||||
|
@ -82,12 +82,12 @@ func (msg *MsgVersion) Command() MessageCommand {
|
|||||||
// Message interface using the passed parameters and defaults for the remaining
|
// Message interface using the passed parameters and defaults for the remaining
|
||||||
// fields.
|
// fields.
|
||||||
func NewMsgVersion(addr *NetAddress, id *id.ID, network string,
|
func NewMsgVersion(addr *NetAddress, id *id.ID, network string,
|
||||||
subnetworkID *externalapi.DomainSubnetworkID) *MsgVersion {
|
subnetworkID *externalapi.DomainSubnetworkID, protocolVersion uint32) *MsgVersion {
|
||||||
|
|
||||||
// Limit the timestamp to one millisecond precision since the protocol
|
// Limit the timestamp to one millisecond precision since the protocol
|
||||||
// doesn't support better.
|
// doesn't support better.
|
||||||
return &MsgVersion{
|
return &MsgVersion{
|
||||||
ProtocolVersion: ProtocolVersion,
|
ProtocolVersion: protocolVersion,
|
||||||
Network: network,
|
Network: network,
|
||||||
Services: 0,
|
Services: 0,
|
||||||
Timestamp: mstime.Now(),
|
Timestamp: mstime.Now(),
|
||||||
|
@ -15,18 +15,18 @@ import (
|
|||||||
|
|
||||||
// TestVersion tests the MsgVersion API.
|
// TestVersion tests the MsgVersion API.
|
||||||
func TestVersion(t *testing.T) {
|
func TestVersion(t *testing.T) {
|
||||||
pver := ProtocolVersion
|
pver := uint32(4)
|
||||||
|
|
||||||
// Create version message data.
|
// Create version message data.
|
||||||
tcpAddrMe := &net.TCPAddr{IP: net.ParseIP("127.0.0.1"), Port: 16111}
|
tcpAddrMe := &net.TCPAddr{IP: net.ParseIP("127.0.0.1"), Port: 16111}
|
||||||
me := NewNetAddress(tcpAddrMe, SFNodeNetwork)
|
me := NewNetAddress(tcpAddrMe)
|
||||||
generatedID, err := id.GenerateID()
|
generatedID, err := id.GenerateID()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("id.GenerateID: %s", err)
|
t.Fatalf("id.GenerateID: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure we get the correct data back out.
|
// Ensure we get the correct data back out.
|
||||||
msg := NewMsgVersion(me, generatedID, "mainnet", nil)
|
msg := NewMsgVersion(me, generatedID, "mainnet", nil, 4)
|
||||||
if msg.ProtocolVersion != pver {
|
if msg.ProtocolVersion != pver {
|
||||||
t.Errorf("NewMsgVersion: wrong protocol version - got %v, want %v",
|
t.Errorf("NewMsgVersion: wrong protocol version - got %v, want %v",
|
||||||
msg.ProtocolVersion, pver)
|
msg.ProtocolVersion, pver)
|
||||||
|
@ -5,8 +5,9 @@
|
|||||||
package appmessage
|
package appmessage
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/kaspanet/kaspad/util/mstime"
|
|
||||||
"net"
|
"net"
|
||||||
|
|
||||||
|
"github.com/kaspanet/kaspad/util/mstime"
|
||||||
)
|
)
|
||||||
|
|
||||||
// NetAddress defines information about a peer on the network including the time
|
// NetAddress defines information about a peer on the network including the time
|
||||||
@ -15,9 +16,6 @@ type NetAddress struct {
|
|||||||
// Last time the address was seen.
|
// Last time the address was seen.
|
||||||
Timestamp mstime.Time
|
Timestamp mstime.Time
|
||||||
|
|
||||||
// Bitfield which identifies the services supported by the address.
|
|
||||||
Services ServiceFlag
|
|
||||||
|
|
||||||
// IP address of the peer.
|
// IP address of the peer.
|
||||||
IP net.IP
|
IP net.IP
|
||||||
|
|
||||||
@ -26,17 +24,6 @@ type NetAddress struct {
|
|||||||
Port uint16
|
Port uint16
|
||||||
}
|
}
|
||||||
|
|
||||||
// HasService returns whether the specified service is supported by the address.
|
|
||||||
func (na *NetAddress) HasService(service ServiceFlag) bool {
|
|
||||||
return na.Services&service == service
|
|
||||||
}
|
|
||||||
|
|
||||||
// AddService adds service as a supported service by the peer generating the
|
|
||||||
// message.
|
|
||||||
func (na *NetAddress) AddService(service ServiceFlag) {
|
|
||||||
na.Services |= service
|
|
||||||
}
|
|
||||||
|
|
||||||
// TCPAddress converts the NetAddress to *net.TCPAddr
|
// TCPAddress converts the NetAddress to *net.TCPAddr
|
||||||
func (na *NetAddress) TCPAddress() *net.TCPAddr {
|
func (na *NetAddress) TCPAddress() *net.TCPAddr {
|
||||||
return &net.TCPAddr{
|
return &net.TCPAddr{
|
||||||
@ -47,20 +34,19 @@ func (na *NetAddress) TCPAddress() *net.TCPAddr {
|
|||||||
|
|
||||||
// NewNetAddressIPPort returns a new NetAddress using the provided IP, port, and
|
// NewNetAddressIPPort returns a new NetAddress using the provided IP, port, and
|
||||||
// supported services with defaults for the remaining fields.
|
// supported services with defaults for the remaining fields.
|
||||||
func NewNetAddressIPPort(ip net.IP, port uint16, services ServiceFlag) *NetAddress {
|
func NewNetAddressIPPort(ip net.IP, port uint16) *NetAddress {
|
||||||
return NewNetAddressTimestamp(mstime.Now(), services, ip, port)
|
return NewNetAddressTimestamp(mstime.Now(), ip, port)
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewNetAddressTimestamp returns a new NetAddress using the provided
|
// NewNetAddressTimestamp returns a new NetAddress using the provided
|
||||||
// timestamp, IP, port, and supported services. The timestamp is rounded to
|
// timestamp, IP, port, and supported services. The timestamp is rounded to
|
||||||
// single millisecond precision.
|
// single millisecond precision.
|
||||||
func NewNetAddressTimestamp(
|
func NewNetAddressTimestamp(
|
||||||
timestamp mstime.Time, services ServiceFlag, ip net.IP, port uint16) *NetAddress {
|
timestamp mstime.Time, ip net.IP, port uint16) *NetAddress {
|
||||||
// Limit the timestamp to one millisecond precision since the protocol
|
// Limit the timestamp to one millisecond precision since the protocol
|
||||||
// doesn't support better.
|
// doesn't support better.
|
||||||
na := NetAddress{
|
na := NetAddress{
|
||||||
Timestamp: timestamp,
|
Timestamp: timestamp,
|
||||||
Services: services,
|
|
||||||
IP: ip,
|
IP: ip,
|
||||||
Port: port,
|
Port: port,
|
||||||
}
|
}
|
||||||
@ -69,6 +55,10 @@ func NewNetAddressTimestamp(
|
|||||||
|
|
||||||
// NewNetAddress returns a new NetAddress using the provided TCP address and
|
// NewNetAddress returns a new NetAddress using the provided TCP address and
|
||||||
// supported services with defaults for the remaining fields.
|
// supported services with defaults for the remaining fields.
|
||||||
func NewNetAddress(addr *net.TCPAddr, services ServiceFlag) *NetAddress {
|
func NewNetAddress(addr *net.TCPAddr) *NetAddress {
|
||||||
return NewNetAddressIPPort(addr.IP, uint16(addr.Port), services)
|
return NewNetAddressIPPort(addr.IP, uint16(addr.Port))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (na NetAddress) String() string {
|
||||||
|
return na.TCPAddress().String()
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,7 @@ func TestNetAddress(t *testing.T) {
|
|||||||
port := 16111
|
port := 16111
|
||||||
|
|
||||||
// Test NewNetAddress.
|
// Test NewNetAddress.
|
||||||
na := NewNetAddress(&net.TCPAddr{IP: ip, Port: port}, 0)
|
na := NewNetAddress(&net.TCPAddr{IP: ip, Port: port})
|
||||||
|
|
||||||
// Ensure we get the same ip, port, and services back out.
|
// Ensure we get the same ip, port, and services back out.
|
||||||
if !na.IP.Equal(ip) {
|
if !na.IP.Equal(ip) {
|
||||||
@ -25,21 +25,4 @@ func TestNetAddress(t *testing.T) {
|
|||||||
t.Errorf("NetNetAddress: wrong port - got %v, want %v", na.Port,
|
t.Errorf("NetNetAddress: wrong port - got %v, want %v", na.Port,
|
||||||
port)
|
port)
|
||||||
}
|
}
|
||||||
if na.Services != 0 {
|
|
||||||
t.Errorf("NetNetAddress: wrong services - got %v, want %v",
|
|
||||||
na.Services, 0)
|
|
||||||
}
|
|
||||||
if na.HasService(SFNodeNetwork) {
|
|
||||||
t.Errorf("HasService: SFNodeNetwork service is set")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure adding the full service node flag works.
|
|
||||||
na.AddService(SFNodeNetwork)
|
|
||||||
if na.Services != SFNodeNetwork {
|
|
||||||
t.Errorf("AddService: wrong services - got %v, want %v",
|
|
||||||
na.Services, SFNodeNetwork)
|
|
||||||
}
|
|
||||||
if !na.HasService(SFNodeNetwork) {
|
|
||||||
t.Errorf("HasService: SFNodeNetwork service not set")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
22
app/appmessage/p2p_ready.go
Normal file
22
app/appmessage/p2p_ready.go
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MsgReady implements the Message interface and represents a kaspa
|
||||||
|
// Ready message. It is used to notify that the peer is ready to receive
|
||||||
|
// messages.
|
||||||
|
//
|
||||||
|
// This message has no payload.
|
||||||
|
type MsgReady struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message. This is part
|
||||||
|
// of the Message interface implementation.
|
||||||
|
func (msg *MsgReady) Command() MessageCommand {
|
||||||
|
return CmdReady
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMsgReady returns a new kaspa Ready message that conforms to the
|
||||||
|
// Message interface.
|
||||||
|
func NewMsgReady() *MsgReady {
|
||||||
|
return &MsgReady{}
|
||||||
|
}
|
@ -1,16 +0,0 @@
|
|||||||
package appmessage
|
|
||||||
|
|
||||||
// MsgRequestPruningPointHashMessage represents a kaspa RequestPruningPointHashMessage message
|
|
||||||
type MsgRequestPruningPointHashMessage struct {
|
|
||||||
baseMessage
|
|
||||||
}
|
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
|
||||||
func (msg *MsgRequestPruningPointHashMessage) Command() MessageCommand {
|
|
||||||
return CmdRequestPruningPointHash
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewMsgRequestPruningPointHashMessage returns a new kaspa RequestPruningPointHash message
|
|
||||||
func NewMsgRequestPruningPointHashMessage() *MsgRequestPruningPointHashMessage {
|
|
||||||
return &MsgRequestPruningPointHashMessage{}
|
|
||||||
}
|
|
@ -11,9 +11,6 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
// ProtocolVersion is the latest protocol version this package supports.
|
|
||||||
ProtocolVersion uint32 = 1
|
|
||||||
|
|
||||||
// DefaultServices describes the default services that are supported by
|
// DefaultServices describes the default services that are supported by
|
||||||
// the server.
|
// the server.
|
||||||
DefaultServices = SFNodeNetwork | SFNodeBloom | SFNodeCF
|
DefaultServices = SFNodeNetwork | SFNodeBloom | SFNodeCF
|
||||||
|
43
app/appmessage/rpc_estimate_network_hashes_per_second.go
Normal file
43
app/appmessage/rpc_estimate_network_hashes_per_second.go
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// EstimateNetworkHashesPerSecondRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type EstimateNetworkHashesPerSecondRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
StartHash string
|
||||||
|
WindowSize uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *EstimateNetworkHashesPerSecondRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdEstimateNetworkHashesPerSecondRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewEstimateNetworkHashesPerSecondRequestMessage returns a instance of the message
|
||||||
|
func NewEstimateNetworkHashesPerSecondRequestMessage(startHash string, windowSize uint32) *EstimateNetworkHashesPerSecondRequestMessage {
|
||||||
|
return &EstimateNetworkHashesPerSecondRequestMessage{
|
||||||
|
StartHash: startHash,
|
||||||
|
WindowSize: windowSize,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// EstimateNetworkHashesPerSecondResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type EstimateNetworkHashesPerSecondResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
NetworkHashesPerSecond uint64
|
||||||
|
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *EstimateNetworkHashesPerSecondResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdEstimateNetworkHashesPerSecondResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewEstimateNetworkHashesPerSecondResponseMessage returns a instance of the message
|
||||||
|
func NewEstimateNetworkHashesPerSecondResponseMessage(networkHashesPerSecond uint64) *EstimateNetworkHashesPerSecondResponseMessage {
|
||||||
|
return &EstimateNetworkHashesPerSecondResponseMessage{
|
||||||
|
NetworkHashesPerSecond: networkHashesPerSecond,
|
||||||
|
}
|
||||||
|
}
|
47
app/appmessage/rpc_fee_estimate.go
Normal file
47
app/appmessage/rpc_fee_estimate.go
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// GetFeeEstimateRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetFeeEstimateRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetFeeEstimateRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdGetFeeEstimateRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetFeeEstimateRequestMessage returns a instance of the message
|
||||||
|
func NewGetFeeEstimateRequestMessage() *GetFeeEstimateRequestMessage {
|
||||||
|
return &GetFeeEstimateRequestMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
type RPCFeeRateBucket struct {
|
||||||
|
Feerate float64
|
||||||
|
EstimatedSeconds float64
|
||||||
|
}
|
||||||
|
|
||||||
|
type RPCFeeEstimate struct {
|
||||||
|
PriorityBucket RPCFeeRateBucket
|
||||||
|
NormalBuckets []RPCFeeRateBucket
|
||||||
|
LowBuckets []RPCFeeRateBucket
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCoinSupplyResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetFeeEstimateResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Estimate RPCFeeEstimate
|
||||||
|
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetFeeEstimateResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdGetFeeEstimateResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetFeeEstimateResponseMessage returns a instance of the message
|
||||||
|
func NewGetFeeEstimateResponseMessage() *GetFeeEstimateResponseMessage {
|
||||||
|
return &GetFeeEstimateResponseMessage{}
|
||||||
|
}
|
41
app/appmessage/rpc_get_balance_by_address.go
Normal file
41
app/appmessage/rpc_get_balance_by_address.go
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// GetBalanceByAddressRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetBalanceByAddressRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Address string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetBalanceByAddressRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdGetBalanceByAddressRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetBalanceByAddressRequest returns a instance of the message
|
||||||
|
func NewGetBalanceByAddressRequest(address string) *GetBalanceByAddressRequestMessage {
|
||||||
|
return &GetBalanceByAddressRequestMessage{
|
||||||
|
Address: address,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetBalanceByAddressResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetBalanceByAddressResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Balance uint64
|
||||||
|
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetBalanceByAddressResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdGetBalanceByAddressResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetBalanceByAddressResponse returns an instance of the message
|
||||||
|
func NewGetBalanceByAddressResponse(Balance uint64) *GetBalanceByAddressResponseMessage {
|
||||||
|
return &GetBalanceByAddressResponseMessage{
|
||||||
|
Balance: Balance,
|
||||||
|
}
|
||||||
|
}
|
47
app/appmessage/rpc_get_balances_by_addresses.go
Normal file
47
app/appmessage/rpc_get_balances_by_addresses.go
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// GetBalancesByAddressesRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetBalancesByAddressesRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Addresses []string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetBalancesByAddressesRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdGetBalancesByAddressesRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetBalancesByAddressesRequest returns a instance of the message
|
||||||
|
func NewGetBalancesByAddressesRequest(addresses []string) *GetBalancesByAddressesRequestMessage {
|
||||||
|
return &GetBalancesByAddressesRequestMessage{
|
||||||
|
Addresses: addresses,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BalancesByAddressesEntry represents the balance of some address
|
||||||
|
type BalancesByAddressesEntry struct {
|
||||||
|
Address string
|
||||||
|
Balance uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetBalancesByAddressesResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetBalancesByAddressesResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Entries []*BalancesByAddressesEntry
|
||||||
|
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetBalancesByAddressesResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdGetBalancesByAddressesResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetBalancesByAddressesResponse returns an instance of the message
|
||||||
|
func NewGetBalancesByAddressesResponse(entries []*BalancesByAddressesEntry) *GetBalancesByAddressesResponseMessage {
|
||||||
|
return &GetBalancesByAddressesResponseMessage{
|
||||||
|
Entries: entries,
|
||||||
|
}
|
||||||
|
}
|
@ -5,7 +5,7 @@ package appmessage
|
|||||||
type GetBlockRequestMessage struct {
|
type GetBlockRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
Hash string
|
Hash string
|
||||||
IncludeTransactionVerboseData bool
|
IncludeTransactions bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -14,10 +14,10 @@ func (msg *GetBlockRequestMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetBlockRequestMessage returns a instance of the message
|
// NewGetBlockRequestMessage returns a instance of the message
|
||||||
func NewGetBlockRequestMessage(hash string, includeTransactionVerboseData bool) *GetBlockRequestMessage {
|
func NewGetBlockRequestMessage(hash string, includeTransactions bool) *GetBlockRequestMessage {
|
||||||
return &GetBlockRequestMessage{
|
return &GetBlockRequestMessage{
|
||||||
Hash: hash,
|
Hash: hash,
|
||||||
IncludeTransactionVerboseData: includeTransactionVerboseData,
|
IncludeTransactions: includeTransactions,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -25,7 +25,7 @@ func NewGetBlockRequestMessage(hash string, includeTransactionVerboseData bool)
|
|||||||
// its respective RPC message
|
// its respective RPC message
|
||||||
type GetBlockResponseMessage struct {
|
type GetBlockResponseMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
BlockVerboseData *BlockVerboseData
|
Block *RPCBlock
|
||||||
|
|
||||||
Error *RPCError
|
Error *RPCError
|
||||||
}
|
}
|
||||||
@ -39,69 +39,3 @@ func (msg *GetBlockResponseMessage) Command() MessageCommand {
|
|||||||
func NewGetBlockResponseMessage() *GetBlockResponseMessage {
|
func NewGetBlockResponseMessage() *GetBlockResponseMessage {
|
||||||
return &GetBlockResponseMessage{}
|
return &GetBlockResponseMessage{}
|
||||||
}
|
}
|
||||||
|
|
||||||
// BlockVerboseData holds verbose data about a block
|
|
||||||
type BlockVerboseData struct {
|
|
||||||
Hash string
|
|
||||||
Version uint16
|
|
||||||
VersionHex string
|
|
||||||
HashMerkleRoot string
|
|
||||||
AcceptedIDMerkleRoot string
|
|
||||||
UTXOCommitment string
|
|
||||||
TxIDs []string
|
|
||||||
TransactionVerboseData []*TransactionVerboseData
|
|
||||||
Time int64
|
|
||||||
Nonce uint64
|
|
||||||
Bits string
|
|
||||||
Difficulty float64
|
|
||||||
ParentHashes []string
|
|
||||||
SelectedParentHash string
|
|
||||||
BlueScore uint64
|
|
||||||
IsHeaderOnly bool
|
|
||||||
}
|
|
||||||
|
|
||||||
// TransactionVerboseData holds verbose data about a transaction
|
|
||||||
type TransactionVerboseData struct {
|
|
||||||
TxID string
|
|
||||||
Hash string
|
|
||||||
Size uint64
|
|
||||||
Version uint16
|
|
||||||
LockTime uint64
|
|
||||||
SubnetworkID string
|
|
||||||
Gas uint64
|
|
||||||
PayloadHash string
|
|
||||||
Payload string
|
|
||||||
TransactionVerboseInputs []*TransactionVerboseInput
|
|
||||||
TransactionVerboseOutputs []*TransactionVerboseOutput
|
|
||||||
BlockHash string
|
|
||||||
Time uint64
|
|
||||||
BlockTime uint64
|
|
||||||
}
|
|
||||||
|
|
||||||
// TransactionVerboseInput holds data about a transaction input
|
|
||||||
type TransactionVerboseInput struct {
|
|
||||||
TxID string
|
|
||||||
OutputIndex uint32
|
|
||||||
ScriptSig *ScriptSig
|
|
||||||
Sequence uint64
|
|
||||||
}
|
|
||||||
|
|
||||||
// ScriptSig holds data about a script signature
|
|
||||||
type ScriptSig struct {
|
|
||||||
Asm string
|
|
||||||
Hex string
|
|
||||||
}
|
|
||||||
|
|
||||||
// TransactionVerboseOutput holds data about a transaction output
|
|
||||||
type TransactionVerboseOutput struct {
|
|
||||||
Value uint64
|
|
||||||
Index uint32
|
|
||||||
ScriptPubKey *ScriptPubKeyResult
|
|
||||||
}
|
|
||||||
|
|
||||||
// ScriptPubKeyResult holds data about a script public key
|
|
||||||
type ScriptPubKeyResult struct {
|
|
||||||
Hex string
|
|
||||||
Type string
|
|
||||||
Address string
|
|
||||||
}
|
|
||||||
|
@ -27,6 +27,8 @@ type GetBlockDAGInfoResponseMessage struct {
|
|||||||
VirtualParentHashes []string
|
VirtualParentHashes []string
|
||||||
Difficulty float64
|
Difficulty float64
|
||||||
PastMedianTime int64
|
PastMedianTime int64
|
||||||
|
PruningPointHash string
|
||||||
|
VirtualDAAScore uint64
|
||||||
|
|
||||||
Error *RPCError
|
Error *RPCError
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ package appmessage
|
|||||||
type GetBlockTemplateRequestMessage struct {
|
type GetBlockTemplateRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
PayAddress string
|
PayAddress string
|
||||||
|
ExtraData string
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -13,9 +14,10 @@ func (msg *GetBlockTemplateRequestMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetBlockTemplateRequestMessage returns a instance of the message
|
// NewGetBlockTemplateRequestMessage returns a instance of the message
|
||||||
func NewGetBlockTemplateRequestMessage(payAddress string) *GetBlockTemplateRequestMessage {
|
func NewGetBlockTemplateRequestMessage(payAddress, extraData string) *GetBlockTemplateRequestMessage {
|
||||||
return &GetBlockTemplateRequestMessage{
|
return &GetBlockTemplateRequestMessage{
|
||||||
PayAddress: payAddress,
|
PayAddress: payAddress,
|
||||||
|
ExtraData: extraData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -23,7 +25,7 @@ func NewGetBlockTemplateRequestMessage(payAddress string) *GetBlockTemplateReque
|
|||||||
// its respective RPC message
|
// its respective RPC message
|
||||||
type GetBlockTemplateResponseMessage struct {
|
type GetBlockTemplateResponseMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
MsgBlock *MsgBlock
|
Block *RPCBlock
|
||||||
IsSynced bool
|
IsSynced bool
|
||||||
|
|
||||||
Error *RPCError
|
Error *RPCError
|
||||||
@ -35,9 +37,9 @@ func (msg *GetBlockTemplateResponseMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetBlockTemplateResponseMessage returns a instance of the message
|
// NewGetBlockTemplateResponseMessage returns a instance of the message
|
||||||
func NewGetBlockTemplateResponseMessage(msgBlock *MsgBlock, isSynced bool) *GetBlockTemplateResponseMessage {
|
func NewGetBlockTemplateResponseMessage(block *RPCBlock, isSynced bool) *GetBlockTemplateResponseMessage {
|
||||||
return &GetBlockTemplateResponseMessage{
|
return &GetBlockTemplateResponseMessage{
|
||||||
MsgBlock: msgBlock,
|
Block: block,
|
||||||
IsSynced: isSynced,
|
IsSynced: isSynced,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,8 +5,8 @@ package appmessage
|
|||||||
type GetBlocksRequestMessage struct {
|
type GetBlocksRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
LowHash string
|
LowHash string
|
||||||
IncludeBlockVerboseData bool
|
IncludeBlocks bool
|
||||||
IncludeTransactionVerboseData bool
|
IncludeTransactions bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -15,12 +15,12 @@ func (msg *GetBlocksRequestMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetBlocksRequestMessage returns a instance of the message
|
// NewGetBlocksRequestMessage returns a instance of the message
|
||||||
func NewGetBlocksRequestMessage(lowHash string, includeBlockVerboseData bool,
|
func NewGetBlocksRequestMessage(lowHash string, includeBlocks bool,
|
||||||
includeTransactionVerboseData bool) *GetBlocksRequestMessage {
|
includeTransactions bool) *GetBlocksRequestMessage {
|
||||||
return &GetBlocksRequestMessage{
|
return &GetBlocksRequestMessage{
|
||||||
LowHash: lowHash,
|
LowHash: lowHash,
|
||||||
IncludeBlockVerboseData: includeBlockVerboseData,
|
IncludeBlocks: includeBlocks,
|
||||||
IncludeTransactionVerboseData: includeTransactionVerboseData,
|
IncludeTransactions: includeTransactions,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -29,7 +29,7 @@ func NewGetBlocksRequestMessage(lowHash string, includeBlockVerboseData bool,
|
|||||||
type GetBlocksResponseMessage struct {
|
type GetBlocksResponseMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
BlockHashes []string
|
BlockHashes []string
|
||||||
BlockVerboseData []*BlockVerboseData
|
Blocks []*RPCBlock
|
||||||
|
|
||||||
Error *RPCError
|
Error *RPCError
|
||||||
}
|
}
|
||||||
@ -40,11 +40,6 @@ func (msg *GetBlocksResponseMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetBlocksResponseMessage returns a instance of the message
|
// NewGetBlocksResponseMessage returns a instance of the message
|
||||||
func NewGetBlocksResponseMessage(blockHashes []string, blockHexes []string,
|
func NewGetBlocksResponseMessage() *GetBlocksResponseMessage {
|
||||||
blockVerboseData []*BlockVerboseData) *GetBlocksResponseMessage {
|
return &GetBlocksResponseMessage{}
|
||||||
|
|
||||||
return &GetBlocksResponseMessage{
|
|
||||||
BlockHashes: blockHashes,
|
|
||||||
BlockVerboseData: blockVerboseData,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
40
app/appmessage/rpc_get_coin_supply.go
Normal file
40
app/appmessage/rpc_get_coin_supply.go
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// GetCoinSupplyRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetCoinSupplyRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetCoinSupplyRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdGetCoinSupplyRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetCoinSupplyRequestMessage returns a instance of the message
|
||||||
|
func NewGetCoinSupplyRequestMessage() *GetCoinSupplyRequestMessage {
|
||||||
|
return &GetCoinSupplyRequestMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCoinSupplyResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetCoinSupplyResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
MaxSompi uint64
|
||||||
|
CirculatingSompi uint64
|
||||||
|
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetCoinSupplyResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdGetCoinSupplyResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetCoinSupplyResponseMessage returns a instance of the message
|
||||||
|
func NewGetCoinSupplyResponseMessage(maxSompi uint64, circulatingSompi uint64) *GetCoinSupplyResponseMessage {
|
||||||
|
return &GetCoinSupplyResponseMessage{
|
||||||
|
MaxSompi: maxSompi,
|
||||||
|
CirculatingSompi: circulatingSompi,
|
||||||
|
}
|
||||||
|
}
|
@ -11,8 +11,8 @@ func (msg *GetInfoRequestMessage) Command() MessageCommand {
|
|||||||
return CmdGetInfoRequestMessage
|
return CmdGetInfoRequestMessage
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewGeInfoRequestMessage returns a instance of the message
|
// NewGetInfoRequestMessage returns a instance of the message
|
||||||
func NewGeInfoRequestMessage() *GetInfoRequestMessage {
|
func NewGetInfoRequestMessage() *GetInfoRequestMessage {
|
||||||
return &GetInfoRequestMessage{}
|
return &GetInfoRequestMessage{}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -21,6 +21,10 @@ func NewGeInfoRequestMessage() *GetInfoRequestMessage {
|
|||||||
type GetInfoResponseMessage struct {
|
type GetInfoResponseMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
P2PID string
|
P2PID string
|
||||||
|
MempoolSize uint64
|
||||||
|
ServerVersion string
|
||||||
|
IsUtxoIndexed bool
|
||||||
|
IsSynced bool
|
||||||
|
|
||||||
Error *RPCError
|
Error *RPCError
|
||||||
}
|
}
|
||||||
@ -31,8 +35,12 @@ func (msg *GetInfoResponseMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetInfoResponseMessage returns a instance of the message
|
// NewGetInfoResponseMessage returns a instance of the message
|
||||||
func NewGetInfoResponseMessage(p2pID string) *GetInfoResponseMessage {
|
func NewGetInfoResponseMessage(p2pID string, mempoolSize uint64, serverVersion string, isUtxoIndexed bool, isSynced bool) *GetInfoResponseMessage {
|
||||||
return &GetInfoResponseMessage{
|
return &GetInfoResponseMessage{
|
||||||
P2PID: p2pID,
|
P2PID: p2pID,
|
||||||
|
MempoolSize: mempoolSize,
|
||||||
|
ServerVersion: serverVersion,
|
||||||
|
IsUtxoIndexed: isUtxoIndexed,
|
||||||
|
IsSynced: isSynced,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,8 @@ package appmessage
|
|||||||
// its respective RPC message
|
// its respective RPC message
|
||||||
type GetMempoolEntriesRequestMessage struct {
|
type GetMempoolEntriesRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
|
IncludeOrphanPool bool
|
||||||
|
FilterTransactionPool bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -12,8 +14,11 @@ func (msg *GetMempoolEntriesRequestMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetMempoolEntriesRequestMessage returns a instance of the message
|
// NewGetMempoolEntriesRequestMessage returns a instance of the message
|
||||||
func NewGetMempoolEntriesRequestMessage() *GetMempoolEntriesRequestMessage {
|
func NewGetMempoolEntriesRequestMessage(includeOrphanPool bool, filterTransactionPool bool) *GetMempoolEntriesRequestMessage {
|
||||||
return &GetMempoolEntriesRequestMessage{}
|
return &GetMempoolEntriesRequestMessage{
|
||||||
|
IncludeOrphanPool: includeOrphanPool,
|
||||||
|
FilterTransactionPool: filterTransactionPool,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetMempoolEntriesResponseMessage is an appmessage corresponding to
|
// GetMempoolEntriesResponseMessage is an appmessage corresponding to
|
||||||
|
52
app/appmessage/rpc_get_mempool_entries_by_addresses.go
Normal file
52
app/appmessage/rpc_get_mempool_entries_by_addresses.go
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// MempoolEntryByAddress represents MempoolEntries associated with some address
|
||||||
|
type MempoolEntryByAddress struct {
|
||||||
|
Address string
|
||||||
|
Receiving []*MempoolEntry
|
||||||
|
Sending []*MempoolEntry
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetMempoolEntriesByAddressesRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetMempoolEntriesByAddressesRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Addresses []string
|
||||||
|
IncludeOrphanPool bool
|
||||||
|
FilterTransactionPool bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetMempoolEntriesByAddressesRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdGetMempoolEntriesByAddressesRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetMempoolEntriesByAddressesRequestMessage returns a instance of the message
|
||||||
|
func NewGetMempoolEntriesByAddressesRequestMessage(addresses []string, includeOrphanPool bool, filterTransactionPool bool) *GetMempoolEntriesByAddressesRequestMessage {
|
||||||
|
return &GetMempoolEntriesByAddressesRequestMessage{
|
||||||
|
Addresses: addresses,
|
||||||
|
IncludeOrphanPool: includeOrphanPool,
|
||||||
|
FilterTransactionPool: filterTransactionPool,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetMempoolEntriesByAddressesResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type GetMempoolEntriesByAddressesResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Entries []*MempoolEntryByAddress
|
||||||
|
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *GetMempoolEntriesByAddressesResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdGetMempoolEntriesByAddressesResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGetMempoolEntriesByAddressesResponseMessage returns a instance of the message
|
||||||
|
func NewGetMempoolEntriesByAddressesResponseMessage(entries []*MempoolEntryByAddress) *GetMempoolEntriesByAddressesResponseMessage {
|
||||||
|
return &GetMempoolEntriesByAddressesResponseMessage{
|
||||||
|
Entries: entries,
|
||||||
|
}
|
||||||
|
}
|
@ -5,6 +5,8 @@ package appmessage
|
|||||||
type GetMempoolEntryRequestMessage struct {
|
type GetMempoolEntryRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
TxID string
|
TxID string
|
||||||
|
IncludeOrphanPool bool
|
||||||
|
FilterTransactionPool bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -13,8 +15,12 @@ func (msg *GetMempoolEntryRequestMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetMempoolEntryRequestMessage returns a instance of the message
|
// NewGetMempoolEntryRequestMessage returns a instance of the message
|
||||||
func NewGetMempoolEntryRequestMessage(txID string) *GetMempoolEntryRequestMessage {
|
func NewGetMempoolEntryRequestMessage(txID string, includeOrphanPool bool, filterTransactionPool bool) *GetMempoolEntryRequestMessage {
|
||||||
return &GetMempoolEntryRequestMessage{TxID: txID}
|
return &GetMempoolEntryRequestMessage{
|
||||||
|
TxID: txID,
|
||||||
|
IncludeOrphanPool: includeOrphanPool,
|
||||||
|
FilterTransactionPool: filterTransactionPool,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetMempoolEntryResponseMessage is an appmessage corresponding to
|
// GetMempoolEntryResponseMessage is an appmessage corresponding to
|
||||||
@ -29,7 +35,8 @@ type GetMempoolEntryResponseMessage struct {
|
|||||||
// MempoolEntry represents a transaction in the mempool.
|
// MempoolEntry represents a transaction in the mempool.
|
||||||
type MempoolEntry struct {
|
type MempoolEntry struct {
|
||||||
Fee uint64
|
Fee uint64
|
||||||
TransactionVerboseData *TransactionVerboseData
|
Transaction *RPCTransaction
|
||||||
|
IsOrphan bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -38,11 +45,12 @@ func (msg *GetMempoolEntryResponseMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetMempoolEntryResponseMessage returns a instance of the message
|
// NewGetMempoolEntryResponseMessage returns a instance of the message
|
||||||
func NewGetMempoolEntryResponseMessage(fee uint64, transactionVerboseData *TransactionVerboseData) *GetMempoolEntryResponseMessage {
|
func NewGetMempoolEntryResponseMessage(fee uint64, transaction *RPCTransaction, isOrphan bool) *GetMempoolEntryResponseMessage {
|
||||||
return &GetMempoolEntryResponseMessage{
|
return &GetMempoolEntryResponseMessage{
|
||||||
Entry: &MempoolEntry{
|
Entry: &MempoolEntry{
|
||||||
Fee: fee,
|
Fee: fee,
|
||||||
TransactionVerboseData: transactionVerboseData,
|
Transaction: transaction,
|
||||||
|
IsOrphan: isOrphan,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ package appmessage
|
|||||||
type GetVirtualSelectedParentChainFromBlockRequestMessage struct {
|
type GetVirtualSelectedParentChainFromBlockRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
StartHash string
|
StartHash string
|
||||||
|
IncludeAcceptedTransactionIDs bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -13,18 +14,29 @@ func (msg *GetVirtualSelectedParentChainFromBlockRequestMessage) Command() Messa
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetVirtualSelectedParentChainFromBlockRequestMessage returns a instance of the message
|
// NewGetVirtualSelectedParentChainFromBlockRequestMessage returns a instance of the message
|
||||||
func NewGetVirtualSelectedParentChainFromBlockRequestMessage(startHash string) *GetVirtualSelectedParentChainFromBlockRequestMessage {
|
func NewGetVirtualSelectedParentChainFromBlockRequestMessage(
|
||||||
|
startHash string, includeAcceptedTransactionIDs bool) *GetVirtualSelectedParentChainFromBlockRequestMessage {
|
||||||
|
|
||||||
return &GetVirtualSelectedParentChainFromBlockRequestMessage{
|
return &GetVirtualSelectedParentChainFromBlockRequestMessage{
|
||||||
StartHash: startHash,
|
StartHash: startHash,
|
||||||
|
IncludeAcceptedTransactionIDs: includeAcceptedTransactionIDs,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AcceptedTransactionIDs is a part of the GetVirtualSelectedParentChainFromBlockResponseMessage and
|
||||||
|
// VirtualSelectedParentChainChangedNotificationMessage appmessages
|
||||||
|
type AcceptedTransactionIDs struct {
|
||||||
|
AcceptingBlockHash string
|
||||||
|
AcceptedTransactionIDs []string
|
||||||
|
}
|
||||||
|
|
||||||
// GetVirtualSelectedParentChainFromBlockResponseMessage is an appmessage corresponding to
|
// GetVirtualSelectedParentChainFromBlockResponseMessage is an appmessage corresponding to
|
||||||
// its respective RPC message
|
// its respective RPC message
|
||||||
type GetVirtualSelectedParentChainFromBlockResponseMessage struct {
|
type GetVirtualSelectedParentChainFromBlockResponseMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
RemovedChainBlockHashes []string
|
RemovedChainBlockHashes []string
|
||||||
AddedChainBlocks []*ChainBlock
|
AddedChainBlockHashes []string
|
||||||
|
AcceptedTransactionIDs []*AcceptedTransactionIDs
|
||||||
|
|
||||||
Error *RPCError
|
Error *RPCError
|
||||||
}
|
}
|
||||||
@ -35,11 +47,12 @@ func (msg *GetVirtualSelectedParentChainFromBlockResponseMessage) Command() Mess
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewGetVirtualSelectedParentChainFromBlockResponseMessage returns a instance of the message
|
// NewGetVirtualSelectedParentChainFromBlockResponseMessage returns a instance of the message
|
||||||
func NewGetVirtualSelectedParentChainFromBlockResponseMessage(removedChainBlockHashes []string,
|
func NewGetVirtualSelectedParentChainFromBlockResponseMessage(removedChainBlockHashes,
|
||||||
addedChainBlocks []*ChainBlock) *GetVirtualSelectedParentChainFromBlockResponseMessage {
|
addedChainBlockHashes []string, acceptedTransactionIDs []*AcceptedTransactionIDs) *GetVirtualSelectedParentChainFromBlockResponseMessage {
|
||||||
|
|
||||||
return &GetVirtualSelectedParentChainFromBlockResponseMessage{
|
return &GetVirtualSelectedParentChainFromBlockResponseMessage{
|
||||||
RemovedChainBlockHashes: removedChainBlockHashes,
|
RemovedChainBlockHashes: removedChainBlockHashes,
|
||||||
AddedChainBlocks: addedChainBlocks,
|
AddedChainBlockHashes: addedChainBlockHashes,
|
||||||
|
AcceptedTransactionIDs: acceptedTransactionIDs,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -37,8 +37,7 @@ func NewNotifyBlockAddedResponseMessage() *NotifyBlockAddedResponseMessage {
|
|||||||
// its respective RPC message
|
// its respective RPC message
|
||||||
type BlockAddedNotificationMessage struct {
|
type BlockAddedNotificationMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
Block *MsgBlock
|
Block *RPCBlock
|
||||||
BlockVerboseData *BlockVerboseData
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -47,9 +46,8 @@ func (msg *BlockAddedNotificationMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewBlockAddedNotificationMessage returns a instance of the message
|
// NewBlockAddedNotificationMessage returns a instance of the message
|
||||||
func NewBlockAddedNotificationMessage(block *MsgBlock, blockVerboseData *BlockVerboseData) *BlockAddedNotificationMessage {
|
func NewBlockAddedNotificationMessage(block *RPCBlock) *BlockAddedNotificationMessage {
|
||||||
return &BlockAddedNotificationMessage{
|
return &BlockAddedNotificationMessage{
|
||||||
Block: block,
|
Block: block,
|
||||||
BlockVerboseData: blockVerboseData,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
50
app/appmessage/rpc_notify_new_block_template.go
Normal file
50
app/appmessage/rpc_notify_new_block_template.go
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// NotifyNewBlockTemplateRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type NotifyNewBlockTemplateRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *NotifyNewBlockTemplateRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyNewBlockTemplateRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNotifyNewBlockTemplateRequestMessage returns an instance of the message
|
||||||
|
func NewNotifyNewBlockTemplateRequestMessage() *NotifyNewBlockTemplateRequestMessage {
|
||||||
|
return &NotifyNewBlockTemplateRequestMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyNewBlockTemplateResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type NotifyNewBlockTemplateResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *NotifyNewBlockTemplateResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyNewBlockTemplateResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNotifyNewBlockTemplateResponseMessage returns an instance of the message
|
||||||
|
func NewNotifyNewBlockTemplateResponseMessage() *NotifyNewBlockTemplateResponseMessage {
|
||||||
|
return &NotifyNewBlockTemplateResponseMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewBlockTemplateNotificationMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type NewBlockTemplateNotificationMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *NewBlockTemplateNotificationMessage) Command() MessageCommand {
|
||||||
|
return CmdNewBlockTemplateNotificationMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNewBlockTemplateNotificationMessage returns an instance of the message
|
||||||
|
func NewNewBlockTemplateNotificationMessage() *NewBlockTemplateNotificationMessage {
|
||||||
|
return &NewBlockTemplateNotificationMessage{}
|
||||||
|
}
|
83
app/appmessage/rpc_notify_pruning_point_utxo_set_override.go
Normal file
83
app/appmessage/rpc_notify_pruning_point_utxo_set_override.go
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// NotifyPruningPointUTXOSetOverrideRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type NotifyPruningPointUTXOSetOverrideRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *NotifyPruningPointUTXOSetOverrideRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyPruningPointUTXOSetOverrideRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNotifyPruningPointUTXOSetOverrideRequestMessage returns a instance of the message
|
||||||
|
func NewNotifyPruningPointUTXOSetOverrideRequestMessage() *NotifyPruningPointUTXOSetOverrideRequestMessage {
|
||||||
|
return &NotifyPruningPointUTXOSetOverrideRequestMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyPruningPointUTXOSetOverrideResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type NotifyPruningPointUTXOSetOverrideResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *NotifyPruningPointUTXOSetOverrideResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyPruningPointUTXOSetOverrideResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNotifyPruningPointUTXOSetOverrideResponseMessage returns a instance of the message
|
||||||
|
func NewNotifyPruningPointUTXOSetOverrideResponseMessage() *NotifyPruningPointUTXOSetOverrideResponseMessage {
|
||||||
|
return &NotifyPruningPointUTXOSetOverrideResponseMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// PruningPointUTXOSetOverrideNotificationMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type PruningPointUTXOSetOverrideNotificationMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *PruningPointUTXOSetOverrideNotificationMessage) Command() MessageCommand {
|
||||||
|
return CmdPruningPointUTXOSetOverrideNotificationMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewPruningPointUTXOSetOverrideNotificationMessage returns a instance of the message
|
||||||
|
func NewPruningPointUTXOSetOverrideNotificationMessage() *PruningPointUTXOSetOverrideNotificationMessage {
|
||||||
|
return &PruningPointUTXOSetOverrideNotificationMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// StopNotifyingPruningPointUTXOSetOverrideRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type StopNotifyingPruningPointUTXOSetOverrideRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *StopNotifyingPruningPointUTXOSetOverrideRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyPruningPointUTXOSetOverrideRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStopNotifyingPruningPointUTXOSetOverrideRequestMessage returns a instance of the message
|
||||||
|
func NewStopNotifyingPruningPointUTXOSetOverrideRequestMessage() *StopNotifyingPruningPointUTXOSetOverrideRequestMessage {
|
||||||
|
return &StopNotifyingPruningPointUTXOSetOverrideRequestMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// StopNotifyingPruningPointUTXOSetOverrideResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type StopNotifyingPruningPointUTXOSetOverrideResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *StopNotifyingPruningPointUTXOSetOverrideResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyPruningPointUTXOSetOverrideResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStopNotifyingPruningPointUTXOSetOverrideResponseMessage returns a instance of the message
|
||||||
|
func NewStopNotifyingPruningPointUTXOSetOverrideResponseMessage() *StopNotifyingPruningPointUTXOSetOverrideResponseMessage {
|
||||||
|
return &StopNotifyingPruningPointUTXOSetOverrideResponseMessage{}
|
||||||
|
}
|
55
app/appmessage/rpc_notify_virtual_daa_score_changed.go
Normal file
55
app/appmessage/rpc_notify_virtual_daa_score_changed.go
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// NotifyVirtualDaaScoreChangedRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type NotifyVirtualDaaScoreChangedRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *NotifyVirtualDaaScoreChangedRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyVirtualDaaScoreChangedRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNotifyVirtualDaaScoreChangedRequestMessage returns a instance of the message
|
||||||
|
func NewNotifyVirtualDaaScoreChangedRequestMessage() *NotifyVirtualDaaScoreChangedRequestMessage {
|
||||||
|
return &NotifyVirtualDaaScoreChangedRequestMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyVirtualDaaScoreChangedResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type NotifyVirtualDaaScoreChangedResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *NotifyVirtualDaaScoreChangedResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdNotifyVirtualDaaScoreChangedResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNotifyVirtualDaaScoreChangedResponseMessage returns a instance of the message
|
||||||
|
func NewNotifyVirtualDaaScoreChangedResponseMessage() *NotifyVirtualDaaScoreChangedResponseMessage {
|
||||||
|
return &NotifyVirtualDaaScoreChangedResponseMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// VirtualDaaScoreChangedNotificationMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type VirtualDaaScoreChangedNotificationMessage struct {
|
||||||
|
baseMessage
|
||||||
|
VirtualDaaScore uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *VirtualDaaScoreChangedNotificationMessage) Command() MessageCommand {
|
||||||
|
return CmdVirtualDaaScoreChangedNotificationMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewVirtualDaaScoreChangedNotificationMessage returns a instance of the message
|
||||||
|
func NewVirtualDaaScoreChangedNotificationMessage(
|
||||||
|
virtualDaaScore uint64) *VirtualDaaScoreChangedNotificationMessage {
|
||||||
|
|
||||||
|
return &VirtualDaaScoreChangedNotificationMessage{
|
||||||
|
VirtualDaaScore: virtualDaaScore,
|
||||||
|
}
|
||||||
|
}
|
@ -4,6 +4,7 @@ package appmessage
|
|||||||
// its respective RPC message
|
// its respective RPC message
|
||||||
type NotifyVirtualSelectedParentChainChangedRequestMessage struct {
|
type NotifyVirtualSelectedParentChainChangedRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
|
IncludeAcceptedTransactionIDs bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -11,9 +12,13 @@ func (msg *NotifyVirtualSelectedParentChainChangedRequestMessage) Command() Mess
|
|||||||
return CmdNotifyVirtualSelectedParentChainChangedRequestMessage
|
return CmdNotifyVirtualSelectedParentChainChangedRequestMessage
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewNotifyVirtualSelectedParentChainChangedRequestMessage returns a instance of the message
|
// NewNotifyVirtualSelectedParentChainChangedRequestMessage returns an instance of the message
|
||||||
func NewNotifyVirtualSelectedParentChainChangedRequestMessage() *NotifyVirtualSelectedParentChainChangedRequestMessage {
|
func NewNotifyVirtualSelectedParentChainChangedRequestMessage(
|
||||||
return &NotifyVirtualSelectedParentChainChangedRequestMessage{}
|
includeAcceptedTransactionIDs bool) *NotifyVirtualSelectedParentChainChangedRequestMessage {
|
||||||
|
|
||||||
|
return &NotifyVirtualSelectedParentChainChangedRequestMessage{
|
||||||
|
IncludeAcceptedTransactionIDs: includeAcceptedTransactionIDs,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// NotifyVirtualSelectedParentChainChangedResponseMessage is an appmessage corresponding to
|
// NotifyVirtualSelectedParentChainChangedResponseMessage is an appmessage corresponding to
|
||||||
@ -38,19 +43,8 @@ func NewNotifyVirtualSelectedParentChainChangedResponseMessage() *NotifyVirtualS
|
|||||||
type VirtualSelectedParentChainChangedNotificationMessage struct {
|
type VirtualSelectedParentChainChangedNotificationMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
RemovedChainBlockHashes []string
|
RemovedChainBlockHashes []string
|
||||||
AddedChainBlocks []*ChainBlock
|
AddedChainBlockHashes []string
|
||||||
}
|
AcceptedTransactionIDs []*AcceptedTransactionIDs
|
||||||
|
|
||||||
// ChainBlock represents a DAG chain-block
|
|
||||||
type ChainBlock struct {
|
|
||||||
Hash string
|
|
||||||
AcceptedBlocks []*AcceptedBlock
|
|
||||||
}
|
|
||||||
|
|
||||||
// AcceptedBlock represents a block accepted into the DAG
|
|
||||||
type AcceptedBlock struct {
|
|
||||||
Hash string
|
|
||||||
AcceptedTransactionIDs []string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -59,11 +53,12 @@ func (msg *VirtualSelectedParentChainChangedNotificationMessage) Command() Messa
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewVirtualSelectedParentChainChangedNotificationMessage returns a instance of the message
|
// NewVirtualSelectedParentChainChangedNotificationMessage returns a instance of the message
|
||||||
func NewVirtualSelectedParentChainChangedNotificationMessage(removedChainBlockHashes []string,
|
func NewVirtualSelectedParentChainChangedNotificationMessage(removedChainBlockHashes,
|
||||||
addedChainBlocks []*ChainBlock) *VirtualSelectedParentChainChangedNotificationMessage {
|
addedChainBlocks []string, acceptedTransactionIDs []*AcceptedTransactionIDs) *VirtualSelectedParentChainChangedNotificationMessage {
|
||||||
|
|
||||||
return &VirtualSelectedParentChainChangedNotificationMessage{
|
return &VirtualSelectedParentChainChangedNotificationMessage{
|
||||||
RemovedChainBlockHashes: removedChainBlockHashes,
|
RemovedChainBlockHashes: removedChainBlockHashes,
|
||||||
AddedChainBlocks: addedChainBlocks,
|
AddedChainBlockHashes: addedChainBlocks,
|
||||||
|
AcceptedTransactionIDs: acceptedTransactionIDs,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
37
app/appmessage/rpc_stop_notifying_utxos_changed.go
Normal file
37
app/appmessage/rpc_stop_notifying_utxos_changed.go
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// StopNotifyingUTXOsChangedRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type StopNotifyingUTXOsChangedRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Addresses []string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *StopNotifyingUTXOsChangedRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdStopNotifyingUTXOsChangedRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStopNotifyingUTXOsChangedRequestMessage returns a instance of the message
|
||||||
|
func NewStopNotifyingUTXOsChangedRequestMessage(addresses []string) *StopNotifyingUTXOsChangedRequestMessage {
|
||||||
|
return &StopNotifyingUTXOsChangedRequestMessage{
|
||||||
|
Addresses: addresses,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// StopNotifyingUTXOsChangedResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type StopNotifyingUTXOsChangedResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *StopNotifyingUTXOsChangedResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdStopNotifyingUTXOsChangedResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStopNotifyingUTXOsChangedResponseMessage returns a instance of the message
|
||||||
|
func NewStopNotifyingUTXOsChangedResponseMessage() *StopNotifyingUTXOsChangedResponseMessage {
|
||||||
|
return &StopNotifyingUTXOsChangedResponseMessage{}
|
||||||
|
}
|
@ -4,7 +4,8 @@ package appmessage
|
|||||||
// its respective RPC message
|
// its respective RPC message
|
||||||
type SubmitBlockRequestMessage struct {
|
type SubmitBlockRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
Block *MsgBlock
|
Block *RPCBlock
|
||||||
|
AllowNonDAABlocks bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -13,9 +14,10 @@ func (msg *SubmitBlockRequestMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewSubmitBlockRequestMessage returns a instance of the message
|
// NewSubmitBlockRequestMessage returns a instance of the message
|
||||||
func NewSubmitBlockRequestMessage(block *MsgBlock) *SubmitBlockRequestMessage {
|
func NewSubmitBlockRequestMessage(block *RPCBlock, allowNonDAABlocks bool) *SubmitBlockRequestMessage {
|
||||||
return &SubmitBlockRequestMessage{
|
return &SubmitBlockRequestMessage{
|
||||||
Block: block,
|
Block: block,
|
||||||
|
AllowNonDAABlocks: allowNonDAABlocks,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -53,7 +55,51 @@ func (msg *SubmitBlockResponseMessage) Command() MessageCommand {
|
|||||||
return CmdSubmitBlockResponseMessage
|
return CmdSubmitBlockResponseMessage
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewSubmitBlockResponseMessage returns a instance of the message
|
// NewSubmitBlockResponseMessage returns an instance of the message
|
||||||
func NewSubmitBlockResponseMessage() *SubmitBlockResponseMessage {
|
func NewSubmitBlockResponseMessage() *SubmitBlockResponseMessage {
|
||||||
return &SubmitBlockResponseMessage{}
|
return &SubmitBlockResponseMessage{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RPCBlock is a kaspad block representation meant to be
|
||||||
|
// used over RPC
|
||||||
|
type RPCBlock struct {
|
||||||
|
Header *RPCBlockHeader
|
||||||
|
Transactions []*RPCTransaction
|
||||||
|
VerboseData *RPCBlockVerboseData
|
||||||
|
}
|
||||||
|
|
||||||
|
// RPCBlockHeader is a kaspad block header representation meant to be
|
||||||
|
// used over RPC
|
||||||
|
type RPCBlockHeader struct {
|
||||||
|
Version uint32
|
||||||
|
Parents []*RPCBlockLevelParents
|
||||||
|
HashMerkleRoot string
|
||||||
|
AcceptedIDMerkleRoot string
|
||||||
|
UTXOCommitment string
|
||||||
|
Timestamp int64
|
||||||
|
Bits uint32
|
||||||
|
Nonce uint64
|
||||||
|
DAAScore uint64
|
||||||
|
BlueScore uint64
|
||||||
|
BlueWork string
|
||||||
|
PruningPoint string
|
||||||
|
}
|
||||||
|
|
||||||
|
// RPCBlockLevelParents holds parent hashes for one block level
|
||||||
|
type RPCBlockLevelParents struct {
|
||||||
|
ParentHashes []string
|
||||||
|
}
|
||||||
|
|
||||||
|
// RPCBlockVerboseData holds verbose data about a block
|
||||||
|
type RPCBlockVerboseData struct {
|
||||||
|
Hash string
|
||||||
|
Difficulty float64
|
||||||
|
SelectedParentHash string
|
||||||
|
TransactionIDs []string
|
||||||
|
IsHeaderOnly bool
|
||||||
|
BlueScore uint64
|
||||||
|
ChildrenHashes []string
|
||||||
|
MergeSetBluesHashes []string
|
||||||
|
MergeSetRedsHashes []string
|
||||||
|
IsChainBlock bool
|
||||||
|
}
|
||||||
|
@ -5,6 +5,7 @@ package appmessage
|
|||||||
type SubmitTransactionRequestMessage struct {
|
type SubmitTransactionRequestMessage struct {
|
||||||
baseMessage
|
baseMessage
|
||||||
Transaction *RPCTransaction
|
Transaction *RPCTransaction
|
||||||
|
AllowOrphan bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Command returns the protocol command string for the message
|
// Command returns the protocol command string for the message
|
||||||
@ -13,9 +14,10 @@ func (msg *SubmitTransactionRequestMessage) Command() MessageCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewSubmitTransactionRequestMessage returns a instance of the message
|
// NewSubmitTransactionRequestMessage returns a instance of the message
|
||||||
func NewSubmitTransactionRequestMessage(transaction *RPCTransaction) *SubmitTransactionRequestMessage {
|
func NewSubmitTransactionRequestMessage(transaction *RPCTransaction, allowOrphan bool) *SubmitTransactionRequestMessage {
|
||||||
return &SubmitTransactionRequestMessage{
|
return &SubmitTransactionRequestMessage{
|
||||||
Transaction: transaction,
|
Transaction: transaction,
|
||||||
|
AllowOrphan: allowOrphan,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -49,8 +51,9 @@ type RPCTransaction struct {
|
|||||||
LockTime uint64
|
LockTime uint64
|
||||||
SubnetworkID string
|
SubnetworkID string
|
||||||
Gas uint64
|
Gas uint64
|
||||||
PayloadHash string
|
|
||||||
Payload string
|
Payload string
|
||||||
|
Mass uint64
|
||||||
|
VerboseData *RPCTransactionVerboseData
|
||||||
}
|
}
|
||||||
|
|
||||||
// RPCTransactionInput is a kaspad transaction input representation
|
// RPCTransactionInput is a kaspad transaction input representation
|
||||||
@ -59,6 +62,8 @@ type RPCTransactionInput struct {
|
|||||||
PreviousOutpoint *RPCOutpoint
|
PreviousOutpoint *RPCOutpoint
|
||||||
SignatureScript string
|
SignatureScript string
|
||||||
Sequence uint64
|
Sequence uint64
|
||||||
|
SigOpCount byte
|
||||||
|
VerboseData *RPCTransactionInputVerboseData
|
||||||
}
|
}
|
||||||
|
|
||||||
// RPCScriptPublicKey is a kaspad ScriptPublicKey representation
|
// RPCScriptPublicKey is a kaspad ScriptPublicKey representation
|
||||||
@ -72,6 +77,7 @@ type RPCScriptPublicKey struct {
|
|||||||
type RPCTransactionOutput struct {
|
type RPCTransactionOutput struct {
|
||||||
Amount uint64
|
Amount uint64
|
||||||
ScriptPublicKey *RPCScriptPublicKey
|
ScriptPublicKey *RPCScriptPublicKey
|
||||||
|
VerboseData *RPCTransactionOutputVerboseData
|
||||||
}
|
}
|
||||||
|
|
||||||
// RPCOutpoint is a kaspad outpoint representation meant to be used
|
// RPCOutpoint is a kaspad outpoint representation meant to be used
|
||||||
@ -86,6 +92,25 @@ type RPCOutpoint struct {
|
|||||||
type RPCUTXOEntry struct {
|
type RPCUTXOEntry struct {
|
||||||
Amount uint64
|
Amount uint64
|
||||||
ScriptPublicKey *RPCScriptPublicKey
|
ScriptPublicKey *RPCScriptPublicKey
|
||||||
BlockBlueScore uint64
|
BlockDAAScore uint64
|
||||||
IsCoinbase bool
|
IsCoinbase bool
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RPCTransactionVerboseData holds verbose data about a transaction
|
||||||
|
type RPCTransactionVerboseData struct {
|
||||||
|
TransactionID string
|
||||||
|
Hash string
|
||||||
|
Mass uint64
|
||||||
|
BlockHash string
|
||||||
|
BlockTime uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// RPCTransactionInputVerboseData holds data about a transaction input
|
||||||
|
type RPCTransactionInputVerboseData struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
// RPCTransactionOutputVerboseData holds data about a transaction output
|
||||||
|
type RPCTransactionOutputVerboseData struct {
|
||||||
|
ScriptPublicKeyType string
|
||||||
|
ScriptPublicKeyAddress string
|
||||||
|
}
|
||||||
|
42
app/appmessage/rpc_submit_transaction_replacement.go
Normal file
42
app/appmessage/rpc_submit_transaction_replacement.go
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
package appmessage
|
||||||
|
|
||||||
|
// SubmitTransactionReplacementRequestMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type SubmitTransactionReplacementRequestMessage struct {
|
||||||
|
baseMessage
|
||||||
|
Transaction *RPCTransaction
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *SubmitTransactionReplacementRequestMessage) Command() MessageCommand {
|
||||||
|
return CmdSubmitTransactionReplacementRequestMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewSubmitTransactionReplacementRequestMessage returns a instance of the message
|
||||||
|
func NewSubmitTransactionReplacementRequestMessage(transaction *RPCTransaction) *SubmitTransactionReplacementRequestMessage {
|
||||||
|
return &SubmitTransactionReplacementRequestMessage{
|
||||||
|
Transaction: transaction,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SubmitTransactionReplacementResponseMessage is an appmessage corresponding to
|
||||||
|
// its respective RPC message
|
||||||
|
type SubmitTransactionReplacementResponseMessage struct {
|
||||||
|
baseMessage
|
||||||
|
TransactionID string
|
||||||
|
ReplacedTransaction *RPCTransaction
|
||||||
|
|
||||||
|
Error *RPCError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Command returns the protocol command string for the message
|
||||||
|
func (msg *SubmitTransactionReplacementResponseMessage) Command() MessageCommand {
|
||||||
|
return CmdSubmitTransactionReplacementResponseMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewSubmitTransactionReplacementResponseMessage returns a instance of the message
|
||||||
|
func NewSubmitTransactionReplacementResponseMessage(transactionID string) *SubmitTransactionReplacementResponseMessage {
|
||||||
|
return &SubmitTransactionReplacementResponseMessage{
|
||||||
|
TransactionID: transactionID,
|
||||||
|
}
|
||||||
|
}
|
@ -4,23 +4,21 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/utxoindex"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
|
|
||||||
infrastructuredatabase "github.com/kaspanet/kaspad/infrastructure/db/database"
|
"github.com/kaspanet/kaspad/domain/miningmanager/mempool"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/addressmanager"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/id"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol"
|
"github.com/kaspanet/kaspad/app/protocol"
|
||||||
"github.com/kaspanet/kaspad/app/rpc"
|
"github.com/kaspanet/kaspad/app/rpc"
|
||||||
|
"github.com/kaspanet/kaspad/domain"
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus"
|
||||||
|
"github.com/kaspanet/kaspad/domain/utxoindex"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/config"
|
"github.com/kaspanet/kaspad/infrastructure/config"
|
||||||
|
infrastructuredatabase "github.com/kaspanet/kaspad/infrastructure/db/database"
|
||||||
|
"github.com/kaspanet/kaspad/infrastructure/network/addressmanager"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/connmanager"
|
"github.com/kaspanet/kaspad/infrastructure/network/connmanager"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/dnsseed"
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter"
|
"github.com/kaspanet/kaspad/infrastructure/network/netadapter"
|
||||||
|
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/id"
|
||||||
"github.com/kaspanet/kaspad/util/panics"
|
"github.com/kaspanet/kaspad/util/panics"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -50,8 +48,6 @@ func (a *ComponentManager) Start() {
|
|||||||
panics.Exit(log, fmt.Sprintf("Error starting the net adapter: %+v", err))
|
panics.Exit(log, fmt.Sprintf("Error starting the net adapter: %+v", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
a.maybeSeedFromDNS()
|
|
||||||
|
|
||||||
a.connectionManager.Start()
|
a.connectionManager.Start()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -72,6 +68,9 @@ func (a *ComponentManager) Stop() {
|
|||||||
log.Errorf("Error stopping the net adapter: %+v", err)
|
log.Errorf("Error stopping the net adapter: %+v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
a.protocolManager.Close()
|
||||||
|
close(a.protocolManager.Context().Domain().ConsensusEventsChannel())
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -80,7 +79,16 @@ func (a *ComponentManager) Stop() {
|
|||||||
func NewComponentManager(cfg *config.Config, db infrastructuredatabase.Database, interrupt chan<- struct{}) (
|
func NewComponentManager(cfg *config.Config, db infrastructuredatabase.Database, interrupt chan<- struct{}) (
|
||||||
*ComponentManager, error) {
|
*ComponentManager, error) {
|
||||||
|
|
||||||
domain, err := domain.New(cfg.ActiveNetParams, db, cfg.IsArchivalNode)
|
consensusConfig := consensus.Config{
|
||||||
|
Params: *cfg.ActiveNetParams,
|
||||||
|
IsArchival: cfg.IsArchivalNode,
|
||||||
|
EnableSanityCheckPruningUTXOSet: cfg.EnableSanityCheckPruningUTXOSet,
|
||||||
|
}
|
||||||
|
mempoolConfig := mempool.DefaultConfig(&consensusConfig.Params)
|
||||||
|
mempoolConfig.MaximumOrphanTransactionCount = cfg.MaxOrphanTxs
|
||||||
|
mempoolConfig.MinimumRelayTransactionFee = cfg.MinRelayTxFee
|
||||||
|
|
||||||
|
domain, err := domain.New(&consensusConfig, mempoolConfig, db)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -90,14 +98,18 @@ func NewComponentManager(cfg *config.Config, db infrastructuredatabase.Database,
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
addressManager, err := addressmanager.New(addressmanager.NewConfig(cfg))
|
addressManager, err := addressmanager.New(addressmanager.NewConfig(cfg), db)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
var utxoIndex *utxoindex.UTXOIndex
|
var utxoIndex *utxoindex.UTXOIndex
|
||||||
if cfg.UTXOIndex {
|
if cfg.UTXOIndex {
|
||||||
utxoIndex = utxoindex.New(domain.Consensus(), db)
|
utxoIndex, err = utxoindex.New(domain, db)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
log.Infof("UTXO index started")
|
log.Infof("UTXO index started")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -109,7 +121,7 @@ func NewComponentManager(cfg *config.Config, db infrastructuredatabase.Database,
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
rpcManager := setupRPC(cfg, domain, netAdapter, protocolManager, connectionManager, addressManager, utxoIndex, interrupt)
|
rpcManager := setupRPC(cfg, domain, netAdapter, protocolManager, connectionManager, addressManager, utxoIndex, domain.ConsensusEventsChannel(), interrupt)
|
||||||
|
|
||||||
return &ComponentManager{
|
return &ComponentManager{
|
||||||
cfg: cfg,
|
cfg: cfg,
|
||||||
@ -130,6 +142,7 @@ func setupRPC(
|
|||||||
connectionManager *connmanager.ConnectionManager,
|
connectionManager *connmanager.ConnectionManager,
|
||||||
addressManager *addressmanager.AddressManager,
|
addressManager *addressmanager.AddressManager,
|
||||||
utxoIndex *utxoindex.UTXOIndex,
|
utxoIndex *utxoindex.UTXOIndex,
|
||||||
|
consensusEventsChan chan externalapi.ConsensusEvent,
|
||||||
shutDownChan chan<- struct{},
|
shutDownChan chan<- struct{},
|
||||||
) *rpc.Manager {
|
) *rpc.Manager {
|
||||||
|
|
||||||
@ -141,30 +154,15 @@ func setupRPC(
|
|||||||
connectionManager,
|
connectionManager,
|
||||||
addressManager,
|
addressManager,
|
||||||
utxoIndex,
|
utxoIndex,
|
||||||
|
consensusEventsChan,
|
||||||
shutDownChan,
|
shutDownChan,
|
||||||
)
|
)
|
||||||
protocolManager.SetOnBlockAddedToDAGHandler(rpcManager.NotifyBlockAddedToDAG)
|
protocolManager.SetOnNewBlockTemplateHandler(rpcManager.NotifyNewBlockTemplate)
|
||||||
|
protocolManager.SetOnPruningPointUTXOSetOverrideHandler(rpcManager.NotifyPruningPointUTXOSetOverride)
|
||||||
|
|
||||||
return rpcManager
|
return rpcManager
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *ComponentManager) maybeSeedFromDNS() {
|
|
||||||
if !a.cfg.DisableDNSSeed {
|
|
||||||
dnsseed.SeedFromDNS(a.cfg.NetParams(), a.cfg.DNSSeed, appmessage.SFNodeNetwork, false, nil,
|
|
||||||
a.cfg.Lookup, func(addresses []*appmessage.NetAddress) {
|
|
||||||
// Kaspad uses a lookup of the dns seeder here. Since seeder returns
|
|
||||||
// IPs of nodes and not its own IP, we can not know real IP of
|
|
||||||
// source. So we'll take first returned address as source.
|
|
||||||
a.addressManager.AddAddresses(addresses...)
|
|
||||||
})
|
|
||||||
|
|
||||||
dnsseed.SeedFromGRPC(a.cfg.NetParams(), a.cfg.GRPCSeed, appmessage.SFNodeNetwork, false, nil,
|
|
||||||
func(addresses []*appmessage.NetAddress) {
|
|
||||||
a.addressManager.AddAddresses(addresses...)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// P2PNodeID returns the network ID associated with this ComponentManager
|
// P2PNodeID returns the network ID associated with this ComponentManager
|
||||||
func (a *ComponentManager) P2PNodeID() *id.ID {
|
func (a *ComponentManager) P2PNodeID() *id.ID {
|
||||||
return a.netAdapter.ID()
|
return a.netAdapter.ID()
|
||||||
|
57
app/db_version.go
Normal file
57
app/db_version.go
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
const currentDatabaseVersion = 1
|
||||||
|
|
||||||
|
func checkDatabaseVersion(dbPath string) (err error) {
|
||||||
|
versionFileName := versionFilePath(dbPath)
|
||||||
|
|
||||||
|
versionBytes, err := os.ReadFile(versionFileName)
|
||||||
|
if err != nil {
|
||||||
|
if os.IsNotExist(err) { // If version file doesn't exist, we assume that the database is new
|
||||||
|
return createDatabaseVersionFile(dbPath, versionFileName)
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
databaseVersion, err := strconv.Atoi(string(versionBytes))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if databaseVersion != currentDatabaseVersion {
|
||||||
|
// TODO: Once there's more then one database version, it might make sense to add upgrade logic at this point
|
||||||
|
return errors.Errorf("Invalid database version %d. Expected version: %d", databaseVersion, currentDatabaseVersion)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func createDatabaseVersionFile(dbPath string, versionFileName string) error {
|
||||||
|
err := os.MkdirAll(dbPath, 0700)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
versionFile, err := os.Create(versionFileName)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
defer versionFile.Close()
|
||||||
|
|
||||||
|
versionString := strconv.Itoa(currentDatabaseVersion)
|
||||||
|
_, err = versionFile.Write([]byte(versionString))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func versionFilePath(dbPath string) string {
|
||||||
|
dbVersionFileName := path.Join(dbPath, "version")
|
||||||
|
return dbVersionFileName
|
||||||
|
}
|
@ -7,8 +7,6 @@ package app
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/kaspanet/kaspad/infrastructure/logger"
|
"github.com/kaspanet/kaspad/infrastructure/logger"
|
||||||
"github.com/kaspanet/kaspad/util/panics"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var log, _ = logger.Get(logger.SubsystemTags.KASD)
|
var log = logger.RegisterSubSystem("KASD")
|
||||||
var spawn = panics.GoroutineWrapperFunc(log)
|
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
package common
|
package common
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
peerpkg "github.com/kaspanet/kaspad/app/protocol/peer"
|
||||||
|
routerpkg "github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
@ -8,7 +10,18 @@ import (
|
|||||||
|
|
||||||
// DefaultTimeout is the default duration to wait for enqueuing/dequeuing
|
// DefaultTimeout is the default duration to wait for enqueuing/dequeuing
|
||||||
// to/from routes.
|
// to/from routes.
|
||||||
const DefaultTimeout = 30 * time.Second
|
const DefaultTimeout = 120 * time.Second
|
||||||
|
|
||||||
// ErrPeerWithSameIDExists signifies that a peer with the same ID already exist.
|
// ErrPeerWithSameIDExists signifies that a peer with the same ID already exist.
|
||||||
var ErrPeerWithSameIDExists = errors.New("ready peer with the same ID already exists")
|
var ErrPeerWithSameIDExists = errors.New("ready peer with the same ID already exists")
|
||||||
|
|
||||||
|
type flowExecuteFunc func(peer *peerpkg.Peer)
|
||||||
|
|
||||||
|
// Flow is a a data structure that is used in order to associate a p2p flow to some route in a router.
|
||||||
|
type Flow struct {
|
||||||
|
Name string
|
||||||
|
ExecuteFunc flowExecuteFunc
|
||||||
|
}
|
||||||
|
|
||||||
|
// FlowInitializeFunc is a function that is used in order to initialize a flow
|
||||||
|
type FlowInitializeFunc func(route *routerpkg.Route, peer *peerpkg.Peer) error
|
||||||
|
@ -1,65 +1,73 @@
|
|||||||
package flowcontext
|
package flowcontext
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
peerpkg "github.com/kaspanet/kaspad/app/protocol/peer"
|
peerpkg "github.com/kaspanet/kaspad/app/protocol/peer"
|
||||||
|
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/ruleerrors"
|
"github.com/kaspanet/kaspad/domain/consensus/ruleerrors"
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
"github.com/kaspanet/kaspad/app/appmessage"
|
||||||
"github.com/kaspanet/kaspad/app/protocol/flows/blockrelay"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// OnNewBlock updates the mempool after a new block arrival, and
|
// OnNewBlock updates the mempool after a new block arrival, and
|
||||||
// relays newly unorphaned transactions and possibly rebroadcast
|
// relays newly unorphaned transactions and possibly rebroadcast
|
||||||
// manually added transactions when not in IBD.
|
// manually added transactions when not in IBD.
|
||||||
func (f *FlowContext) OnNewBlock(block *externalapi.DomainBlock,
|
func (f *FlowContext) OnNewBlock(block *externalapi.DomainBlock) error {
|
||||||
blockInsertionResult *externalapi.BlockInsertionResult) error {
|
|
||||||
|
|
||||||
hash := consensushashing.BlockHash(block)
|
hash := consensushashing.BlockHash(block)
|
||||||
log.Debugf("OnNewBlock start for block %s", hash)
|
log.Tracef("OnNewBlock start for block %s", hash)
|
||||||
defer log.Debugf("OnNewBlock end for block %s", hash)
|
defer log.Tracef("OnNewBlock end for block %s", hash)
|
||||||
|
|
||||||
unorphaningResults, err := f.UnorphanBlocks(block)
|
unorphanedBlocks, err := f.UnorphanBlocks(block)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Debugf("OnNewBlock: block %s unorphaned %d blocks", hash, len(unorphaningResults))
|
log.Debugf("OnNewBlock: block %s unorphaned %d blocks", hash, len(unorphanedBlocks))
|
||||||
|
|
||||||
newBlocks := []*externalapi.DomainBlock{block}
|
newBlocks := []*externalapi.DomainBlock{block}
|
||||||
newBlockInsertionResults := []*externalapi.BlockInsertionResult{blockInsertionResult}
|
newBlocks = append(newBlocks, unorphanedBlocks...)
|
||||||
for _, unorphaningResult := range unorphaningResults {
|
|
||||||
newBlocks = append(newBlocks, unorphaningResult.block)
|
|
||||||
newBlockInsertionResults = append(newBlockInsertionResults, unorphaningResult.blockInsertionResult)
|
|
||||||
}
|
|
||||||
|
|
||||||
for i, newBlock := range newBlocks {
|
allAcceptedTransactions := make([]*externalapi.DomainTransaction, 0)
|
||||||
|
for _, newBlock := range newBlocks {
|
||||||
log.Debugf("OnNewBlock: passing block %s transactions to mining manager", hash)
|
log.Debugf("OnNewBlock: passing block %s transactions to mining manager", hash)
|
||||||
_, err = f.Domain().MiningManager().HandleNewBlockTransactions(newBlock.Transactions)
|
acceptedTransactions, err := f.Domain().MiningManager().HandleNewBlockTransactions(newBlock.Transactions)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
allAcceptedTransactions = append(allAcceptedTransactions, acceptedTransactions...)
|
||||||
|
}
|
||||||
|
|
||||||
if f.onBlockAddedToDAGHandler != nil {
|
return f.broadcastTransactionsAfterBlockAdded(newBlocks, allAcceptedTransactions)
|
||||||
log.Debugf("OnNewBlock: calling f.onBlockAddedToDAGHandler for block %s", hash)
|
}
|
||||||
blockInsertionResult = newBlockInsertionResults[i]
|
|
||||||
err := f.onBlockAddedToDAGHandler(newBlock, blockInsertionResult)
|
// OnNewBlockTemplate calls the handler function whenever a new block template is available for miners.
|
||||||
if err != nil {
|
func (f *FlowContext) OnNewBlockTemplate() error {
|
||||||
return err
|
// Clear current template cache. Note we call this even if the handler is nil, in order to keep the
|
||||||
}
|
// state consistent without dependency on external event registration
|
||||||
}
|
f.Domain().MiningManager().ClearBlockTemplate()
|
||||||
|
if f.onNewBlockTemplateHandler != nil {
|
||||||
|
return f.onNewBlockTemplateHandler()
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *FlowContext) broadcastTransactionsAfterBlockAdded(
|
// OnPruningPointUTXOSetOverride calls the handler function whenever the UTXO set
|
||||||
block *externalapi.DomainBlock, transactionsAcceptedToMempool []*externalapi.DomainTransaction) error {
|
// resets due to pruning point change via IBD.
|
||||||
|
func (f *FlowContext) OnPruningPointUTXOSetOverride() error {
|
||||||
|
if f.onPruningPointUTXOSetOverrideHandler != nil {
|
||||||
|
return f.onPruningPointUTXOSetOverrideHandler()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
f.updateTransactionsToRebroadcast(block)
|
func (f *FlowContext) broadcastTransactionsAfterBlockAdded(
|
||||||
|
addedBlocks []*externalapi.DomainBlock, transactionsAcceptedToMempool []*externalapi.DomainTransaction) error {
|
||||||
|
|
||||||
// Don't relay transactions when in IBD.
|
// Don't relay transactions when in IBD.
|
||||||
if f.IsIBDRunning() {
|
if f.IsIBDRunning() {
|
||||||
@ -68,7 +76,12 @@ func (f *FlowContext) broadcastTransactionsAfterBlockAdded(
|
|||||||
|
|
||||||
var txIDsToRebroadcast []*externalapi.DomainTransactionID
|
var txIDsToRebroadcast []*externalapi.DomainTransactionID
|
||||||
if f.shouldRebroadcastTransactions() {
|
if f.shouldRebroadcastTransactions() {
|
||||||
txIDsToRebroadcast = f.txIDsToRebroadcast()
|
txsToRebroadcast, err := f.Domain().MiningManager().RevalidateHighPriorityTransactions()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
txIDsToRebroadcast = consensushashing.TransactionIDs(txsToRebroadcast)
|
||||||
|
f.lastRebroadcastTime = time.Now()
|
||||||
}
|
}
|
||||||
|
|
||||||
txIDsToBroadcast := make([]*externalapi.DomainTransactionID, len(transactionsAcceptedToMempool)+len(txIDsToRebroadcast))
|
txIDsToBroadcast := make([]*externalapi.DomainTransactionID, len(transactionsAcceptedToMempool)+len(txIDsToRebroadcast))
|
||||||
@ -79,33 +92,33 @@ func (f *FlowContext) broadcastTransactionsAfterBlockAdded(
|
|||||||
for i, txID := range txIDsToRebroadcast {
|
for i, txID := range txIDsToRebroadcast {
|
||||||
txIDsToBroadcast[offset+i] = txID
|
txIDsToBroadcast[offset+i] = txID
|
||||||
}
|
}
|
||||||
|
return f.EnqueueTransactionIDsForPropagation(txIDsToBroadcast)
|
||||||
if len(txIDsToBroadcast) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if len(txIDsToBroadcast) > appmessage.MaxInvPerTxInvMsg {
|
|
||||||
txIDsToBroadcast = txIDsToBroadcast[:appmessage.MaxInvPerTxInvMsg]
|
|
||||||
}
|
|
||||||
inv := appmessage.NewMsgInvTransaction(txIDsToBroadcast)
|
|
||||||
return f.Broadcast(inv)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// SharedRequestedBlocks returns a *blockrelay.SharedRequestedBlocks for sharing
|
// SharedRequestedBlocks returns a *blockrelay.SharedRequestedBlocks for sharing
|
||||||
// data about requested blocks between different peers.
|
// data about requested blocks between different peers.
|
||||||
func (f *FlowContext) SharedRequestedBlocks() *blockrelay.SharedRequestedBlocks {
|
func (f *FlowContext) SharedRequestedBlocks() *SharedRequestedBlocks {
|
||||||
return f.sharedRequestedBlocks
|
return f.sharedRequestedBlocks
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddBlock adds the given block to the DAG and propagates it.
|
// AddBlock adds the given block to the DAG and propagates it.
|
||||||
func (f *FlowContext) AddBlock(block *externalapi.DomainBlock) error {
|
func (f *FlowContext) AddBlock(block *externalapi.DomainBlock) error {
|
||||||
blockInsertionResult, err := f.Domain().Consensus().ValidateAndInsertBlock(block)
|
if len(block.Transactions) == 0 {
|
||||||
|
return protocolerrors.Errorf(false, "cannot add header only block")
|
||||||
|
}
|
||||||
|
|
||||||
|
err := f.Domain().Consensus().ValidateAndInsertBlock(block, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errors.As(err, &ruleerrors.RuleError{}) {
|
if errors.As(err, &ruleerrors.RuleError{}) {
|
||||||
log.Warnf("Validation failed for block %s: %s", consensushashing.BlockHash(block), err)
|
log.Warnf("Validation failed for block %s: %s", consensushashing.BlockHash(block), err)
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = f.OnNewBlock(block, blockInsertionResult)
|
err = f.OnNewBlockTemplate()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = f.OnNewBlock(block)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -130,7 +143,7 @@ func (f *FlowContext) TrySetIBDRunning(ibdPeer *peerpkg.Peer) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
f.ibdPeer = ibdPeer
|
f.ibdPeer = ibdPeer
|
||||||
log.Infof("IBD started")
|
log.Infof("IBD started with peer %s", ibdPeer)
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
@ -145,7 +158,6 @@ func (f *FlowContext) UnsetIBDRunning() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
f.ibdPeer = nil
|
f.ibdPeer = nil
|
||||||
log.Infof("IBD finished")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// IBDPeer returns the current IBD peer or null if the node is not
|
// IBDPeer returns the current IBD peer or null if the node is not
|
||||||
|
@ -2,6 +2,7 @@ package flowcontext
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
|
"strings"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
||||||
@ -9,6 +10,11 @@ import (
|
|||||||
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// ErrPingTimeout signifies that a ping operation timed out.
|
||||||
|
ErrPingTimeout = protocolerrors.New(false, "timeout expired on ping")
|
||||||
|
)
|
||||||
|
|
||||||
// HandleError handles an error from a flow,
|
// HandleError handles an error from a flow,
|
||||||
// It sends the error to errChan if isStopping == 0 and increments isStopping
|
// It sends the error to errChan if isStopping == 0 and increments isStopping
|
||||||
//
|
//
|
||||||
@ -18,14 +24,26 @@ import (
|
|||||||
func (*FlowContext) HandleError(err error, flowName string, isStopping *uint32, errChan chan<- error) {
|
func (*FlowContext) HandleError(err error, flowName string, isStopping *uint32, errChan chan<- error) {
|
||||||
isErrRouteClosed := errors.Is(err, router.ErrRouteClosed)
|
isErrRouteClosed := errors.Is(err, router.ErrRouteClosed)
|
||||||
if !isErrRouteClosed {
|
if !isErrRouteClosed {
|
||||||
if protocolErr := &(protocolerrors.ProtocolError{}); !errors.As(err, &protocolErr) {
|
if protocolErr := (protocolerrors.ProtocolError{}); !errors.As(err, &protocolErr) {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
if errors.Is(err, ErrPingTimeout) {
|
||||||
|
// Avoid printing the call stack on ping timeouts, since users get panicked and this case is not interesting
|
||||||
log.Errorf("error from %s: %s", flowName, err)
|
log.Errorf("error from %s: %s", flowName, err)
|
||||||
|
} else {
|
||||||
|
// Explain to the user that this is not a panic, but only a protocol error with a specific peer
|
||||||
|
logFrame := strings.Repeat("=", 52)
|
||||||
|
log.Errorf("Non-critical peer protocol error from %s, printing the full stack for debug purposes: \n%s\n%+v \n%s",
|
||||||
|
flowName, logFrame, err, logFrame)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if atomic.AddUint32(isStopping, 1) == 1 {
|
if atomic.AddUint32(isStopping, 1) == 1 {
|
||||||
errChan <- err
|
errChan <- err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsRecoverableError returns whether the error is recoverable
|
||||||
|
func (*FlowContext) IsRecoverableError(err error) bool {
|
||||||
|
return err == nil || errors.Is(err, router.ErrRouteClosed) || errors.As(err, &protocolerrors.ProtocolError{})
|
||||||
|
}
|
||||||
|
@ -1,16 +1,15 @@
|
|||||||
package flowcontext
|
package flowcontext
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/kaspanet/kaspad/util/mstime"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/kaspanet/kaspad/util/mstime"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain"
|
"github.com/kaspanet/kaspad/domain"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/flows/blockrelay"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/flows/transactionrelay"
|
|
||||||
peerpkg "github.com/kaspanet/kaspad/app/protocol/peer"
|
peerpkg "github.com/kaspanet/kaspad/app/protocol/peer"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/config"
|
"github.com/kaspanet/kaspad/infrastructure/config"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/addressmanager"
|
"github.com/kaspanet/kaspad/infrastructure/network/addressmanager"
|
||||||
@ -19,9 +18,12 @@ import (
|
|||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/id"
|
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/id"
|
||||||
)
|
)
|
||||||
|
|
||||||
// OnBlockAddedToDAGHandler is a handler function that's triggered
|
// OnNewBlockTemplateHandler is a handler function that's triggered when a new block template is available
|
||||||
// when a block is added to the DAG
|
type OnNewBlockTemplateHandler func() error
|
||||||
type OnBlockAddedToDAGHandler func(block *externalapi.DomainBlock, blockInsertionResult *externalapi.BlockInsertionResult) error
|
|
||||||
|
// OnPruningPointUTXOSetOverrideHandler is a handle function that's triggered whenever the UTXO set
|
||||||
|
// resets due to pruning point change via IBD.
|
||||||
|
type OnPruningPointUTXOSetOverrideHandler func() error
|
||||||
|
|
||||||
// OnTransactionAddedToMempoolHandler is a handler function that's triggered
|
// OnTransactionAddedToMempoolHandler is a handler function that's triggered
|
||||||
// when a transaction is added to the mempool
|
// when a transaction is added to the mempool
|
||||||
@ -38,15 +40,14 @@ type FlowContext struct {
|
|||||||
|
|
||||||
timeStarted int64
|
timeStarted int64
|
||||||
|
|
||||||
onBlockAddedToDAGHandler OnBlockAddedToDAGHandler
|
onNewBlockTemplateHandler OnNewBlockTemplateHandler
|
||||||
|
onPruningPointUTXOSetOverrideHandler OnPruningPointUTXOSetOverrideHandler
|
||||||
onTransactionAddedToMempoolHandler OnTransactionAddedToMempoolHandler
|
onTransactionAddedToMempoolHandler OnTransactionAddedToMempoolHandler
|
||||||
|
|
||||||
transactionsToRebroadcastLock sync.Mutex
|
|
||||||
transactionsToRebroadcast map[externalapi.DomainTransactionID]*externalapi.DomainTransaction
|
|
||||||
lastRebroadcastTime time.Time
|
lastRebroadcastTime time.Time
|
||||||
sharedRequestedTransactions *transactionrelay.SharedRequestedTransactions
|
sharedRequestedTransactions *SharedRequestedTransactions
|
||||||
|
|
||||||
sharedRequestedBlocks *blockrelay.SharedRequestedBlocks
|
sharedRequestedBlocks *SharedRequestedBlocks
|
||||||
|
|
||||||
ibdPeer *peerpkg.Peer
|
ibdPeer *peerpkg.Peer
|
||||||
ibdPeerMutex sync.RWMutex
|
ibdPeerMutex sync.RWMutex
|
||||||
@ -56,6 +57,12 @@ type FlowContext struct {
|
|||||||
|
|
||||||
orphans map[externalapi.DomainHash]*externalapi.DomainBlock
|
orphans map[externalapi.DomainHash]*externalapi.DomainBlock
|
||||||
orphansMutex sync.RWMutex
|
orphansMutex sync.RWMutex
|
||||||
|
|
||||||
|
transactionIDsToPropagate []*externalapi.DomainTransactionID
|
||||||
|
lastTransactionIDPropagationTime time.Time
|
||||||
|
transactionIDPropagationLock sync.Mutex
|
||||||
|
|
||||||
|
shutdownChan chan struct{}
|
||||||
}
|
}
|
||||||
|
|
||||||
// New returns a new instance of FlowContext.
|
// New returns a new instance of FlowContext.
|
||||||
@ -68,18 +75,41 @@ func New(cfg *config.Config, domain domain.Domain, addressManager *addressmanage
|
|||||||
domain: domain,
|
domain: domain,
|
||||||
addressManager: addressManager,
|
addressManager: addressManager,
|
||||||
connectionManager: connectionManager,
|
connectionManager: connectionManager,
|
||||||
sharedRequestedTransactions: transactionrelay.NewSharedRequestedTransactions(),
|
sharedRequestedTransactions: NewSharedRequestedTransactions(),
|
||||||
sharedRequestedBlocks: blockrelay.NewSharedRequestedBlocks(),
|
sharedRequestedBlocks: NewSharedRequestedBlocks(),
|
||||||
peers: make(map[id.ID]*peerpkg.Peer),
|
peers: make(map[id.ID]*peerpkg.Peer),
|
||||||
transactionsToRebroadcast: make(map[externalapi.DomainTransactionID]*externalapi.DomainTransaction),
|
|
||||||
orphans: make(map[externalapi.DomainHash]*externalapi.DomainBlock),
|
orphans: make(map[externalapi.DomainHash]*externalapi.DomainBlock),
|
||||||
timeStarted: mstime.Now().UnixMilliseconds(),
|
timeStarted: mstime.Now().UnixMilliseconds(),
|
||||||
|
transactionIDsToPropagate: []*externalapi.DomainTransactionID{},
|
||||||
|
lastTransactionIDPropagationTime: time.Now(),
|
||||||
|
shutdownChan: make(chan struct{}),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetOnBlockAddedToDAGHandler sets the onBlockAddedToDAG handler
|
// Close signals to all flows the the protocol manager is closed.
|
||||||
func (f *FlowContext) SetOnBlockAddedToDAGHandler(onBlockAddedToDAGHandler OnBlockAddedToDAGHandler) {
|
func (f *FlowContext) Close() {
|
||||||
f.onBlockAddedToDAGHandler = onBlockAddedToDAGHandler
|
close(f.shutdownChan)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ShutdownChan is a chan where flows can subscribe to shutdown
|
||||||
|
// event.
|
||||||
|
func (f *FlowContext) ShutdownChan() <-chan struct{} {
|
||||||
|
return f.shutdownChan
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsNearlySynced returns whether current consensus is considered synced or close to being synced.
|
||||||
|
func (f *FlowContext) IsNearlySynced() (bool, error) {
|
||||||
|
return f.Domain().Consensus().IsNearlySynced()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetOnNewBlockTemplateHandler sets the onNewBlockTemplateHandler handler
|
||||||
|
func (f *FlowContext) SetOnNewBlockTemplateHandler(onNewBlockTemplateHandler OnNewBlockTemplateHandler) {
|
||||||
|
f.onNewBlockTemplateHandler = onNewBlockTemplateHandler
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetOnPruningPointUTXOSetOverrideHandler sets the onPruningPointUTXOSetOverrideHandler handler
|
||||||
|
func (f *FlowContext) SetOnPruningPointUTXOSetOverrideHandler(onPruningPointUTXOSetOverrideHandler OnPruningPointUTXOSetOverrideHandler) {
|
||||||
|
f.onPruningPointUTXOSetOverrideHandler = onPruningPointUTXOSetOverrideHandler
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetOnTransactionAddedToMempoolHandler sets the onTransactionAddedToMempool handler
|
// SetOnTransactionAddedToMempoolHandler sets the onTransactionAddedToMempool handler
|
||||||
|
@ -2,8 +2,6 @@ package flowcontext
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/kaspanet/kaspad/infrastructure/logger"
|
"github.com/kaspanet/kaspad/infrastructure/logger"
|
||||||
"github.com/kaspanet/kaspad/util/panics"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var log, _ = logger.Get(logger.SubsystemTags.PROT)
|
var log = logger.RegisterSubSystem("PROT")
|
||||||
var spawn = panics.GoroutineWrapperFunc(log)
|
|
||||||
|
@ -72,3 +72,10 @@ func (f *FlowContext) Peers() []*peerpkg.Peer {
|
|||||||
}
|
}
|
||||||
return peers
|
return peers
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// HasPeers returns whether there are currently active peers
|
||||||
|
func (f *FlowContext) HasPeers() bool {
|
||||||
|
f.peersMutex.RLock()
|
||||||
|
defer f.peersMutex.RUnlock()
|
||||||
|
return len(f.peers) > 0
|
||||||
|
}
|
||||||
|
@ -15,12 +15,6 @@ import (
|
|||||||
// on: 2^orphanResolutionRange * PHANTOM K.
|
// on: 2^orphanResolutionRange * PHANTOM K.
|
||||||
const maxOrphans = 600
|
const maxOrphans = 600
|
||||||
|
|
||||||
// UnorphaningResult is the result of unorphaning a block
|
|
||||||
type UnorphaningResult struct {
|
|
||||||
block *externalapi.DomainBlock
|
|
||||||
blockInsertionResult *externalapi.BlockInsertionResult
|
|
||||||
}
|
|
||||||
|
|
||||||
// AddOrphan adds the block to the orphan set
|
// AddOrphan adds the block to the orphan set
|
||||||
func (f *FlowContext) AddOrphan(orphanBlock *externalapi.DomainBlock) {
|
func (f *FlowContext) AddOrphan(orphanBlock *externalapi.DomainBlock) {
|
||||||
f.orphansMutex.Lock()
|
f.orphansMutex.Lock()
|
||||||
@ -57,7 +51,7 @@ func (f *FlowContext) IsOrphan(blockHash *externalapi.DomainHash) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// UnorphanBlocks removes the block from the orphan set, and remove all of the blocks that are not orphans anymore.
|
// UnorphanBlocks removes the block from the orphan set, and remove all of the blocks that are not orphans anymore.
|
||||||
func (f *FlowContext) UnorphanBlocks(rootBlock *externalapi.DomainBlock) ([]*UnorphaningResult, error) {
|
func (f *FlowContext) UnorphanBlocks(rootBlock *externalapi.DomainBlock) ([]*externalapi.DomainBlock, error) {
|
||||||
f.orphansMutex.Lock()
|
f.orphansMutex.Lock()
|
||||||
defer f.orphansMutex.Unlock()
|
defer f.orphansMutex.Unlock()
|
||||||
|
|
||||||
@ -66,17 +60,17 @@ func (f *FlowContext) UnorphanBlocks(rootBlock *externalapi.DomainBlock) ([]*Uno
|
|||||||
rootBlockHash := consensushashing.BlockHash(rootBlock)
|
rootBlockHash := consensushashing.BlockHash(rootBlock)
|
||||||
processQueue := f.addChildOrphansToProcessQueue(rootBlockHash, []externalapi.DomainHash{})
|
processQueue := f.addChildOrphansToProcessQueue(rootBlockHash, []externalapi.DomainHash{})
|
||||||
|
|
||||||
var unorphaningResults []*UnorphaningResult
|
var unorphanedBlocks []*externalapi.DomainBlock
|
||||||
for len(processQueue) > 0 {
|
for len(processQueue) > 0 {
|
||||||
var orphanHash externalapi.DomainHash
|
var orphanHash externalapi.DomainHash
|
||||||
orphanHash, processQueue = processQueue[0], processQueue[1:]
|
orphanHash, processQueue = processQueue[0], processQueue[1:]
|
||||||
orphanBlock := f.orphans[orphanHash]
|
orphanBlock := f.orphans[orphanHash]
|
||||||
|
|
||||||
log.Debugf("Considering to unorphan block %s with parents %s",
|
log.Debugf("Considering to unorphan block %s with parents %s",
|
||||||
orphanHash, orphanBlock.Header.ParentHashes())
|
orphanHash, orphanBlock.Header.DirectParents())
|
||||||
|
|
||||||
canBeUnorphaned := true
|
canBeUnorphaned := true
|
||||||
for _, orphanBlockParentHash := range orphanBlock.Header.ParentHashes() {
|
for _, orphanBlockParentHash := range orphanBlock.Header.DirectParents() {
|
||||||
orphanBlockParentInfo, err := f.domain.Consensus().GetBlockInfo(orphanBlockParentHash)
|
orphanBlockParentInfo, err := f.domain.Consensus().GetBlockInfo(orphanBlockParentHash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -90,21 +84,18 @@ func (f *FlowContext) UnorphanBlocks(rootBlock *externalapi.DomainBlock) ([]*Uno
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if canBeUnorphaned {
|
if canBeUnorphaned {
|
||||||
blockInsertionResult, unorphaningSucceeded, err := f.unorphanBlock(orphanHash)
|
unorphaningSucceeded, err := f.unorphanBlock(orphanHash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if unorphaningSucceeded {
|
if unorphaningSucceeded {
|
||||||
unorphaningResults = append(unorphaningResults, &UnorphaningResult{
|
unorphanedBlocks = append(unorphanedBlocks, orphanBlock)
|
||||||
block: orphanBlock,
|
|
||||||
blockInsertionResult: blockInsertionResult,
|
|
||||||
})
|
|
||||||
processQueue = f.addChildOrphansToProcessQueue(&orphanHash, processQueue)
|
processQueue = f.addChildOrphansToProcessQueue(&orphanHash, processQueue)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return unorphaningResults, nil
|
return unorphanedBlocks, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// addChildOrphansToProcessQueue finds all child orphans of `blockHash`
|
// addChildOrphansToProcessQueue finds all child orphans of `blockHash`
|
||||||
@ -133,7 +124,7 @@ func (f *FlowContext) addChildOrphansToProcessQueue(blockHash *externalapi.Domai
|
|||||||
func (f *FlowContext) findChildOrphansOfBlock(blockHash *externalapi.DomainHash) []externalapi.DomainHash {
|
func (f *FlowContext) findChildOrphansOfBlock(blockHash *externalapi.DomainHash) []externalapi.DomainHash {
|
||||||
var childOrphans []externalapi.DomainHash
|
var childOrphans []externalapi.DomainHash
|
||||||
for orphanHash, orphanBlock := range f.orphans {
|
for orphanHash, orphanBlock := range f.orphans {
|
||||||
for _, orphanBlockParentHash := range orphanBlock.Header.ParentHashes() {
|
for _, orphanBlockParentHash := range orphanBlock.Header.DirectParents() {
|
||||||
if orphanBlockParentHash.Equal(blockHash) {
|
if orphanBlockParentHash.Equal(blockHash) {
|
||||||
childOrphans = append(childOrphans, orphanHash)
|
childOrphans = append(childOrphans, orphanHash)
|
||||||
break
|
break
|
||||||
@ -143,24 +134,24 @@ func (f *FlowContext) findChildOrphansOfBlock(blockHash *externalapi.DomainHash)
|
|||||||
return childOrphans
|
return childOrphans
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *FlowContext) unorphanBlock(orphanHash externalapi.DomainHash) (*externalapi.BlockInsertionResult, bool, error) {
|
func (f *FlowContext) unorphanBlock(orphanHash externalapi.DomainHash) (bool, error) {
|
||||||
orphanBlock, ok := f.orphans[orphanHash]
|
orphanBlock, ok := f.orphans[orphanHash]
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, false, errors.Errorf("attempted to unorphan a non-orphan block %s", orphanHash)
|
return false, errors.Errorf("attempted to unorphan a non-orphan block %s", orphanHash)
|
||||||
}
|
}
|
||||||
delete(f.orphans, orphanHash)
|
delete(f.orphans, orphanHash)
|
||||||
|
|
||||||
blockInsertionResult, err := f.domain.Consensus().ValidateAndInsertBlock(orphanBlock)
|
err := f.domain.Consensus().ValidateAndInsertBlock(orphanBlock, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errors.As(err, &ruleerrors.RuleError{}) {
|
if errors.As(err, &ruleerrors.RuleError{}) {
|
||||||
log.Warnf("Validation failed for orphan block %s: %s", orphanHash, err)
|
log.Warnf("Validation failed for orphan block %s: %s", orphanHash, err)
|
||||||
return nil, false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
return nil, false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Infof("Unorphaned block %s", orphanHash)
|
log.Infof("Unorphaned block %s", orphanHash)
|
||||||
return blockInsertionResult, true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetOrphanRoots returns the roots of the missing ancestors DAG of the given orphan
|
// GetOrphanRoots returns the roots of the missing ancestors DAG of the given orphan
|
||||||
@ -201,7 +192,7 @@ func (f *FlowContext) GetOrphanRoots(orphan *externalapi.DomainHash) ([]*externa
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, parent := range block.Header.ParentHashes() {
|
for _, parent := range block.Header.DirectParents() {
|
||||||
if !addedToQueueSet.Contains(parent) {
|
if !addedToQueueSet.Contains(parent) {
|
||||||
queue = append(queue, parent)
|
queue = append(queue, parent)
|
||||||
addedToQueueSet.Add(parent)
|
addedToQueueSet.Add(parent)
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
package blockrelay
|
package flowcontext
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sync"
|
"sync"
|
||||||
@ -13,13 +13,15 @@ type SharedRequestedBlocks struct {
|
|||||||
sync.Mutex
|
sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SharedRequestedBlocks) remove(hash *externalapi.DomainHash) {
|
// Remove removes a block from the set.
|
||||||
|
func (s *SharedRequestedBlocks) Remove(hash *externalapi.DomainHash) {
|
||||||
s.Lock()
|
s.Lock()
|
||||||
defer s.Unlock()
|
defer s.Unlock()
|
||||||
delete(s.blocks, *hash)
|
delete(s.blocks, *hash)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SharedRequestedBlocks) removeSet(blockHashes map[externalapi.DomainHash]struct{}) {
|
// RemoveSet removes a set of blocks from the set.
|
||||||
|
func (s *SharedRequestedBlocks) RemoveSet(blockHashes map[externalapi.DomainHash]struct{}) {
|
||||||
s.Lock()
|
s.Lock()
|
||||||
defer s.Unlock()
|
defer s.Unlock()
|
||||||
for hash := range blockHashes {
|
for hash := range blockHashes {
|
||||||
@ -27,7 +29,8 @@ func (s *SharedRequestedBlocks) removeSet(blockHashes map[externalapi.DomainHash
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SharedRequestedBlocks) addIfNotExists(hash *externalapi.DomainHash) (exists bool) {
|
// AddIfNotExists adds a block to the set if it doesn't exist yet.
|
||||||
|
func (s *SharedRequestedBlocks) AddIfNotExists(hash *externalapi.DomainHash) (exists bool) {
|
||||||
s.Lock()
|
s.Lock()
|
||||||
defer s.Unlock()
|
defer s.Unlock()
|
||||||
_, ok := s.blocks[*hash]
|
_, ok := s.blocks[*hash]
|
@ -1,4 +1,4 @@
|
|||||||
package transactionrelay
|
package flowcontext
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sync"
|
"sync"
|
||||||
@ -13,13 +13,15 @@ type SharedRequestedTransactions struct {
|
|||||||
sync.Mutex
|
sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SharedRequestedTransactions) remove(txID *externalapi.DomainTransactionID) {
|
// Remove removes a transaction from the set.
|
||||||
|
func (s *SharedRequestedTransactions) Remove(txID *externalapi.DomainTransactionID) {
|
||||||
s.Lock()
|
s.Lock()
|
||||||
defer s.Unlock()
|
defer s.Unlock()
|
||||||
delete(s.transactions, *txID)
|
delete(s.transactions, *txID)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SharedRequestedTransactions) removeMany(txIDs []*externalapi.DomainTransactionID) {
|
// RemoveMany removes a set of transactions from the set.
|
||||||
|
func (s *SharedRequestedTransactions) RemoveMany(txIDs []*externalapi.DomainTransactionID) {
|
||||||
s.Lock()
|
s.Lock()
|
||||||
defer s.Unlock()
|
defer s.Unlock()
|
||||||
for _, txID := range txIDs {
|
for _, txID := range txIDs {
|
||||||
@ -27,7 +29,8 @@ func (s *SharedRequestedTransactions) removeMany(txIDs []*externalapi.DomainTran
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *SharedRequestedTransactions) addIfNotExists(txID *externalapi.DomainTransactionID) (exists bool) {
|
// AddIfNotExists adds a transaction to the set if it doesn't exist yet.
|
||||||
|
func (s *SharedRequestedTransactions) AddIfNotExists(txID *externalapi.DomainTransactionID) (exists bool) {
|
||||||
s.Lock()
|
s.Lock()
|
||||||
defer s.Unlock()
|
defer s.Unlock()
|
||||||
_, ok := s.transactions[*txID]
|
_, ok := s.transactions[*txID]
|
@ -1,43 +0,0 @@
|
|||||||
package flowcontext
|
|
||||||
|
|
||||||
import "github.com/kaspanet/kaspad/util/mstime"
|
|
||||||
|
|
||||||
const (
|
|
||||||
maxSelectedParentTimeDiffToAllowMiningInMilliSeconds = 60 * 60 * 1000 // 1 Hour
|
|
||||||
)
|
|
||||||
|
|
||||||
// ShouldMine returns whether it's ok to use block template from this node
|
|
||||||
// for mining purposes.
|
|
||||||
func (f *FlowContext) ShouldMine() (bool, error) {
|
|
||||||
peers := f.Peers()
|
|
||||||
if len(peers) == 0 {
|
|
||||||
log.Debugf("The node is not connected, so ShouldMine returns false")
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if f.IsIBDRunning() {
|
|
||||||
log.Debugf("IBD is running, so ShouldMine returns false")
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
virtualSelectedParent, err := f.domain.Consensus().GetVirtualSelectedParent()
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
virtualSelectedParentHeader, err := f.domain.Consensus().GetBlockHeader(virtualSelectedParent)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
now := mstime.Now().UnixMilliseconds()
|
|
||||||
if now-virtualSelectedParentHeader.TimeInMilliseconds() < maxSelectedParentTimeDiffToAllowMiningInMilliSeconds {
|
|
||||||
log.Debugf("The selected tip timestamp is recent (%d), so ShouldMine returns true",
|
|
||||||
virtualSelectedParentHeader.TimeInMilliseconds())
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Debugf("The selected tip timestamp is old (%d), so ShouldMine returns false",
|
|
||||||
virtualSelectedParentHeader.TimeInMilliseconds())
|
|
||||||
return false, nil
|
|
||||||
}
|
|
@ -4,36 +4,22 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
"github.com/kaspanet/kaspad/app/appmessage"
|
||||||
"github.com/kaspanet/kaspad/app/protocol/flows/transactionrelay"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
||||||
)
|
)
|
||||||
|
|
||||||
// AddTransaction adds transaction to the mempool and propagates it.
|
// TransactionIDPropagationInterval is the interval between transaction IDs propagations
|
||||||
func (f *FlowContext) AddTransaction(tx *externalapi.DomainTransaction) error {
|
const TransactionIDPropagationInterval = 500 * time.Millisecond
|
||||||
f.transactionsToRebroadcastLock.Lock()
|
|
||||||
defer f.transactionsToRebroadcastLock.Unlock()
|
|
||||||
|
|
||||||
err := f.Domain().MiningManager().ValidateAndInsertTransaction(tx, false)
|
// AddTransaction adds transaction to the mempool and propagates it.
|
||||||
|
func (f *FlowContext) AddTransaction(tx *externalapi.DomainTransaction, allowOrphan bool) error {
|
||||||
|
acceptedTransactions, err := f.Domain().MiningManager().ValidateAndInsertTransaction(tx, true, allowOrphan)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
transactionID := consensushashing.TransactionID(tx)
|
acceptedTransactionIDs := consensushashing.TransactionIDs(acceptedTransactions)
|
||||||
f.transactionsToRebroadcast[*transactionID] = tx
|
return f.EnqueueTransactionIDsForPropagation(acceptedTransactionIDs)
|
||||||
inv := appmessage.NewMsgInvTransaction([]*externalapi.DomainTransactionID{transactionID})
|
|
||||||
return f.Broadcast(inv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *FlowContext) updateTransactionsToRebroadcast(block *externalapi.DomainBlock) {
|
|
||||||
f.transactionsToRebroadcastLock.Lock()
|
|
||||||
defer f.transactionsToRebroadcastLock.Unlock()
|
|
||||||
// Note: if the block is red, its transactions won't be rebroadcasted
|
|
||||||
// anymore, although they are not included in the UTXO set.
|
|
||||||
// This is probably ok, since red blocks are quite rare.
|
|
||||||
for _, tx := range block.Transactions {
|
|
||||||
delete(f.transactionsToRebroadcast, *consensushashing.TransactionID(tx))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *FlowContext) shouldRebroadcastTransactions() bool {
|
func (f *FlowContext) shouldRebroadcastTransactions() bool {
|
||||||
@ -41,22 +27,9 @@ func (f *FlowContext) shouldRebroadcastTransactions() bool {
|
|||||||
return time.Since(f.lastRebroadcastTime) > rebroadcastInterval
|
return time.Since(f.lastRebroadcastTime) > rebroadcastInterval
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *FlowContext) txIDsToRebroadcast() []*externalapi.DomainTransactionID {
|
|
||||||
f.transactionsToRebroadcastLock.Lock()
|
|
||||||
defer f.transactionsToRebroadcastLock.Unlock()
|
|
||||||
|
|
||||||
txIDs := make([]*externalapi.DomainTransactionID, len(f.transactionsToRebroadcast))
|
|
||||||
i := 0
|
|
||||||
for _, tx := range f.transactionsToRebroadcast {
|
|
||||||
txIDs[i] = consensushashing.TransactionID(tx)
|
|
||||||
i++
|
|
||||||
}
|
|
||||||
return txIDs
|
|
||||||
}
|
|
||||||
|
|
||||||
// SharedRequestedTransactions returns a *transactionrelay.SharedRequestedTransactions for sharing
|
// SharedRequestedTransactions returns a *transactionrelay.SharedRequestedTransactions for sharing
|
||||||
// data about requested transactions between different peers.
|
// data about requested transactions between different peers.
|
||||||
func (f *FlowContext) SharedRequestedTransactions() *transactionrelay.SharedRequestedTransactions {
|
func (f *FlowContext) SharedRequestedTransactions() *SharedRequestedTransactions {
|
||||||
return f.sharedRequestedTransactions
|
return f.sharedRequestedTransactions
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,3 +40,42 @@ func (f *FlowContext) OnTransactionAddedToMempool() {
|
|||||||
f.onTransactionAddedToMempoolHandler()
|
f.onTransactionAddedToMempoolHandler()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EnqueueTransactionIDsForPropagation add the given transactions IDs to a set of IDs to
|
||||||
|
// propagate. The IDs will be broadcast to all peers within a single transaction Inv message.
|
||||||
|
// The broadcast itself may happen only during a subsequent call to this method
|
||||||
|
func (f *FlowContext) EnqueueTransactionIDsForPropagation(transactionIDs []*externalapi.DomainTransactionID) error {
|
||||||
|
f.transactionIDPropagationLock.Lock()
|
||||||
|
defer f.transactionIDPropagationLock.Unlock()
|
||||||
|
|
||||||
|
f.transactionIDsToPropagate = append(f.transactionIDsToPropagate, transactionIDs...)
|
||||||
|
|
||||||
|
return f.maybePropagateTransactions()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *FlowContext) maybePropagateTransactions() error {
|
||||||
|
if time.Since(f.lastTransactionIDPropagationTime) < TransactionIDPropagationInterval &&
|
||||||
|
len(f.transactionIDsToPropagate) < appmessage.MaxInvPerTxInvMsg {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for len(f.transactionIDsToPropagate) > 0 {
|
||||||
|
transactionIDsToBroadcast := f.transactionIDsToPropagate
|
||||||
|
if len(transactionIDsToBroadcast) > appmessage.MaxInvPerTxInvMsg {
|
||||||
|
transactionIDsToBroadcast = f.transactionIDsToPropagate[:len(transactionIDsToBroadcast)]
|
||||||
|
}
|
||||||
|
log.Debugf("Transaction propagation: broadcasting %d transactions", len(transactionIDsToBroadcast))
|
||||||
|
|
||||||
|
inv := appmessage.NewMsgInvTransaction(transactionIDsToBroadcast)
|
||||||
|
err := f.Broadcast(inv)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
f.transactionIDsToPropagate = f.transactionIDsToPropagate[len(transactionIDsToBroadcast):]
|
||||||
|
}
|
||||||
|
|
||||||
|
f.lastTransactionIDPropagationTime = time.Now()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
@ -1,29 +0,0 @@
|
|||||||
package blockrelay
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/common"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) sendGetBlockLocator(lowHash *externalapi.DomainHash,
|
|
||||||
highHash *externalapi.DomainHash, limit uint32) error {
|
|
||||||
|
|
||||||
msgGetBlockLocator := appmessage.NewMsgRequestBlockLocator(lowHash, highHash, limit)
|
|
||||||
return flow.outgoingRoute.Enqueue(msgGetBlockLocator)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) receiveBlockLocator() (blockLocatorHashes []*externalapi.DomainHash, err error) {
|
|
||||||
message, err := flow.dequeueIncomingMessageAndSkipInvs(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
msgBlockLocator, ok := message.(*appmessage.MsgBlockLocator)
|
|
||||||
if !ok {
|
|
||||||
return nil,
|
|
||||||
protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s, got: %s", appmessage.CmdBlockLocator, message.Command())
|
|
||||||
}
|
|
||||||
return msgBlockLocator.BlockLocatorHashes, nil
|
|
||||||
}
|
|
@ -1,51 +0,0 @@
|
|||||||
package blockrelay
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
|
||||||
"github.com/kaspanet/kaspad/domain"
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
|
||||||
)
|
|
||||||
|
|
||||||
// HandlePruningPointHashRequestsFlowContext is the interface for the context needed for the handlePruningPointHashRequestsFlow flow.
|
|
||||||
type HandlePruningPointHashRequestsFlowContext interface {
|
|
||||||
Domain() domain.Domain
|
|
||||||
}
|
|
||||||
|
|
||||||
type handlePruningPointHashRequestsFlow struct {
|
|
||||||
HandlePruningPointHashRequestsFlowContext
|
|
||||||
incomingRoute, outgoingRoute *router.Route
|
|
||||||
}
|
|
||||||
|
|
||||||
// HandlePruningPointHashRequests listens to appmessage.MsgRequestPruningPointHashMessage messages and sends
|
|
||||||
// the pruning point hash as response.
|
|
||||||
func HandlePruningPointHashRequests(context HandlePruningPointHashRequestsFlowContext, incomingRoute,
|
|
||||||
outgoingRoute *router.Route) error {
|
|
||||||
flow := &handlePruningPointHashRequestsFlow{
|
|
||||||
HandlePruningPointHashRequestsFlowContext: context,
|
|
||||||
incomingRoute: incomingRoute,
|
|
||||||
outgoingRoute: outgoingRoute,
|
|
||||||
}
|
|
||||||
|
|
||||||
return flow.start()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handlePruningPointHashRequestsFlow) start() error {
|
|
||||||
for {
|
|
||||||
_, err := flow.incomingRoute.Dequeue()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
log.Debugf("Got request for a pruning point hash")
|
|
||||||
|
|
||||||
pruningPoint, err := flow.Domain().Consensus().PruningPoint()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = flow.outgoingRoute.Enqueue(appmessage.NewPruningPointHashMessage(pruningPoint))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
log.Debugf("Sent pruning point hash %s", pruningPoint)
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,144 +0,0 @@
|
|||||||
package blockrelay
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/common"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
|
||||||
"github.com/kaspanet/kaspad/domain"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/ruleerrors"
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/logger"
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
|
||||||
)
|
|
||||||
|
|
||||||
// HandleRequestPruningPointUTXOSetAndBlockContext is the interface for the context needed for the HandleRequestPruningPointUTXOSetAndBlock flow.
|
|
||||||
type HandleRequestPruningPointUTXOSetAndBlockContext interface {
|
|
||||||
Domain() domain.Domain
|
|
||||||
}
|
|
||||||
|
|
||||||
type handleRequestPruningPointUTXOSetAndBlockFlow struct {
|
|
||||||
HandleRequestPruningPointUTXOSetAndBlockContext
|
|
||||||
incomingRoute, outgoingRoute *router.Route
|
|
||||||
}
|
|
||||||
|
|
||||||
// HandleRequestPruningPointUTXOSetAndBlock listens to appmessage.MsgRequestPruningPointUTXOSetAndBlock messages and sends
|
|
||||||
// the pruning point UTXO set and block body.
|
|
||||||
func HandleRequestPruningPointUTXOSetAndBlock(context HandleRequestPruningPointUTXOSetAndBlockContext, incomingRoute,
|
|
||||||
outgoingRoute *router.Route) error {
|
|
||||||
flow := &handleRequestPruningPointUTXOSetAndBlockFlow{
|
|
||||||
HandleRequestPruningPointUTXOSetAndBlockContext: context,
|
|
||||||
incomingRoute: incomingRoute,
|
|
||||||
outgoingRoute: outgoingRoute,
|
|
||||||
}
|
|
||||||
|
|
||||||
return flow.start()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRequestPruningPointUTXOSetAndBlockFlow) start() error {
|
|
||||||
for {
|
|
||||||
msgRequestPruningPointUTXOSetAndBlock, err := flow.waitForRequestPruningPointUTXOSetAndBlockMessages()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = flow.handleRequestPruningPointUTXOSetAndBlockMessage(msgRequestPruningPointUTXOSetAndBlock)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRequestPruningPointUTXOSetAndBlockFlow) handleRequestPruningPointUTXOSetAndBlockMessage(
|
|
||||||
msgRequestPruningPointUTXOSetAndBlock *appmessage.MsgRequestPruningPointUTXOSetAndBlock) error {
|
|
||||||
|
|
||||||
onEnd := logger.LogAndMeasureExecutionTime(log, "handleRequestPruningPointUTXOSetAndBlockFlow")
|
|
||||||
defer onEnd()
|
|
||||||
|
|
||||||
log.Debugf("Got request for PruningPointHash UTXOSet and Block")
|
|
||||||
|
|
||||||
err := flow.sendPruningPointBlock(msgRequestPruningPointUTXOSetAndBlock)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return flow.sendPruningPointUTXOSet(msgRequestPruningPointUTXOSetAndBlock)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRequestPruningPointUTXOSetAndBlockFlow) waitForRequestPruningPointUTXOSetAndBlockMessages() (
|
|
||||||
*appmessage.MsgRequestPruningPointUTXOSetAndBlock, error) {
|
|
||||||
|
|
||||||
message, err := flow.incomingRoute.Dequeue()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
msgRequestPruningPointUTXOSetAndBlock, ok := message.(*appmessage.MsgRequestPruningPointUTXOSetAndBlock)
|
|
||||||
if !ok {
|
|
||||||
return nil, protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s, got: %s", appmessage.CmdRequestPruningPointUTXOSetAndBlock, message.Command())
|
|
||||||
}
|
|
||||||
return msgRequestPruningPointUTXOSetAndBlock, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRequestPruningPointUTXOSetAndBlockFlow) sendPruningPointBlock(
|
|
||||||
msgRequestPruningPointUTXOSetAndBlock *appmessage.MsgRequestPruningPointUTXOSetAndBlock) error {
|
|
||||||
|
|
||||||
block, err := flow.Domain().Consensus().GetBlock(msgRequestPruningPointUTXOSetAndBlock.PruningPointHash)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
log.Debugf("Retrieved pruning block %s", msgRequestPruningPointUTXOSetAndBlock.PruningPointHash)
|
|
||||||
|
|
||||||
return flow.outgoingRoute.Enqueue(appmessage.NewMsgIBDBlock(appmessage.DomainBlockToMsgBlock(block)))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRequestPruningPointUTXOSetAndBlockFlow) sendPruningPointUTXOSet(
|
|
||||||
msgRequestPruningPointUTXOSetAndBlock *appmessage.MsgRequestPruningPointUTXOSetAndBlock) error {
|
|
||||||
|
|
||||||
// Send the UTXO set in `step`-sized chunks
|
|
||||||
const step = 1000
|
|
||||||
var fromOutpoint *externalapi.DomainOutpoint
|
|
||||||
chunksSent := 0
|
|
||||||
for {
|
|
||||||
pruningPointUTXOs, err := flow.Domain().Consensus().GetPruningPointUTXOs(
|
|
||||||
msgRequestPruningPointUTXOSetAndBlock.PruningPointHash, fromOutpoint, step)
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(err, ruleerrors.ErrWrongPruningPointHash) {
|
|
||||||
return flow.outgoingRoute.Enqueue(appmessage.NewMsgUnexpectedPruningPoint())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Debugf("Retrieved %d UTXOs for pruning block %s",
|
|
||||||
len(pruningPointUTXOs), msgRequestPruningPointUTXOSetAndBlock.PruningPointHash)
|
|
||||||
|
|
||||||
outpointAndUTXOEntryPairs :=
|
|
||||||
appmessage.DomainOutpointAndUTXOEntryPairsToOutpointAndUTXOEntryPairs(pruningPointUTXOs)
|
|
||||||
err = flow.outgoingRoute.Enqueue(appmessage.NewMsgPruningPointUTXOSetChunk(outpointAndUTXOEntryPairs))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(pruningPointUTXOs) < step {
|
|
||||||
log.Debugf("Finished sending UTXOs for pruning block %s",
|
|
||||||
msgRequestPruningPointUTXOSetAndBlock.PruningPointHash)
|
|
||||||
|
|
||||||
return flow.outgoingRoute.Enqueue(appmessage.NewMsgDonePruningPointUTXOSetChunks())
|
|
||||||
}
|
|
||||||
|
|
||||||
fromOutpoint = pruningPointUTXOs[len(pruningPointUTXOs)-1].Outpoint
|
|
||||||
chunksSent++
|
|
||||||
|
|
||||||
// Wait for the peer to request more chunks every `ibdBatchSize` chunks
|
|
||||||
if chunksSent%ibdBatchSize == 0 {
|
|
||||||
message, err := flow.incomingRoute.DequeueWithTimeout(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
_, ok := message.(*appmessage.MsgRequestNextPruningPointUTXOSetChunk)
|
|
||||||
if !ok {
|
|
||||||
return protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s, got: %s", appmessage.CmdRequestNextPruningPointUTXOSetChunk, message.Command())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,567 +0,0 @@
|
|||||||
package blockrelay
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/infrastructure/logger"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model"
|
|
||||||
|
|
||||||
"github.com/kaspanet/kaspad/app/appmessage"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/common"
|
|
||||||
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/ruleerrors"
|
|
||||||
"github.com/kaspanet/kaspad/domain/consensus/utils/consensushashing"
|
|
||||||
"github.com/pkg/errors"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) runIBDIfNotRunning(highHash *externalapi.DomainHash) error {
|
|
||||||
wasIBDNotRunning := flow.TrySetIBDRunning(flow.peer)
|
|
||||||
if !wasIBDNotRunning {
|
|
||||||
log.Debugf("IBD is already running")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
defer flow.UnsetIBDRunning()
|
|
||||||
|
|
||||||
log.Debugf("IBD started with peer %s and highHash %s", flow.peer, highHash)
|
|
||||||
|
|
||||||
log.Debugf("Syncing headers up to %s", highHash)
|
|
||||||
headersSynced, err := flow.syncHeaders(highHash)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if !headersSynced {
|
|
||||||
log.Debugf("Aborting IBD because the headers failed to sync")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
log.Debugf("Finished syncing headers up to %s", highHash)
|
|
||||||
|
|
||||||
log.Debugf("Syncing the current pruning point UTXO set")
|
|
||||||
syncedPruningPointUTXOSetSuccessfully, err := flow.syncPruningPointUTXOSet()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if !syncedPruningPointUTXOSetSuccessfully {
|
|
||||||
log.Debugf("Aborting IBD because the pruning point UTXO set failed to sync")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
log.Debugf("Finished syncing the current pruning point UTXO set")
|
|
||||||
|
|
||||||
log.Debugf("Downloading block bodies up to %s", highHash)
|
|
||||||
err = flow.syncMissingBlockBodies(highHash)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
log.Debugf("Finished downloading block bodies up to %s", highHash)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// syncHeaders attempts to sync headers from the peer. This method may fail
|
|
||||||
// because the peer and us have conflicting pruning points. In that case we
|
|
||||||
// return (false, nil) so that we may stop IBD gracefully.
|
|
||||||
func (flow *handleRelayInvsFlow) syncHeaders(highHash *externalapi.DomainHash) (bool, error) {
|
|
||||||
log.Debugf("Trying to find highest shared chain block with peer %s with high hash %s", flow.peer, highHash)
|
|
||||||
highestSharedBlockHash, highestSharedBlockFound, err := flow.findHighestSharedBlockHash(highHash)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
if !highestSharedBlockFound {
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
log.Debugf("Found highest shared chain block %s with peer %s", highestSharedBlockHash, flow.peer)
|
|
||||||
|
|
||||||
err = flow.downloadHeaders(highestSharedBlockHash, highHash)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the highHash has not been received, the peer is misbehaving
|
|
||||||
highHashBlockInfo, err := flow.Domain().Consensus().GetBlockInfo(highHash)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
if !highHashBlockInfo.Exists {
|
|
||||||
return false, protocolerrors.Errorf(true, "did not receive "+
|
|
||||||
"highHash header %s from peer %s during header download", highHash, flow.peer)
|
|
||||||
}
|
|
||||||
log.Debugf("Headers downloaded from peer %s", flow.peer)
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// findHighestSharedBlock attempts to find the highest shared block between the peer
|
|
||||||
// and this node. This method may fail because the peer and us have conflicting pruning
|
|
||||||
// points. In that case we return (nil, false, nil) so that we may stop IBD gracefully.
|
|
||||||
func (flow *handleRelayInvsFlow) findHighestSharedBlockHash(
|
|
||||||
targetHash *externalapi.DomainHash) (*externalapi.DomainHash, bool, error) {
|
|
||||||
|
|
||||||
log.Debugf("Sending a blockLocator to %s between pruning point and headers selected tip", flow.peer)
|
|
||||||
blockLocator, err := flow.Domain().Consensus().CreateFullHeadersSelectedChainBlockLocator()
|
|
||||||
if err != nil {
|
|
||||||
return nil, false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
for {
|
|
||||||
highestHash, highestHashFound, err := flow.fetchHighestHash(targetHash, blockLocator)
|
|
||||||
if err != nil {
|
|
||||||
return nil, false, err
|
|
||||||
}
|
|
||||||
if !highestHashFound {
|
|
||||||
return nil, false, nil
|
|
||||||
}
|
|
||||||
highestHashIndex, err := flow.findHighestHashIndex(highestHash, blockLocator)
|
|
||||||
if err != nil {
|
|
||||||
return nil, false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if highestHashIndex == 0 ||
|
|
||||||
// If the block locator contains only two adjacent chain blocks, the
|
|
||||||
// syncer will always find the same highest chain block, so to avoid
|
|
||||||
// an endless loop, we explicitly stop the loop in such situation.
|
|
||||||
(len(blockLocator) == 2 && highestHashIndex == 1) {
|
|
||||||
|
|
||||||
return highestHash, true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
locatorHashAboveHighestHash := highestHash
|
|
||||||
if highestHashIndex > 0 {
|
|
||||||
locatorHashAboveHighestHash = blockLocator[highestHashIndex-1]
|
|
||||||
}
|
|
||||||
|
|
||||||
blockLocator, err = flow.nextBlockLocator(highestHash, locatorHashAboveHighestHash)
|
|
||||||
if err != nil {
|
|
||||||
return nil, false, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) nextBlockLocator(lowHash, highHash *externalapi.DomainHash) (externalapi.BlockLocator, error) {
|
|
||||||
log.Debugf("Sending a blockLocator to %s between %s and %s", flow.peer, lowHash, highHash)
|
|
||||||
blockLocator, err := flow.Domain().Consensus().CreateHeadersSelectedChainBlockLocator(lowHash, highHash)
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(model.ErrBlockNotInSelectedParentChain, err) {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
log.Debugf("Headers selected parent chain moved since findHighestSharedBlockHash - " +
|
|
||||||
"restarting with full block locator")
|
|
||||||
blockLocator, err = flow.Domain().Consensus().CreateFullHeadersSelectedChainBlockLocator()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return blockLocator, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) findHighestHashIndex(
|
|
||||||
highestHash *externalapi.DomainHash, blockLocator externalapi.BlockLocator) (int, error) {
|
|
||||||
|
|
||||||
highestHashIndex := 0
|
|
||||||
highestHashIndexFound := false
|
|
||||||
for i, blockLocatorHash := range blockLocator {
|
|
||||||
if highestHash.Equal(blockLocatorHash) {
|
|
||||||
highestHashIndex = i
|
|
||||||
highestHashIndexFound = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !highestHashIndexFound {
|
|
||||||
return 0, protocolerrors.Errorf(true, "highest hash %s "+
|
|
||||||
"returned from peer %s is not in the original blockLocator", highestHash, flow.peer)
|
|
||||||
}
|
|
||||||
log.Debugf("The index of the highest hash in the original "+
|
|
||||||
"blockLocator sent to %s is %d", flow.peer, highestHashIndex)
|
|
||||||
|
|
||||||
return highestHashIndex, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// fetchHighestHash attempts to fetch the highest hash the peer knows amongst the given
|
|
||||||
// blockLocator. This method may fail because the peer and us have conflicting pruning
|
|
||||||
// points. In that case we return (nil, false, nil) so that we may stop IBD gracefully.
|
|
||||||
func (flow *handleRelayInvsFlow) fetchHighestHash(
|
|
||||||
targetHash *externalapi.DomainHash, blockLocator externalapi.BlockLocator) (*externalapi.DomainHash, bool, error) {
|
|
||||||
|
|
||||||
ibdBlockLocatorMessage := appmessage.NewMsgIBDBlockLocator(targetHash, blockLocator)
|
|
||||||
err := flow.outgoingRoute.Enqueue(ibdBlockLocatorMessage)
|
|
||||||
if err != nil {
|
|
||||||
return nil, false, err
|
|
||||||
}
|
|
||||||
message, err := flow.dequeueIncomingMessageAndSkipInvs(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return nil, false, err
|
|
||||||
}
|
|
||||||
switch message := message.(type) {
|
|
||||||
case *appmessage.MsgIBDBlockLocatorHighestHash:
|
|
||||||
highestHash := message.HighestHash
|
|
||||||
log.Debugf("The highest hash the peer %s knows is %s", flow.peer, highestHash)
|
|
||||||
|
|
||||||
return highestHash, true, nil
|
|
||||||
case *appmessage.MsgIBDBlockLocatorHighestHashNotFound:
|
|
||||||
log.Debugf("Peer %s does not know any block within our blockLocator. "+
|
|
||||||
"This should only happen if there's a DAG split deeper than the pruning point.", flow.peer)
|
|
||||||
return nil, false, nil
|
|
||||||
default:
|
|
||||||
return nil, false, protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s, got: %s", appmessage.CmdIBDBlockLocatorHighestHash, message.Command())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) downloadHeaders(highestSharedBlockHash *externalapi.DomainHash,
|
|
||||||
highHash *externalapi.DomainHash) error {
|
|
||||||
|
|
||||||
err := flow.sendRequestHeaders(highestSharedBlockHash, highHash)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Keep a short queue of blockHeadersMessages so that there's
|
|
||||||
// never a moment when the node is not validating and inserting
|
|
||||||
// headers
|
|
||||||
blockHeadersMessageChan := make(chan *appmessage.BlockHeadersMessage, 2)
|
|
||||||
errChan := make(chan error)
|
|
||||||
doneChan := make(chan interface{})
|
|
||||||
spawn("handleRelayInvsFlow-downloadHeaders", func() {
|
|
||||||
for {
|
|
||||||
blockHeadersMessage, doneIBD, err := flow.receiveHeaders()
|
|
||||||
if err != nil {
|
|
||||||
errChan <- err
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if doneIBD {
|
|
||||||
doneChan <- struct{}{}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
blockHeadersMessageChan <- blockHeadersMessage
|
|
||||||
|
|
||||||
err = flow.outgoingRoute.Enqueue(appmessage.NewMsgRequestNextHeaders())
|
|
||||||
if err != nil {
|
|
||||||
errChan <- err
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case blockHeadersMessage := <-blockHeadersMessageChan:
|
|
||||||
for _, header := range blockHeadersMessage.BlockHeaders {
|
|
||||||
err = flow.processHeader(header)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case err := <-errChan:
|
|
||||||
return err
|
|
||||||
case <-doneChan:
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) sendRequestHeaders(highestSharedBlockHash *externalapi.DomainHash,
|
|
||||||
peerSelectedTipHash *externalapi.DomainHash) error {
|
|
||||||
|
|
||||||
msgGetBlockInvs := appmessage.NewMsgRequstHeaders(highestSharedBlockHash, peerSelectedTipHash)
|
|
||||||
return flow.outgoingRoute.Enqueue(msgGetBlockInvs)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) receiveHeaders() (msgIBDBlock *appmessage.BlockHeadersMessage, doneIBD bool, err error) {
|
|
||||||
message, err := flow.dequeueIncomingMessageAndSkipInvs(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return nil, false, err
|
|
||||||
}
|
|
||||||
switch message := message.(type) {
|
|
||||||
case *appmessage.BlockHeadersMessage:
|
|
||||||
return message, false, nil
|
|
||||||
case *appmessage.MsgDoneHeaders:
|
|
||||||
return nil, true, nil
|
|
||||||
default:
|
|
||||||
return nil, false,
|
|
||||||
protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s or %s, got: %s", appmessage.CmdHeader, appmessage.CmdDoneHeaders, message.Command())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) processHeader(msgBlockHeader *appmessage.MsgBlockHeader) error {
|
|
||||||
header := appmessage.BlockHeaderToDomainBlockHeader(msgBlockHeader)
|
|
||||||
block := &externalapi.DomainBlock{
|
|
||||||
Header: header,
|
|
||||||
Transactions: nil,
|
|
||||||
}
|
|
||||||
|
|
||||||
blockHash := consensushashing.BlockHash(block)
|
|
||||||
blockInfo, err := flow.Domain().Consensus().GetBlockInfo(blockHash)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if blockInfo.Exists {
|
|
||||||
log.Debugf("Block header %s is already in the DAG. Skipping...", blockHash)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
_, err = flow.Domain().Consensus().ValidateAndInsertBlock(block)
|
|
||||||
if err != nil {
|
|
||||||
if !errors.As(err, &ruleerrors.RuleError{}) {
|
|
||||||
return errors.Wrapf(err, "failed to process header %s during IBD", blockHash)
|
|
||||||
}
|
|
||||||
|
|
||||||
if errors.Is(err, ruleerrors.ErrDuplicateBlock) {
|
|
||||||
log.Debugf("Skipping block header %s as it is a duplicate", blockHash)
|
|
||||||
} else {
|
|
||||||
log.Infof("Rejected block header %s from %s during IBD: %s", blockHash, flow.peer, err)
|
|
||||||
return protocolerrors.Wrapf(true, err, "got invalid block header %s during IBD", blockHash)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) syncPruningPointUTXOSet() (bool, error) {
|
|
||||||
log.Debugf("Checking if a new pruning point is available")
|
|
||||||
err := flow.outgoingRoute.Enqueue(appmessage.NewMsgRequestPruningPointHashMessage())
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
message, err := flow.dequeueIncomingMessageAndSkipInvs(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
msgPruningPointHash, ok := message.(*appmessage.MsgPruningPointHashMessage)
|
|
||||||
if !ok {
|
|
||||||
return false, protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s, got: %s", appmessage.CmdPruningPointHash, message.Command())
|
|
||||||
}
|
|
||||||
|
|
||||||
blockInfo, err := flow.Domain().Consensus().GetBlockInfo(msgPruningPointHash.Hash)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if !blockInfo.Exists {
|
|
||||||
return false, errors.Errorf("The pruning point header is missing")
|
|
||||||
}
|
|
||||||
|
|
||||||
if blockInfo.BlockStatus != externalapi.StatusHeaderOnly {
|
|
||||||
log.Debugf("Already has the block data of the new suggested pruning point %s", msgPruningPointHash.Hash)
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Infof("Checking if the suggested pruning point %s is compatible to the node DAG", msgPruningPointHash.Hash)
|
|
||||||
isValid, err := flow.Domain().Consensus().IsValidPruningPoint(msgPruningPointHash.Hash)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if !isValid {
|
|
||||||
log.Infof("The suggested pruning point %s is incompatible to this node DAG, so stopping IBD with this"+
|
|
||||||
" peer", msgPruningPointHash.Hash)
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Info("Fetching the pruning point UTXO set")
|
|
||||||
succeed, err := flow.fetchMissingUTXOSet(msgPruningPointHash.Hash)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if !succeed {
|
|
||||||
log.Infof("Couldn't successfully fetch the pruning point UTXO set. Stopping IBD.")
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Info("Fetched the new pruning point UTXO set")
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) fetchMissingUTXOSet(pruningPointHash *externalapi.DomainHash) (succeed bool, err error) {
|
|
||||||
defer func() {
|
|
||||||
err := flow.Domain().Consensus().ClearImportedPruningPointData()
|
|
||||||
if err != nil {
|
|
||||||
panic(fmt.Sprintf("failed to clear imported pruning point data: %s", err))
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
err = flow.outgoingRoute.Enqueue(appmessage.NewMsgRequestPruningPointUTXOSetAndBlock(pruningPointHash))
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
block, err := flow.receivePruningPointBlock()
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
receivedAll, err := flow.receiveAndInsertPruningPointUTXOSet(pruningPointHash)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
if !receivedAll {
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
err = flow.Domain().Consensus().ValidateAndInsertImportedPruningPoint(block)
|
|
||||||
if err != nil {
|
|
||||||
// TODO: Find a better way to deal with finality conflicts.
|
|
||||||
if errors.Is(err, ruleerrors.ErrSuggestedPruningViolatesFinality) {
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
return false, protocolerrors.ConvertToBanningProtocolErrorIfRuleError(err, "error with pruning point UTXO set")
|
|
||||||
}
|
|
||||||
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) receivePruningPointBlock() (*externalapi.DomainBlock, error) {
|
|
||||||
onEnd := logger.LogAndMeasureExecutionTime(log, "receivePruningPointBlock")
|
|
||||||
defer onEnd()
|
|
||||||
|
|
||||||
message, err := flow.dequeueIncomingMessageAndSkipInvs(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
ibdBlockMessage, ok := message.(*appmessage.MsgIBDBlock)
|
|
||||||
if !ok {
|
|
||||||
return nil, protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s, got: %s", appmessage.CmdIBDBlock, message.Command())
|
|
||||||
}
|
|
||||||
block := appmessage.MsgBlockToDomainBlock(ibdBlockMessage.MsgBlock)
|
|
||||||
|
|
||||||
log.Debugf("Received pruning point block %s", consensushashing.BlockHash(block))
|
|
||||||
|
|
||||||
return block, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) receiveAndInsertPruningPointUTXOSet(
|
|
||||||
pruningPointHash *externalapi.DomainHash) (bool, error) {
|
|
||||||
|
|
||||||
onEnd := logger.LogAndMeasureExecutionTime(log, "receiveAndInsertPruningPointUTXOSet")
|
|
||||||
defer onEnd()
|
|
||||||
|
|
||||||
receivedChunkCount := 0
|
|
||||||
receivedUTXOCount := 0
|
|
||||||
for {
|
|
||||||
message, err := flow.dequeueIncomingMessageAndSkipInvs(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
switch message := message.(type) {
|
|
||||||
case *appmessage.MsgPruningPointUTXOSetChunk:
|
|
||||||
receivedUTXOCount += len(message.OutpointAndUTXOEntryPairs)
|
|
||||||
domainOutpointAndUTXOEntryPairs :=
|
|
||||||
appmessage.OutpointAndUTXOEntryPairsToDomainOutpointAndUTXOEntryPairs(message.OutpointAndUTXOEntryPairs)
|
|
||||||
|
|
||||||
err := flow.Domain().Consensus().AppendImportedPruningPointUTXOs(domainOutpointAndUTXOEntryPairs)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
receivedChunkCount++
|
|
||||||
if receivedChunkCount%ibdBatchSize == 0 {
|
|
||||||
log.Debugf("Received %d UTXO set chunks so far, totaling in %d UTXOs",
|
|
||||||
receivedChunkCount, receivedUTXOCount)
|
|
||||||
|
|
||||||
requestNextPruningPointUTXOSetChunkMessage := appmessage.NewMsgRequestNextPruningPointUTXOSetChunk()
|
|
||||||
err := flow.outgoingRoute.Enqueue(requestNextPruningPointUTXOSetChunkMessage)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
case *appmessage.MsgDonePruningPointUTXOSetChunks:
|
|
||||||
log.Infof("Finished receiving the UTXO set. Total UTXOs: %d", receivedUTXOCount)
|
|
||||||
return true, nil
|
|
||||||
|
|
||||||
case *appmessage.MsgUnexpectedPruningPoint:
|
|
||||||
log.Infof("Could not receive the next UTXO chunk because the pruning point %s "+
|
|
||||||
"is no longer the pruning point of peer %s", pruningPointHash, flow.peer)
|
|
||||||
return false, nil
|
|
||||||
|
|
||||||
default:
|
|
||||||
return false, protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s or %s or %s, got: %s", appmessage.CmdPruningPointUTXOSetChunk,
|
|
||||||
appmessage.CmdDonePruningPointUTXOSetChunks, appmessage.CmdUnexpectedPruningPoint, message.Command(),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (flow *handleRelayInvsFlow) syncMissingBlockBodies(highHash *externalapi.DomainHash) error {
|
|
||||||
hashes, err := flow.Domain().Consensus().GetMissingBlockBodyHashes(highHash)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if len(hashes) == 0 {
|
|
||||||
// Blocks can be inserted inside the DAG during IBD if those were requested before IBD started.
|
|
||||||
// In rare cases, all the IBD blocks might be already inserted by the time we reach this point.
|
|
||||||
// In these cases - GetMissingBlockBodyHashes would return an empty array.
|
|
||||||
log.Debugf("No missing block body hashes found.")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
for offset := 0; offset < len(hashes); offset += ibdBatchSize {
|
|
||||||
var hashesToRequest []*externalapi.DomainHash
|
|
||||||
if offset+ibdBatchSize < len(hashes) {
|
|
||||||
hashesToRequest = hashes[offset : offset+ibdBatchSize]
|
|
||||||
} else {
|
|
||||||
hashesToRequest = hashes[offset:]
|
|
||||||
}
|
|
||||||
|
|
||||||
err := flow.outgoingRoute.Enqueue(appmessage.NewMsgRequestIBDBlocks(hashesToRequest))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, expectedHash := range hashesToRequest {
|
|
||||||
message, err := flow.dequeueIncomingMessageAndSkipInvs(common.DefaultTimeout)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
msgIBDBlock, ok := message.(*appmessage.MsgIBDBlock)
|
|
||||||
if !ok {
|
|
||||||
return protocolerrors.Errorf(true, "received unexpected message type. "+
|
|
||||||
"expected: %s, got: %s", appmessage.CmdIBDBlock, message.Command())
|
|
||||||
}
|
|
||||||
|
|
||||||
block := appmessage.MsgBlockToDomainBlock(msgIBDBlock.MsgBlock)
|
|
||||||
blockHash := consensushashing.BlockHash(block)
|
|
||||||
if !expectedHash.Equal(blockHash) {
|
|
||||||
return protocolerrors.Errorf(true, "expected block %s but got %s", expectedHash, blockHash)
|
|
||||||
}
|
|
||||||
|
|
||||||
blockInsertionResult, err := flow.Domain().Consensus().ValidateAndInsertBlock(block)
|
|
||||||
if err != nil {
|
|
||||||
if errors.Is(err, ruleerrors.ErrDuplicateBlock) {
|
|
||||||
log.Debugf("Skipping IBD Block %s as it has already been added to the DAG", blockHash)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
return protocolerrors.ConvertToBanningProtocolErrorIfRuleError(err, "invalid block %s", blockHash)
|
|
||||||
}
|
|
||||||
err = flow.OnNewBlock(block, blockInsertionResult)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// dequeueIncomingMessageAndSkipInvs is a convenience method to be used during
|
|
||||||
// IBD. Inv messages are expected to arrive at any given moment, but should be
|
|
||||||
// ignored while we're in IBD
|
|
||||||
func (flow *handleRelayInvsFlow) dequeueIncomingMessageAndSkipInvs(timeout time.Duration) (appmessage.Message, error) {
|
|
||||||
for {
|
|
||||||
message, err := flow.incomingRoute.DequeueWithTimeout(timeout)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if _, ok := message.(*appmessage.MsgInvRelayBlock); !ok {
|
|
||||||
return message, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -28,7 +28,7 @@ type HandleHandshakeContext interface {
|
|||||||
HandleError(err error, flowName string, isStopping *uint32, errChan chan<- error)
|
HandleError(err error, flowName string, isStopping *uint32, errChan chan<- error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// HandleHandshake sets up the handshake protocol - It sends a version message and waits for an incoming
|
// HandleHandshake sets up the new_handshake protocol - It sends a version message and waits for an incoming
|
||||||
// version message, as well as a verack for the sent version
|
// version message, as well as a verack for the sent version
|
||||||
func HandleHandshake(context HandleHandshakeContext, netConnection *netadapter.NetConnection,
|
func HandleHandshake(context HandleHandshakeContext, netConnection *netadapter.NetConnection,
|
||||||
receiveVersionRoute *routerpkg.Route, sendVersionRoute *routerpkg.Route, outgoingRoute *routerpkg.Route,
|
receiveVersionRoute *routerpkg.Route, sendVersionRoute *routerpkg.Route, outgoingRoute *routerpkg.Route,
|
||||||
@ -89,13 +89,16 @@ func HandleHandshake(context HandleHandshakeContext, netConnection *netadapter.N
|
|||||||
}
|
}
|
||||||
|
|
||||||
if peerAddress != nil {
|
if peerAddress != nil {
|
||||||
context.AddressManager().AddAddresses(peerAddress)
|
err := context.AddressManager().AddAddresses(peerAddress)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return peer, nil
|
return peer, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handshake is different from other flows, since in it should forward router.ErrRouteClosed to errChan
|
// Handshake is different from other flows, since in it should forward router.ErrRouteClosed to errChan
|
||||||
// Therefore we implement a separate handleError for handshake
|
// Therefore we implement a separate handleError for new_handshake
|
||||||
func handleError(err error, flowName string, isStopping *uint32, errChan chan error) {
|
func handleError(err error, flowName string, isStopping *uint32, errChan chan error) {
|
||||||
if errors.Is(err, routerpkg.ErrRouteClosed) {
|
if errors.Is(err, routerpkg.ErrRouteClosed) {
|
||||||
if atomic.AddUint32(isStopping, 1) == 1 {
|
if atomic.AddUint32(isStopping, 1) == 1 {
|
||||||
@ -104,7 +107,7 @@ func handleError(err error, flowName string, isStopping *uint32, errChan chan er
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if protocolErr := &(protocolerrors.ProtocolError{}); errors.As(err, &protocolErr) {
|
if protocolErr := (protocolerrors.ProtocolError{}); errors.As(err, &protocolErr) {
|
||||||
log.Errorf("Handshake protocol error from %s: %s", flowName, err)
|
log.Errorf("Handshake protocol error from %s: %s", flowName, err)
|
||||||
if atomic.AddUint32(isStopping, 1) == 1 {
|
if atomic.AddUint32(isStopping, 1) == 1 {
|
||||||
errChan <- err
|
errChan <- err
|
||||||
|
@ -5,5 +5,5 @@ import (
|
|||||||
"github.com/kaspanet/kaspad/util/panics"
|
"github.com/kaspanet/kaspad/util/panics"
|
||||||
)
|
)
|
||||||
|
|
||||||
var log, _ = logger.Get(logger.SubsystemTags.PROT)
|
var log = logger.RegisterSubSystem("PROT")
|
||||||
var spawn = panics.GoroutineWrapperFunc(log)
|
var spawn = panics.GoroutineWrapperFunc(log)
|
||||||
|
@ -7,6 +7,7 @@ import (
|
|||||||
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/logger"
|
"github.com/kaspanet/kaspad/infrastructure/logger"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
||||||
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -17,7 +18,9 @@ var (
|
|||||||
|
|
||||||
// minAcceptableProtocolVersion is the lowest protocol version that a
|
// minAcceptableProtocolVersion is the lowest protocol version that a
|
||||||
// connected peer may support.
|
// connected peer may support.
|
||||||
minAcceptableProtocolVersion = appmessage.ProtocolVersion
|
minAcceptableProtocolVersion = uint32(5)
|
||||||
|
|
||||||
|
maxAcceptableProtocolVersion = uint32(5)
|
||||||
)
|
)
|
||||||
|
|
||||||
type receiveVersionFlow struct {
|
type receiveVersionFlow struct {
|
||||||
@ -97,7 +100,12 @@ func (flow *receiveVersionFlow) start() (*appmessage.NetAddress, error) {
|
|||||||
return nil, protocolerrors.New(false, "incompatible subnetworks")
|
return nil, protocolerrors.New(false, "incompatible subnetworks")
|
||||||
}
|
}
|
||||||
|
|
||||||
flow.peer.UpdateFieldsFromMsgVersion(msgVersion)
|
if flow.Config().ProtocolVersion > maxAcceptableProtocolVersion {
|
||||||
|
return nil, errors.Errorf("%d is a non existing protocol version", flow.Config().ProtocolVersion)
|
||||||
|
}
|
||||||
|
|
||||||
|
maxProtocolVersion := flow.Config().ProtocolVersion
|
||||||
|
flow.peer.UpdateFieldsFromMsgVersion(msgVersion, maxProtocolVersion)
|
||||||
err = flow.outgoingRoute.Enqueue(appmessage.NewMsgVerAck())
|
err = flow.outgoingRoute.Enqueue(appmessage.NewMsgVerAck())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -7,6 +7,7 @@ import (
|
|||||||
"github.com/kaspanet/kaspad/infrastructure/logger"
|
"github.com/kaspanet/kaspad/infrastructure/logger"
|
||||||
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
||||||
"github.com/kaspanet/kaspad/version"
|
"github.com/kaspanet/kaspad/version"
|
||||||
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -56,15 +57,18 @@ func (flow *sendVersionFlow) start() error {
|
|||||||
// Version message.
|
// Version message.
|
||||||
localAddress := flow.AddressManager().BestLocalAddress(flow.peer.Connection().NetAddress())
|
localAddress := flow.AddressManager().BestLocalAddress(flow.peer.Connection().NetAddress())
|
||||||
subnetworkID := flow.Config().SubnetworkID
|
subnetworkID := flow.Config().SubnetworkID
|
||||||
|
if flow.Config().ProtocolVersion < minAcceptableProtocolVersion {
|
||||||
|
return errors.Errorf("configured protocol version %d is obsolete", flow.Config().ProtocolVersion)
|
||||||
|
}
|
||||||
msg := appmessage.NewMsgVersion(localAddress, flow.NetAdapter().ID(),
|
msg := appmessage.NewMsgVersion(localAddress, flow.NetAdapter().ID(),
|
||||||
flow.Config().ActiveNetParams.Name, subnetworkID)
|
flow.Config().ActiveNetParams.Name, subnetworkID, flow.Config().ProtocolVersion)
|
||||||
msg.AddUserAgent(userAgentName, userAgentVersion, flow.Config().UserAgentComments...)
|
msg.AddUserAgent(userAgentName, userAgentVersion, flow.Config().UserAgentComments...)
|
||||||
|
|
||||||
// Advertise the services flag
|
// Advertise the services flag
|
||||||
msg.Services = defaultServices
|
msg.Services = defaultServices
|
||||||
|
|
||||||
// Advertise our max supported protocol version.
|
// Advertise our max supported protocol version.
|
||||||
msg.ProtocolVersion = appmessage.ProtocolVersion
|
msg.ProtocolVersion = flow.Config().ProtocolVersion
|
||||||
|
|
||||||
// Advertise if inv messages for transactions are desired.
|
// Advertise if inv messages for transactions are desired.
|
||||||
msg.DisableRelayTx = flow.Config().BlocksOnly
|
msg.DisableRelayTx = flow.Config().BlocksOnly
|
||||||
|
9
app/protocol/flows/ready/log.go
Normal file
9
app/protocol/flows/ready/log.go
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
package ready
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/infrastructure/logger"
|
||||||
|
"github.com/kaspanet/kaspad/util/panics"
|
||||||
|
)
|
||||||
|
|
||||||
|
var log = logger.RegisterSubSystem("PROT")
|
||||||
|
var spawn = panics.GoroutineWrapperFunc(log)
|
56
app/protocol/flows/ready/ready.go
Normal file
56
app/protocol/flows/ready/ready.go
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
package ready
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/app/appmessage"
|
||||||
|
"github.com/kaspanet/kaspad/app/protocol/common"
|
||||||
|
"sync/atomic"
|
||||||
|
|
||||||
|
peerpkg "github.com/kaspanet/kaspad/app/protocol/peer"
|
||||||
|
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
||||||
|
routerpkg "github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
// HandleReady notify the other peer that peer is ready for messages, and wait for the other peer
|
||||||
|
// to send a ready message before start running the flows.
|
||||||
|
func HandleReady(incomingRoute *routerpkg.Route, outgoingRoute *routerpkg.Route,
|
||||||
|
peer *peerpkg.Peer,
|
||||||
|
) error {
|
||||||
|
|
||||||
|
log.Debugf("Sending ready message to %s", peer)
|
||||||
|
|
||||||
|
isStopping := uint32(0)
|
||||||
|
err := outgoingRoute.Enqueue(appmessage.NewMsgReady())
|
||||||
|
if err != nil {
|
||||||
|
return handleError(err, "HandleReady", &isStopping)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = incomingRoute.DequeueWithTimeout(common.DefaultTimeout)
|
||||||
|
if err != nil {
|
||||||
|
return handleError(err, "HandleReady", &isStopping)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debugf("Got ready message from %s", peer)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ready is different from other flows, since in it should forward router.ErrRouteClosed to errChan
|
||||||
|
// Therefore we implement a separate handleError for 'ready'
|
||||||
|
func handleError(err error, flowName string, isStopping *uint32) error {
|
||||||
|
if errors.Is(err, routerpkg.ErrRouteClosed) {
|
||||||
|
if atomic.AddUint32(isStopping, 1) == 1 {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if protocolErr := (protocolerrors.ProtocolError{}); errors.As(err, &protocolErr) {
|
||||||
|
log.Errorf("Ready protocol error from %s: %s", flowName, err)
|
||||||
|
if atomic.AddUint32(isStopping, 1) == 1 {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
panic(err)
|
||||||
|
}
|
File diff suppressed because it is too large
Load Diff
@ -35,6 +35,5 @@ func ReceiveAddresses(context ReceiveAddressesContext, incomingRoute *router.Rou
|
|||||||
return protocolerrors.Errorf(true, "address count exceeded %d", addressmanager.GetAddressesMax)
|
return protocolerrors.Errorf(true, "address count exceeded %d", addressmanager.GetAddressesMax)
|
||||||
}
|
}
|
||||||
|
|
||||||
context.AddressManager().AddAddresses(msgAddresses.AddressList...)
|
return context.AddressManager().AddAddresses(msgAddresses.AddressList...)
|
||||||
return nil
|
|
||||||
}
|
}
|
16
app/protocol/flows/v5/blockrelay/batch_size_test.go
Normal file
16
app/protocol/flows/v5/blockrelay/batch_size_test.go
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
package blockrelay
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/infrastructure/network/netadapter/router"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestIBDBatchSizeLessThanRouteCapacity(t *testing.T) {
|
||||||
|
// The `ibdBatchSize` constant must be equal at both syncer and syncee. Therefore, we do not want
|
||||||
|
// to set it to `router.DefaultMaxMessages` to avoid confusion and human errors.
|
||||||
|
// However, nonetheless we must enforce that it does not exceed `router.DefaultMaxMessages`
|
||||||
|
if ibdBatchSize >= router.DefaultMaxMessages {
|
||||||
|
t.Fatalf("IBD batch size (%d) must be smaller than router.DefaultMaxMessages (%d)",
|
||||||
|
ibdBatchSize, router.DefaultMaxMessages)
|
||||||
|
}
|
||||||
|
}
|
33
app/protocol/flows/v5/blockrelay/block_locator.go
Normal file
33
app/protocol/flows/v5/blockrelay/block_locator.go
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
package blockrelay
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kaspanet/kaspad/app/appmessage"
|
||||||
|
"github.com/kaspanet/kaspad/app/protocol/common"
|
||||||
|
"github.com/kaspanet/kaspad/app/protocol/protocolerrors"
|
||||||
|
"github.com/kaspanet/kaspad/domain/consensus/model/externalapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (flow *handleRelayInvsFlow) sendGetBlockLocator(highHash *externalapi.DomainHash, limit uint32) error {
|
||||||
|
msgGetBlockLocator := appmessage.NewMsgRequestBlockLocator(highHash, limit)
|
||||||
|
return flow.outgoingRoute.Enqueue(msgGetBlockLocator)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (flow *handleRelayInvsFlow) receiveBlockLocator() (blockLocatorHashes []*externalapi.DomainHash, err error) {
|
||||||
|
for {
|
||||||
|
message, err := flow.incomingRoute.DequeueWithTimeout(common.DefaultTimeout)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
switch message := message.(type) {
|
||||||
|
case *appmessage.MsgInvRelayBlock:
|
||||||
|
flow.invsQueue = append(flow.invsQueue, invRelayBlock{Hash: message.Hash, IsOrphanRoot: false})
|
||||||
|
case *appmessage.MsgBlockLocator:
|
||||||
|
return message.BlockLocatorHashes, nil
|
||||||
|
default:
|
||||||
|
return nil,
|
||||||
|
protocolerrors.Errorf(true, "received unexpected message type. "+
|
||||||
|
"expected: %s, got: %s", appmessage.CmdBlockLocator, message.Command())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user