mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-05 18:42:19 +00:00
222bb666e9
``` ➜ erigon-lib git:(fieldalign) ✗ fieldalignment -fix ./... /Users/estensen/Developer/erigon-lib/commitment/bin_patricia_hashed.go:81:16: struct of size 1065120 could be 1065112 /Users/estensen/Developer/erigon-lib/commitment/bin_patricia_hashed.go:1063:14: struct of size 1032 could be 1024 /Users/estensen/Developer/erigon-lib/commitment/hex_patricia_hashed.go:62:24: struct of size 952776 could be 952768 /Users/estensen/Developer/erigon-lib/commitment/hex_patricia_hashed.go:98:12: struct of size 1832 could be 1824 /Users/estensen/Developer/erigon-lib/commitment/hex_patricia_hashed.go:113:12: struct with 208 pointer bytes could be 152 /Users/estensen/Developer/erigon-lib/commitment/hex_patricia_hashed.go:143:11: struct of size 464 could be 456 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:24:11: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:79:11: struct of size 56 could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:134:11: struct with 56 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:189:11: struct with 56 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:244:12: struct with 56 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:299:12: struct with 56 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:355:19: struct of size 56 could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:418:23: struct with 168 pointer bytes could be 128 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:571:20: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:626:20: struct with 136 pointer bytes could be 128 /Users/estensen/Developer/erigon-lib/gointerfaces/types/types.pb.go:721:15: struct of size 168 could be 160 /Users/estensen/Developer/erigon-lib/etl/buffers.go:75:21: struct with 88 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/etl/buffers.go:182:27: struct with 56 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/etl/buffers.go:274:32: struct with 56 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/etl/collector.go:41:16: struct with 72 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/etl/etl.go:66:20: struct with 96 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/etl/heap.go:25:15: struct with 40 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/patricia/patricia.go:29:11: struct with 40 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/patricia/patricia.go:347:12: struct with 32 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/patricia/patricia.go:367:18: struct with 48 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/patricia/patricia.go:377:19: struct with 168 pointer bytes could be 144 /Users/estensen/Developer/erigon-lib/compress/compress.go:52:17: struct with 176 pointer bytes could be 136 /Users/estensen/Developer/erigon-lib/compress/compress.go:241:24: struct with 48 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/compress/compress.go:327:14: struct with 40 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/compress/compress.go:353:18: struct with 48 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/compress/compress.go:450:19: struct with 48 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/compress/compress.go:670:21: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/compress/compress.go:734:23: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/compress/decompress.go:31:15: struct with 32 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/compress/decompress.go:39:19: struct with 40 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/compress/decompress.go:117:15: struct with 64 pointer bytes could be 56 /Users/estensen/Developer/erigon-lib/compress/decompress.go:125:19: struct with 96 pointer bytes could be 80 /Users/estensen/Developer/erigon-lib/compress/decompress.go:386:13: struct with 64 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/compress/parallel_compress.go:208:22: struct with 16 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/recsplit/golomb_rice.go:32:17: struct with 16 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/recsplit/index.go:35:12: struct of size 432 could be 424 /Users/estensen/Developer/erigon-lib/recsplit/index_reader.go:26:18: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/recsplit/recsplit.go:64:15: struct of size 704 could be 680 /Users/estensen/Developer/erigon-lib/recsplit/recsplit.go:111:19: struct of size 104 could be 96 /Users/estensen/Developer/erigon-lib/aggregator/aggregator.go:171:17: struct of size 640 could be 632 /Users/estensen/Developer/erigon-lib/aggregator/aggregator.go:198:17: struct with 168 pointer bytes could be 144 /Users/estensen/Developer/erigon-lib/aggregator/aggregator.go:389:14: struct with 584 pointer bytes could be 568 /Users/estensen/Developer/erigon-lib/aggregator/aggregator.go:921:21: struct with 72 pointer bytes could be 56 /Users/estensen/Developer/erigon-lib/aggregator/aggregator.go:1195:22: struct with 2432 pointer bytes could be 2416 /Users/estensen/Developer/erigon-lib/aggregator/aggregator.go:2123:13: struct with 2448 pointer bytes could be 2416 /Users/estensen/Developer/erigon-lib/aggregator/aggregator.go:2634:17: struct with 96 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/aggregator/history.go:39:14: struct with 96 pointer bytes could be 88 /Users/estensen/Developer/erigon-lib/bptree/binary_file.go:33:17: struct with 40 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/bptree/node.go:79:13: struct of size 88 could be 80 /Users/estensen/Developer/erigon-lib/chain/chain_config.go:28:13: struct with 136 pointer bytes could be 120 /Users/estensen/Developer/erigon-lib/common/background/progress.go:26:18: struct with 40 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/gointerfaces/downloader/downloader.pb.go:25:19: struct with 64 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/downloader/downloader.pb.go:80:22: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/downloader/downloader.pb.go:127:20: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/downloader/downloader.pb.go:165:19: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/downloader/downloader.pb.go:203:17: struct of size 104 could be 96 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:135:23: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:173:21: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:220:24: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:258:22: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:305:26: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:343:24: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:390:30: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:437:26: struct of size 72 could be 64 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:500:30: struct with 64 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:563:28: struct with 64 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:626:37: struct with 56 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:681:35: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:736:29: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:774:27: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:821:27: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:859:25: struct with 48 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:906:23: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:953:21: struct of size 72 could be 64 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1008:24: struct of size 104 could be 88 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1079:25: struct of size 144 could be 136 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1190:19: struct with 56 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1245:17: struct with 72 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1300:23: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1347:21: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1394:23: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1441:21: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1488:17: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/ethbackend.pb.go:1535:24: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:220:13: struct of size 120 could be 112 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:299:11: struct of size 104 could be 96 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:370:20: struct with 56 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:425:20: struct of size 136 could be 128 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:513:23: struct with 56 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:585:18: struct of size 112 could be 104 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:664:25: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:719:23: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/remote/kv.pb.go:757:21: struct with 72 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:300:26: struct of size 72 could be 64 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:355:35: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:410:29: struct with 56 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:465:38: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:520:16: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:567:26: struct of size 56 could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:622:26: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:677:21: struct of size 80 could be 72 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:740:12: struct with 56 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:795:17: struct with 72 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:874:21: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:912:21: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:959:22: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:1006:17: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:1053:23: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:1091:21: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:1138:22: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:1185:20: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:1232:24: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/sentry/sentry.pb.go:1270:16: struct of size 56 could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:25:28: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:63:26: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:110:26: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:148:24: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:195:27: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:233:25: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:280:21: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:318:19: struct with 96 pointer bytes could be 80 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:389:24: struct with 96 pointer bytes could be 88 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:452:22: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:499:28: struct with 56 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:554:26: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:601:22: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:639:20: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:686:20: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/mining.pb.go:724:18: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:132:15: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:179:17: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:226:15: struct with 72 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:281:26: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:328:24: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:375:19: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:413:17: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:460:17: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:498:15: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:545:19: struct with 48 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:592:20: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:630:18: struct of size 56 could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:693:19: struct with 48 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:740:17: struct of size 56 could be 48 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:795:18: struct of size 80 could be 72 /Users/estensen/Developer/erigon-lib/gointerfaces/txpool/txpool.pb.go:858:22: struct of size 80 could be 72 /Users/estensen/Developer/erigon-lib/direct/sentry_client.go:171:25: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/kv/mdbx/kv_mdbx.go:50:15: struct with 104 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/kv/mdbx/kv_mdbx.go:355:13: struct with 160 pointer bytes could be 152 /Users/estensen/Developer/erigon-lib/kv/memdb/memory_mutation_cursor.go:39:27: struct with 200 pointer bytes could be 184 /Users/estensen/Developer/erigon-lib/kv/remotedb/kv_remote.go:22:17: struct with 72 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/kv/remotedb/kv_remote.go:38:15: struct with 80 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/kv/remotedbserver/server.go:314:24: struct with 40 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/state/aggregator.go:44:17: struct with 192 pointer bytes could be 128 /Users/estensen/Developer/erigon-lib/state/aggregator.go:422:13: struct of size 384 could be 360 /Users/estensen/Developer/erigon-lib/state/aggregator.go:455:26: struct with 424 pointer bytes could be 368 /Users/estensen/Developer/erigon-lib/state/aggregator.go:853:22: struct with 24 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/state/aggregator22.go:37:19: struct with 176 pointer bytes could be 136 /Users/estensen/Developer/erigon-lib/state/aggregator22.go:643:15: struct of size 240 could be 216 /Users/estensen/Developer/erigon-lib/state/aggregator22.go:674:28: struct with 272 pointer bytes could be 224 /Users/estensen/Developer/erigon-lib/state/aggregator22.go:1126:26: struct with 104 pointer bytes could be 88 /Users/estensen/Developer/erigon-lib/state/aggregator22.go:1156:23: struct with 24 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/state/domain.go:52:16: struct with 32 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/state/domain.go:89:13: struct with 120 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/state/domain.go:399:17: struct with 96 pointer bytes could be 64 /Users/estensen/Developer/erigon-lib/state/domain.go:443:14: struct with 32 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/state/domain.go:594:16: struct with 72 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/state/history.go:48:14: struct with 72 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/state/history.go:435:20: struct with 48 pointer bytes could be 32 /Users/estensen/Developer/erigon-lib/state/history.go:529:23: struct with 40 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/state/history.go:1150:23: struct of size 344 could be 336 /Users/estensen/Developer/erigon-lib/state/inverted_index.go:46:20: struct with 128 pointer bytes could be 88 /Users/estensen/Developer/erigon-lib/state/inverted_index.go:389:23: struct with 136 pointer bytes could be 88 /Users/estensen/Developer/erigon-lib/state/inverted_index.go:541:24: struct with 184 pointer bytes could be 152 /Users/estensen/Developer/erigon-lib/state/merge.go:69:19: struct of size 72 could be 56 /Users/estensen/Developer/erigon-lib/state/merge.go:143:20: struct of size 48 could be 40 /Users/estensen/Developer/erigon-lib/state/read_indices.go:29:18: struct with 64 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/state/read_indices.go:211:14: struct of size 72 could be 56 /Users/estensen/Developer/erigon-lib/state/read_indices.go:233:27: struct with 72 pointer bytes could be 56 /Users/estensen/Developer/erigon-lib/state/state_recon.go:61:16: struct with 56 pointer bytes could be 16 /Users/estensen/Developer/erigon-lib/state/state_recon.go:112:19: struct with 136 pointer bytes could be 112 /Users/estensen/Developer/erigon-lib/state/state_recon.go:190:22: struct with 128 pointer bytes could be 112 /Users/estensen/Developer/erigon-lib/types/testdata.go:81:26: struct with 40 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/types/txn.go:46:21: struct with 400 pointer bytes could be 40 /Users/estensen/Developer/erigon-lib/types/txn.go:82:13: struct with 200 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/types/txn.go:691:18: struct with 32 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/txpool/fetch.go:43:12: struct with 136 pointer bytes could be 112 /Users/estensen/Developer/erigon-lib/txpool/pool.go:69:13: struct with 104 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/txpool/pool.go:221:13: struct of size 104 could be 96 /Users/estensen/Developer/erigon-lib/txpool/pool.go:291:13: struct with 336 pointer bytes could be 288 /Users/estensen/Developer/erigon-lib/txpool/pool.go:1789:29: struct with 32 pointer bytes could be 8 /Users/estensen/Developer/erigon-lib/txpool/pool.go:1819:19: struct with 32 pointer bytes could be 24 /Users/estensen/Developer/erigon-lib/txpool/pool.go:1994:18: struct of size 64 could be 56 /Users/estensen/Developer/erigon-lib/txpool/pool.go:2102:14: struct of size 64 could be 56 /Users/estensen/Developer/erigon-lib/txpool/send.go:37:11: struct with 64 pointer bytes could be 48 /Users/estensen/Developer/erigon-lib/txpool/test_util.go:31:17: struct with 72 pointer bytes could be 40 ```
1646 lines
40 KiB
Go
1646 lines
40 KiB
Go
/*
|
|
Copyright 2021 Erigon contributors
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package mdbx
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"os"
|
|
"runtime"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/c2h5oh/datasize"
|
|
stack2 "github.com/go-stack/stack"
|
|
"github.com/ledgerwatch/erigon-lib/common/cmp"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"github.com/torquem-ch/mdbx-go/mdbx"
|
|
"go.uber.org/atomic"
|
|
"golang.org/x/sync/semaphore"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
)
|
|
|
|
const NonExistingDBI kv.DBI = 999_999_999
|
|
|
|
type TableCfgFunc func(defaultBuckets kv.TableCfg) kv.TableCfg
|
|
|
|
func WithChaindataTables(defaultBuckets kv.TableCfg) kv.TableCfg {
|
|
return defaultBuckets
|
|
}
|
|
|
|
type MdbxOpts struct {
|
|
// must be in the range from 12.5% (almost empty) to 50% (half empty)
|
|
// which corresponds to the range from 8192 and to 32768 in units respectively
|
|
log log.Logger
|
|
roTxsLimiter *semaphore.Weighted
|
|
bucketsCfg TableCfgFunc
|
|
path string
|
|
syncPeriod time.Duration
|
|
mapSize datasize.ByteSize
|
|
growthStep datasize.ByteSize
|
|
flags uint
|
|
pageSize uint64
|
|
mergeThreshold uint64
|
|
verbosity kv.DBVerbosityLvl
|
|
label kv.Label // marker to distinct db instances - one process may open many databases. for example to collect metrics of only 1 database
|
|
inMem bool
|
|
}
|
|
|
|
func NewMDBX(log log.Logger) MdbxOpts {
|
|
return MdbxOpts{
|
|
bucketsCfg: WithChaindataTables,
|
|
flags: mdbx.NoReadahead | mdbx.Coalesce | mdbx.Durable,
|
|
log: log,
|
|
pageSize: kv.DefaultPageSize(),
|
|
growthStep: 2 * datasize.GB,
|
|
mergeThreshold: 32768,
|
|
}
|
|
}
|
|
|
|
func (opts MdbxOpts) GetLabel() kv.Label { return opts.label }
|
|
func (opts MdbxOpts) GetInMem() bool { return opts.inMem }
|
|
func (opts MdbxOpts) GetPageSize() uint64 { return opts.pageSize }
|
|
|
|
func (opts MdbxOpts) Label(label kv.Label) MdbxOpts {
|
|
opts.label = label
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) RoTxsLimiter(l *semaphore.Weighted) MdbxOpts {
|
|
opts.roTxsLimiter = l
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) PageSize(v uint64) MdbxOpts {
|
|
opts.pageSize = v
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) GrowthStep(v datasize.ByteSize) MdbxOpts {
|
|
opts.growthStep = v
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) Path(path string) MdbxOpts {
|
|
opts.path = path
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) Set(opt MdbxOpts) MdbxOpts {
|
|
return opt
|
|
}
|
|
|
|
func (opts MdbxOpts) InMem(tmpDir string) MdbxOpts {
|
|
if tmpDir != "" {
|
|
if err := os.MkdirAll(tmpDir, 0755); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
path, err := os.MkdirTemp(tmpDir, "erigon-memdb-")
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
opts.path = path
|
|
opts.inMem = true
|
|
opts.flags = mdbx.UtterlyNoSync | mdbx.NoMetaSync | mdbx.LifoReclaim | mdbx.WriteMap
|
|
opts.mapSize = 512 * datasize.MB
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) Exclusive() MdbxOpts {
|
|
opts.flags = opts.flags | mdbx.Exclusive
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) Flags(f func(uint) uint) MdbxOpts {
|
|
opts.flags = f(opts.flags)
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) Readonly() MdbxOpts {
|
|
opts.flags = opts.flags | mdbx.Readonly
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) SyncPeriod(period time.Duration) MdbxOpts {
|
|
opts.syncPeriod = period
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) DBVerbosity(v kv.DBVerbosityLvl) MdbxOpts {
|
|
opts.verbosity = v
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) MapSize(sz datasize.ByteSize) MdbxOpts {
|
|
opts.mapSize = sz
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) WriteMap() MdbxOpts {
|
|
opts.flags |= mdbx.WriteMap
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) WriteMergeThreshold(v uint64) MdbxOpts {
|
|
opts.mergeThreshold = v
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) WithTableCfg(f TableCfgFunc) MdbxOpts {
|
|
opts.bucketsCfg = f
|
|
return opts
|
|
}
|
|
|
|
func (opts MdbxOpts) Open() (kv.RwDB, error) {
|
|
env, err := mdbx.NewEnv()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if opts.verbosity != -1 {
|
|
err = env.SetDebug(mdbx.LogLvl(opts.verbosity), mdbx.DbgDoNotChange, mdbx.LoggerDoNotChange) // temporary disable error, because it works if call it 1 time, but returns error if call it twice in same process (what often happening in tests)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("db verbosity set: %w", err)
|
|
}
|
|
}
|
|
if err = env.SetOption(mdbx.OptMaxDB, 100); err != nil {
|
|
return nil, err
|
|
}
|
|
if err = env.SetOption(mdbx.OptMaxReaders, kv.ReadersLimit); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if opts.mapSize == 0 {
|
|
if !opts.inMem {
|
|
opts.mapSize = 3 * datasize.TB
|
|
}
|
|
}
|
|
if opts.flags&mdbx.Accede == 0 {
|
|
if opts.inMem {
|
|
if err = env.SetGeometry(-1, -1, int(opts.mapSize), int(2*datasize.MB), 0, 4*1024); err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
if err = env.SetGeometry(-1, -1, int(opts.mapSize), int(opts.growthStep), -1, int(opts.pageSize)); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
const MAX_AUGMENT_LIMIT = 0x7fffFFFF
|
|
if err = env.SetOption(mdbx.OptRpAugmentLimit, MAX_AUGMENT_LIMIT); err != nil {
|
|
return nil, err
|
|
}
|
|
if err = os.MkdirAll(opts.path, 0744); err != nil {
|
|
return nil, fmt.Errorf("could not create dir: %s, %w", opts.path, err)
|
|
}
|
|
}
|
|
|
|
err = env.Open(opts.path, opts.flags, 0664)
|
|
if err != nil {
|
|
if err != nil {
|
|
return nil, fmt.Errorf("%w, label: %s, trace: %s", err, opts.label.String(), stack2.Trace().String())
|
|
}
|
|
}
|
|
|
|
// mdbx will not change pageSize if db already exists. means need read real value after env.open()
|
|
in, err := env.Info(nil)
|
|
if err != nil {
|
|
if err != nil {
|
|
return nil, fmt.Errorf("%w, label: %s, trace: %s", err, opts.label.String(), stack2.Trace().String())
|
|
}
|
|
}
|
|
opts.pageSize = uint64(in.PageSize)
|
|
|
|
// erigon using big transactions
|
|
// increase "page measured" options. need do it after env.Open() because default are depend on pageSize known only after env.Open()
|
|
if opts.flags&mdbx.Accede == 0 && opts.flags&mdbx.Readonly == 0 {
|
|
// 1/8 is good for transactions with a lot of modifications - to reduce invalidation size.
|
|
// But Erigon app now using Batch and etl.Collectors to avoid writing to DB frequently changing data.
|
|
// It means most of our writes are: APPEND or "single UPSERT per key during transaction"
|
|
//if err = env.SetOption(mdbx.OptSpillMinDenominator, 8); err != nil {
|
|
// return nil, err
|
|
//}
|
|
|
|
txnDpInitial, err := env.GetOption(mdbx.OptTxnDpInitial)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err = env.SetOption(mdbx.OptTxnDpInitial, txnDpInitial*2); err != nil {
|
|
return nil, err
|
|
}
|
|
dpReserveLimit, err := env.GetOption(mdbx.OptDpReverseLimit)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err = env.SetOption(mdbx.OptDpReverseLimit, dpReserveLimit*2); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
defaultDirtyPagesLimit, err := env.GetOption(mdbx.OptTxnDpLimit)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err = env.SetOption(mdbx.OptTxnDpLimit, defaultDirtyPagesLimit*2); err != nil { // default is RAM/42
|
|
return nil, err
|
|
}
|
|
// must be in the range from 12.5% (almost empty) to 50% (half empty)
|
|
// which corresponds to the range from 8192 and to 32768 in units respectively
|
|
if err = env.SetOption(mdbx.OptMergeThreshold16dot16Percent, opts.mergeThreshold); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
dirtyPagesLimit, err := env.GetOption(mdbx.OptTxnDpLimit)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if opts.syncPeriod != 0 {
|
|
if err = env.SetSyncPeriod(opts.syncPeriod); err != nil {
|
|
env.Close()
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
if opts.roTxsLimiter == nil {
|
|
targetSemCount := int64(cmp.Max(32, runtime.GOMAXPROCS(-1)*8))
|
|
opts.roTxsLimiter = semaphore.NewWeighted(targetSemCount) // 1 less than max to allow unlocking to happen
|
|
}
|
|
db := &MdbxKV{
|
|
opts: opts,
|
|
env: env,
|
|
log: opts.log,
|
|
wg: &sync.WaitGroup{},
|
|
buckets: kv.TableCfg{},
|
|
txSize: dirtyPagesLimit * opts.pageSize,
|
|
roTxsLimiter: opts.roTxsLimiter,
|
|
}
|
|
|
|
customBuckets := opts.bucketsCfg(kv.ChaindataTablesCfg)
|
|
for name, cfg := range customBuckets { // copy map to avoid changing global variable
|
|
db.buckets[name] = cfg
|
|
}
|
|
|
|
buckets := bucketSlice(db.buckets)
|
|
if err := db.openDBIs(buckets); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Configure buckets and open deprecated buckets
|
|
if err := env.View(func(tx *mdbx.Txn) error {
|
|
for _, name := range buckets {
|
|
// Open deprecated buckets if they exist, don't create
|
|
if !db.buckets[name].IsDeprecated {
|
|
continue
|
|
}
|
|
cnfCopy := db.buckets[name]
|
|
dbi, createErr := tx.OpenDBI(name, mdbx.DBAccede, nil, nil)
|
|
if createErr != nil {
|
|
if mdbx.IsNotFound(createErr) {
|
|
cnfCopy.DBI = NonExistingDBI
|
|
db.buckets[name] = cnfCopy
|
|
continue // if deprecated bucket couldn't be open - then it's deleted and it's fine
|
|
} else {
|
|
return fmt.Errorf("bucket: %s, %w", name, createErr)
|
|
}
|
|
}
|
|
cnfCopy.DBI = kv.DBI(dbi)
|
|
db.buckets[name] = cnfCopy
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !opts.inMem {
|
|
if staleReaders, err := db.env.ReaderCheck(); err != nil {
|
|
db.log.Error("failed ReaderCheck", "err", err)
|
|
} else if staleReaders > 0 {
|
|
db.log.Info("[db] cleared reader slots from dead processes", "amount", staleReaders)
|
|
}
|
|
|
|
}
|
|
return db, nil
|
|
}
|
|
|
|
func (opts MdbxOpts) MustOpen() kv.RwDB {
|
|
db, err := opts.Open()
|
|
if err != nil {
|
|
panic(fmt.Errorf("fail to open mdbx: %w", err))
|
|
}
|
|
return db
|
|
}
|
|
|
|
type MdbxKV struct {
|
|
log log.Logger
|
|
env *mdbx.Env
|
|
wg *sync.WaitGroup
|
|
buckets kv.TableCfg
|
|
roTxsLimiter *semaphore.Weighted // does limit amount of concurrent Ro transactions - in most casess runtime.NumCPU() is good value for this channel capacity - this channel can be shared with other components (like Decompressor)
|
|
opts MdbxOpts
|
|
txSize uint64
|
|
closed atomic.Bool
|
|
}
|
|
|
|
func (db *MdbxKV) PageSize() uint64 { return db.opts.pageSize }
|
|
|
|
// openDBIs - first trying to open existing DBI's in RO transaction
|
|
// otherwise re-try by RW transaction
|
|
// it allow open DB from another process - even if main process holding long RW transaction
|
|
func (db *MdbxKV) openDBIs(buckets []string) error {
|
|
if db.opts.flags&mdbx.Readonly != 0 {
|
|
if err := db.View(context.Background(), func(tx kv.Tx) error {
|
|
for _, name := range buckets {
|
|
if db.buckets[name].IsDeprecated {
|
|
continue
|
|
}
|
|
if err := tx.(kv.BucketMigrator).CreateBucket(name); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return tx.Commit() // when open db as read-only, commit of this RO transaction is required
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if err := db.Update(context.Background(), func(tx kv.RwTx) error {
|
|
for _, name := range buckets {
|
|
if db.buckets[name].IsDeprecated {
|
|
continue
|
|
}
|
|
if err := tx.(kv.BucketMigrator).CreateBucket(name); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Close closes db
|
|
// All transactions must be closed before closing the database.
|
|
func (db *MdbxKV) Close() {
|
|
if db.closed.Load() {
|
|
return
|
|
}
|
|
db.closed.Store(true)
|
|
db.wg.Wait()
|
|
db.env.Close()
|
|
db.env = nil
|
|
|
|
if db.opts.inMem {
|
|
if err := os.RemoveAll(db.opts.path); err != nil {
|
|
db.log.Warn("failed to remove in-mem db file", "err", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (db *MdbxKV) BeginRo(ctx context.Context) (txn kv.Tx, err error) {
|
|
if db.closed.Load() {
|
|
return nil, fmt.Errorf("db closed")
|
|
}
|
|
|
|
// don't try to acquire if the context is already done
|
|
done := ctx.Done()
|
|
select {
|
|
case <-done:
|
|
return nil, ctx.Err()
|
|
default:
|
|
// otherwise carry on
|
|
}
|
|
|
|
// will return nil err if context is cancelled (may appear to acquire the semaphore)
|
|
if semErr := db.roTxsLimiter.Acquire(ctx, 1); semErr != nil {
|
|
return nil, semErr
|
|
}
|
|
|
|
defer func() {
|
|
if err == nil {
|
|
db.wg.Add(1)
|
|
}
|
|
if txn == nil {
|
|
// on error, or if there is whatever reason that we don't return a tx,
|
|
// we need to free up the limiter slot, otherwise it could lead to deadlocks
|
|
db.roTxsLimiter.Release(1)
|
|
}
|
|
}()
|
|
|
|
tx, err := db.env.BeginTxn(nil, mdbx.Readonly)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("%w, label: %s, trace: %s", err, db.opts.label.String(), stack2.Trace().String())
|
|
}
|
|
tx.RawRead = true
|
|
return &MdbxTx{
|
|
db: db,
|
|
tx: tx,
|
|
readOnly: true,
|
|
}, nil
|
|
}
|
|
|
|
func (db *MdbxKV) BeginRw(_ context.Context) (txn kv.RwTx, err error) {
|
|
if db.closed.Load() {
|
|
return nil, fmt.Errorf("db closed")
|
|
}
|
|
runtime.LockOSThread()
|
|
defer func() {
|
|
if err == nil {
|
|
db.wg.Add(1)
|
|
}
|
|
}()
|
|
|
|
tx, err := db.env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
runtime.UnlockOSThread() // unlock only in case of error. normal flow is "defer .Rollback()"
|
|
return nil, fmt.Errorf("%w, lable: %s, trace: %s", err, db.opts.label.String(), stack2.Trace().String())
|
|
}
|
|
tx.RawRead = true
|
|
return &MdbxTx{
|
|
db: db,
|
|
tx: tx,
|
|
}, nil
|
|
}
|
|
|
|
type MdbxTx struct {
|
|
tx *mdbx.Txn
|
|
db *MdbxKV
|
|
cursors map[uint64]*mdbx.Cursor
|
|
statelessCursors map[string]kv.Cursor
|
|
readOnly bool
|
|
cursorID uint64
|
|
}
|
|
|
|
type MdbxCursor struct {
|
|
tx *MdbxTx
|
|
c *mdbx.Cursor
|
|
bucketName string
|
|
bucketCfg kv.TableCfgItem
|
|
dbi mdbx.DBI
|
|
id uint64
|
|
}
|
|
|
|
func (db *MdbxKV) Env() *mdbx.Env {
|
|
return db.env
|
|
}
|
|
|
|
func (db *MdbxKV) AllDBI() map[string]kv.DBI {
|
|
res := map[string]kv.DBI{}
|
|
for name, cfg := range db.buckets {
|
|
res[name] = cfg.DBI
|
|
}
|
|
return res
|
|
}
|
|
|
|
func (db *MdbxKV) AllBuckets() kv.TableCfg {
|
|
return db.buckets
|
|
}
|
|
|
|
func (tx *MdbxTx) ForEach(bucket string, fromPrefix []byte, walker func(k, v []byte) error) error {
|
|
c, err := tx.Cursor(bucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer c.Close()
|
|
|
|
for k, v, err := c.Seek(fromPrefix); k != nil; k, v, err = c.Next() {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := walker(k, v); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tx *MdbxTx) ForPrefix(bucket string, prefix []byte, walker func(k, v []byte) error) error {
|
|
c, err := tx.Cursor(bucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer c.Close()
|
|
|
|
for k, v, err := c.Seek(prefix); k != nil; k, v, err = c.Next() {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !bytes.HasPrefix(k, prefix) {
|
|
break
|
|
}
|
|
if err := walker(k, v); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
func (tx *MdbxTx) ForAmount(bucket string, fromPrefix []byte, amount uint32, walker func(k, v []byte) error) error {
|
|
if amount == 0 {
|
|
return nil
|
|
}
|
|
c, err := tx.Cursor(bucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer c.Close()
|
|
|
|
for k, v, err := c.Seek(fromPrefix); k != nil && amount > 0; k, v, err = c.Next() {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := walker(k, v); err != nil {
|
|
return err
|
|
}
|
|
amount--
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tx *MdbxTx) ViewID() uint64 { return tx.tx.ID() }
|
|
|
|
func (tx *MdbxTx) CollectMetrics() {
|
|
if tx.db.opts.label != kv.ChainDB {
|
|
return
|
|
}
|
|
|
|
info, err := tx.db.env.Info(tx.tx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if info.SinceReaderCheck.Hours() > 1 {
|
|
if staleReaders, err := tx.db.env.ReaderCheck(); err != nil {
|
|
tx.db.log.Error("failed ReaderCheck", "err", err)
|
|
} else if staleReaders > 0 {
|
|
tx.db.log.Info("[db] cleared reader slots from dead processes", "amount", staleReaders)
|
|
}
|
|
}
|
|
|
|
kv.DbSize.Set(info.Geo.Current)
|
|
kv.DbPgopsNewly.Set(info.PageOps.Newly)
|
|
kv.DbPgopsCow.Set(info.PageOps.Cow)
|
|
kv.DbPgopsClone.Set(info.PageOps.Clone)
|
|
kv.DbPgopsSplit.Set(info.PageOps.Split)
|
|
kv.DbPgopsMerge.Set(info.PageOps.Merge)
|
|
kv.DbPgopsSpill.Set(info.PageOps.Spill)
|
|
kv.DbPgopsUnspill.Set(info.PageOps.Unspill)
|
|
kv.DbPgopsWops.Set(info.PageOps.Wops)
|
|
kv.DbPgopsGcrtime.Update(info.PageOps.Gcrtime.Seconds())
|
|
|
|
txInfo, err := tx.tx.Info(true)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
kv.TxDirty.Set(txInfo.SpaceDirty)
|
|
kv.TxLimit.Set(tx.db.txSize)
|
|
kv.TxSpill.Set(txInfo.Spill)
|
|
kv.TxUnspill.Set(txInfo.Unspill)
|
|
|
|
gc, err := tx.BucketStat("gc")
|
|
if err != nil {
|
|
return
|
|
}
|
|
kv.GcLeafMetric.Set(gc.LeafPages)
|
|
kv.GcOverflowMetric.Set(gc.OverflowPages)
|
|
kv.GcPagesMetric.Set((gc.LeafPages + gc.OverflowPages) * tx.db.opts.pageSize / 8)
|
|
}
|
|
|
|
// ListBuckets - all buckets stored as keys of un-named bucket
|
|
func (tx *MdbxTx) ListBuckets() ([]string, error) {
|
|
return tx.tx.ListDBI()
|
|
}
|
|
|
|
func (db *MdbxKV) View(ctx context.Context, f func(tx kv.Tx) error) (err error) {
|
|
if db.closed.Load() {
|
|
return fmt.Errorf("db closed")
|
|
}
|
|
// can't use db.evn.View method - because it calls commit for read transactions - it conflicts with write transactions.
|
|
tx, err := db.BeginRo(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
return f(tx)
|
|
}
|
|
|
|
func (db *MdbxKV) Update(ctx context.Context, f func(tx kv.RwTx) error) (err error) {
|
|
if db.closed.Load() {
|
|
return fmt.Errorf("db closed")
|
|
}
|
|
|
|
tx, err := db.BeginRw(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
err = f(tx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = tx.Commit()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tx *MdbxTx) CreateBucket(name string) error {
|
|
cnfCopy := tx.db.buckets[name]
|
|
dbi, err := tx.tx.OpenDBI(name, mdbx.DBAccede, nil, nil)
|
|
if err != nil && !mdbx.IsNotFound(err) {
|
|
return fmt.Errorf("create bucket: %s, %w", name, err)
|
|
}
|
|
if err == nil {
|
|
cnfCopy.DBI = kv.DBI(dbi)
|
|
var flags uint
|
|
flags, err = tx.tx.Flags(dbi)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cnfCopy.Flags = kv.TableFlags(flags)
|
|
|
|
tx.db.buckets[name] = cnfCopy
|
|
return nil
|
|
}
|
|
|
|
// if bucket doesn't exists - create it
|
|
|
|
var flags = tx.db.buckets[name].Flags
|
|
var nativeFlags uint
|
|
if tx.db.opts.flags&mdbx.Readonly == 0 {
|
|
nativeFlags |= mdbx.Create
|
|
}
|
|
|
|
if flags&kv.DupSort != 0 {
|
|
nativeFlags |= mdbx.DupSort
|
|
flags ^= kv.DupSort
|
|
}
|
|
if flags != 0 {
|
|
return fmt.Errorf("some not supported flag provided for bucket")
|
|
}
|
|
|
|
dbi, err = tx.tx.OpenDBI(name, nativeFlags, nil, nil)
|
|
|
|
if err != nil {
|
|
return fmt.Errorf("create bucket: %s, %w", name, err)
|
|
}
|
|
cnfCopy.DBI = kv.DBI(dbi)
|
|
|
|
tx.db.buckets[name] = cnfCopy
|
|
return nil
|
|
}
|
|
|
|
func (tx *MdbxTx) dropEvenIfBucketIsNotDeprecated(name string) error {
|
|
dbi := tx.db.buckets[name].DBI
|
|
// if bucket was not open on db start, then it's may be deprecated
|
|
// try to open it now without `Create` flag, and if fail then nothing to drop
|
|
if dbi == NonExistingDBI {
|
|
nativeDBI, err := tx.tx.OpenDBI(name, 0, nil, nil)
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil // DBI doesn't exists means no drop needed
|
|
}
|
|
return fmt.Errorf("bucket: %s, %w", name, err)
|
|
}
|
|
dbi = kv.DBI(nativeDBI)
|
|
}
|
|
|
|
if err := tx.tx.Drop(mdbx.DBI(dbi), true); err != nil {
|
|
return err
|
|
}
|
|
cnfCopy := tx.db.buckets[name]
|
|
cnfCopy.DBI = NonExistingDBI
|
|
tx.db.buckets[name] = cnfCopy
|
|
return nil
|
|
}
|
|
|
|
func (tx *MdbxTx) ClearBucket(bucket string) error {
|
|
dbi := tx.db.buckets[bucket].DBI
|
|
if dbi == NonExistingDBI {
|
|
return nil
|
|
}
|
|
return tx.tx.Drop(mdbx.DBI(dbi), false)
|
|
}
|
|
|
|
func (tx *MdbxTx) DropBucket(bucket string) error {
|
|
if cfg, ok := tx.db.buckets[bucket]; !(ok && cfg.IsDeprecated) {
|
|
return fmt.Errorf("%w, bucket: %s", kv.ErrAttemptToDeleteNonDeprecatedBucket, bucket)
|
|
}
|
|
|
|
return tx.dropEvenIfBucketIsNotDeprecated(bucket)
|
|
}
|
|
|
|
func (tx *MdbxTx) ExistsBucket(bucket string) (bool, error) {
|
|
if cfg, ok := tx.db.buckets[bucket]; ok {
|
|
return cfg.DBI != NonExistingDBI, nil
|
|
}
|
|
return false, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) Commit() error {
|
|
if tx.tx == nil {
|
|
return nil
|
|
}
|
|
defer func() {
|
|
tx.tx = nil
|
|
tx.db.wg.Done()
|
|
if tx.readOnly {
|
|
tx.db.roTxsLimiter.Release(1)
|
|
} else {
|
|
runtime.UnlockOSThread()
|
|
}
|
|
}()
|
|
tx.closeCursors()
|
|
|
|
//slowTx := 10 * time.Second
|
|
//if debug.SlowCommit() > 0 {
|
|
// slowTx = debug.SlowCommit()
|
|
//}
|
|
//
|
|
//if debug.BigRoTxKb() > 0 || debug.BigRwTxKb() > 0 {
|
|
// tx.PrintDebugInfo()
|
|
//}
|
|
tx.CollectMetrics()
|
|
|
|
latency, err := tx.tx.Commit()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if tx.db.opts.label == kv.ChainDB {
|
|
kv.DbCommitPreparation.Update(latency.Preparation.Seconds())
|
|
kv.DbCommitGc.Update(latency.GC.Seconds())
|
|
kv.DbCommitAudit.Update(latency.Audit.Seconds())
|
|
kv.DbCommitWrite.Update(latency.Write.Seconds())
|
|
kv.DbCommitSync.Update(latency.Sync.Seconds())
|
|
kv.DbCommitEnding.Update(latency.Ending.Seconds())
|
|
kv.DbCommitTotal.Update(latency.Whole.Seconds())
|
|
}
|
|
|
|
//if latency.Whole > slowTx {
|
|
// log.Info("Commit",
|
|
// "preparation", latency.Preparation,
|
|
// "gc", latency.GC,
|
|
// "audit", latency.Audit,
|
|
// "write", latency.Write,
|
|
// "fsync", latency.Sync,
|
|
// "ending", latency.Ending,
|
|
// "whole", latency.Whole,
|
|
// )
|
|
//}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (tx *MdbxTx) Rollback() {
|
|
if tx.tx == nil {
|
|
return
|
|
}
|
|
defer func() {
|
|
tx.tx = nil
|
|
tx.db.wg.Done()
|
|
if tx.readOnly {
|
|
tx.db.roTxsLimiter.Release(1)
|
|
} else {
|
|
runtime.UnlockOSThread()
|
|
}
|
|
}()
|
|
tx.closeCursors()
|
|
//tx.printDebugInfo()
|
|
tx.tx.Abort()
|
|
}
|
|
|
|
func (tx *MdbxTx) SpaceDirty() (uint64, uint64, error) {
|
|
txInfo, err := tx.tx.Info(true)
|
|
if err != nil {
|
|
return 0, 0, err
|
|
}
|
|
|
|
return txInfo.SpaceDirty, tx.db.txSize, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) PrintDebugInfo() {
|
|
/*
|
|
txInfo, err := tx.tx.Info(true)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
txSize := uint(txInfo.SpaceDirty / 1024)
|
|
doPrint := debug.BigRoTxKb() == 0 && debug.BigRwTxKb() == 0 ||
|
|
tx.readOnly && debug.BigRoTxKb() > 0 && txSize > debug.BigRoTxKb() ||
|
|
(!tx.readOnly && debug.BigRwTxKb() > 0 && txSize > debug.BigRwTxKb())
|
|
if doPrint {
|
|
tx.db.log.Info("Tx info",
|
|
"id", txInfo.Id,
|
|
"read_lag", txInfo.ReadLag,
|
|
"ro", tx.readOnly,
|
|
//"space_retired_mb", txInfo.SpaceRetired/1024/1024,
|
|
"space_dirty_mb", txInfo.SpaceDirty/1024/1024,
|
|
//"callers", debug.Callers(7),
|
|
)
|
|
}
|
|
*/
|
|
}
|
|
|
|
func (tx *MdbxTx) closeCursors() {
|
|
for _, c := range tx.cursors {
|
|
if c != nil {
|
|
c.Close()
|
|
}
|
|
}
|
|
tx.cursors = nil
|
|
tx.statelessCursors = nil
|
|
}
|
|
|
|
func (tx *MdbxTx) statelessCursor(bucket string) (kv.RwCursor, error) {
|
|
if tx.statelessCursors == nil {
|
|
tx.statelessCursors = make(map[string]kv.Cursor)
|
|
}
|
|
c, ok := tx.statelessCursors[bucket]
|
|
if !ok {
|
|
var err error
|
|
c, err = tx.Cursor(bucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
tx.statelessCursors[bucket] = c
|
|
}
|
|
return c.(kv.RwCursor), nil
|
|
}
|
|
|
|
func (tx *MdbxTx) Put(table string, k, v []byte) error {
|
|
c, err := tx.statelessCursor(table)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.Put(k, v)
|
|
}
|
|
|
|
func (tx *MdbxTx) Delete(table string, k []byte) error {
|
|
c, err := tx.statelessCursor(table)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.Delete(k)
|
|
}
|
|
|
|
func (tx *MdbxTx) GetOne(bucket string, k []byte) ([]byte, error) {
|
|
c, err := tx.statelessCursor(bucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
_, v, err := c.SeekExact(k)
|
|
return v, err
|
|
}
|
|
|
|
func (tx *MdbxTx) Has(bucket string, key []byte) (bool, error) {
|
|
c, err := tx.statelessCursor(bucket)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
k, _, err := c.Seek(key)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return bytes.Equal(key, k), nil
|
|
}
|
|
|
|
func (tx *MdbxTx) Append(bucket string, k, v []byte) error {
|
|
c, err := tx.statelessCursor(bucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.Append(k, v)
|
|
}
|
|
func (tx *MdbxTx) AppendDup(bucket string, k, v []byte) error {
|
|
c, err := tx.statelessCursor(bucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.(*MdbxDupSortCursor).AppendDup(k, v)
|
|
}
|
|
|
|
func (tx *MdbxTx) IncrementSequence(bucket string, amount uint64) (uint64, error) {
|
|
c, err := tx.statelessCursor(kv.Sequence)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
_, v, err := c.SeekExact([]byte(bucket))
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
var currentV uint64 = 0
|
|
if len(v) > 0 {
|
|
currentV = binary.BigEndian.Uint64(v)
|
|
}
|
|
|
|
newVBytes := make([]byte, 8)
|
|
binary.BigEndian.PutUint64(newVBytes, currentV+amount)
|
|
err = c.Put([]byte(bucket), newVBytes)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return currentV, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) ReadSequence(bucket string) (uint64, error) {
|
|
c, err := tx.statelessCursor(kv.Sequence)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
_, v, err := c.SeekExact([]byte(bucket))
|
|
if err != nil && !mdbx.IsNotFound(err) {
|
|
return 0, err
|
|
}
|
|
|
|
var currentV uint64
|
|
if len(v) > 0 {
|
|
currentV = binary.BigEndian.Uint64(v)
|
|
}
|
|
|
|
return currentV, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) BucketSize(name string) (uint64, error) {
|
|
st, err := tx.BucketStat(name)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return (st.LeafPages + st.BranchPages + st.OverflowPages) * tx.db.opts.pageSize, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) BucketStat(name string) (*mdbx.Stat, error) {
|
|
if name == "freelist" || name == "gc" || name == "free_list" {
|
|
return tx.tx.StatDBI(mdbx.DBI(0))
|
|
}
|
|
if name == "root" {
|
|
return tx.tx.StatDBI(mdbx.DBI(1))
|
|
}
|
|
st, err := tx.tx.StatDBI(mdbx.DBI(tx.db.buckets[name].DBI))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("bucket: %s, %w", name, err)
|
|
}
|
|
return st, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) DBSize() (uint64, error) {
|
|
info, err := tx.db.env.Info(tx.tx)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return info.Geo.Current, err
|
|
}
|
|
|
|
func (tx *MdbxTx) Reset() (err error) {
|
|
tx.Rollback()
|
|
//tx.printDebugInfo()
|
|
if tx.db.closed.Load() {
|
|
return fmt.Errorf("db closed")
|
|
}
|
|
runtime.LockOSThread()
|
|
defer func() {
|
|
if err == nil {
|
|
tx.db.wg.Add(1)
|
|
}
|
|
}()
|
|
|
|
tx.tx, err = tx.db.env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
runtime.UnlockOSThread() // unlock only in case of error. normal flow is "defer .Rollback()"
|
|
return fmt.Errorf("%w, lable: %s, trace: %s", err, tx.db.opts.label.String(), stack2.Trace().String())
|
|
}
|
|
tx.tx.RawRead = true
|
|
return nil
|
|
}
|
|
|
|
func (tx *MdbxTx) RwCursor(bucket string) (kv.RwCursor, error) {
|
|
b := tx.db.buckets[bucket]
|
|
if b.AutoDupSortKeysConversion {
|
|
return tx.stdCursor(bucket)
|
|
}
|
|
|
|
if b.Flags&kv.DupSort != 0 {
|
|
return tx.RwCursorDupSort(bucket)
|
|
}
|
|
|
|
return tx.stdCursor(bucket)
|
|
}
|
|
|
|
func (tx *MdbxTx) Cursor(bucket string) (kv.Cursor, error) {
|
|
return tx.RwCursor(bucket)
|
|
}
|
|
|
|
func (tx *MdbxTx) stdCursor(bucket string) (kv.RwCursor, error) {
|
|
b := tx.db.buckets[bucket]
|
|
c := &MdbxCursor{bucketName: bucket, tx: tx, bucketCfg: b, dbi: mdbx.DBI(tx.db.buckets[bucket].DBI), id: tx.cursorID}
|
|
tx.cursorID++
|
|
|
|
var err error
|
|
c.c, err = tx.tx.OpenCursor(c.dbi)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("table: %s, %w", c.bucketName, err)
|
|
}
|
|
|
|
// add to auto-cleanup on end of transactions
|
|
if tx.cursors == nil {
|
|
tx.cursors = map[uint64]*mdbx.Cursor{}
|
|
}
|
|
tx.cursors[c.id] = c.c
|
|
return c, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) RwCursorDupSort(bucket string) (kv.RwCursorDupSort, error) {
|
|
basicCursor, err := tx.stdCursor(bucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &MdbxDupSortCursor{MdbxCursor: basicCursor.(*MdbxCursor)}, nil
|
|
}
|
|
|
|
func (tx *MdbxTx) CursorDupSort(bucket string) (kv.CursorDupSort, error) {
|
|
return tx.RwCursorDupSort(bucket)
|
|
}
|
|
|
|
// methods here help to see better pprof picture
|
|
func (c *MdbxCursor) set(k []byte) ([]byte, []byte, error) { return c.c.Get(k, nil, mdbx.Set) }
|
|
func (c *MdbxCursor) getCurrent() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.GetCurrent) }
|
|
func (c *MdbxCursor) first() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.First) }
|
|
func (c *MdbxCursor) next() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.Next) }
|
|
func (c *MdbxCursor) nextDup() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.NextDup) }
|
|
func (c *MdbxCursor) nextNoDup() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.NextNoDup) }
|
|
func (c *MdbxCursor) prev() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.Prev) }
|
|
func (c *MdbxCursor) prevDup() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.PrevDup) }
|
|
func (c *MdbxCursor) prevNoDup() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.PrevNoDup) }
|
|
func (c *MdbxCursor) last() ([]byte, []byte, error) { return c.c.Get(nil, nil, mdbx.Last) }
|
|
func (c *MdbxCursor) delCurrent() error { return c.c.Del(mdbx.Current) }
|
|
func (c *MdbxCursor) delAllDupData() error { return c.c.Del(mdbx.AllDups) }
|
|
func (c *MdbxCursor) put(k, v []byte) error { return c.c.Put(k, v, 0) }
|
|
func (c *MdbxCursor) putCurrent(k, v []byte) error { return c.c.Put(k, v, mdbx.Current) }
|
|
func (c *MdbxCursor) putNoOverwrite(k, v []byte) error { return c.c.Put(k, v, mdbx.NoOverwrite) }
|
|
func (c *MdbxCursor) putNoDupData(k, v []byte) error { return c.c.Put(k, v, mdbx.NoDupData) }
|
|
func (c *MdbxCursor) append(k, v []byte) error { return c.c.Put(k, v, mdbx.Append) }
|
|
func (c *MdbxCursor) appendDup(k, v []byte) error { return c.c.Put(k, v, mdbx.AppendDup) }
|
|
func (c *MdbxCursor) getBoth(k, v []byte) ([]byte, error) {
|
|
_, v, err := c.c.Get(k, v, mdbx.GetBoth)
|
|
return v, err
|
|
}
|
|
func (c *MdbxCursor) setRange(k []byte) ([]byte, []byte, error) {
|
|
return c.c.Get(k, nil, mdbx.SetRange)
|
|
}
|
|
func (c *MdbxCursor) getBothRange(k, v []byte) ([]byte, error) {
|
|
_, v, err := c.c.Get(k, v, mdbx.GetBothRange)
|
|
return v, err
|
|
}
|
|
func (c *MdbxCursor) firstDup() ([]byte, error) {
|
|
_, v, err := c.c.Get(nil, nil, mdbx.FirstDup)
|
|
return v, err
|
|
}
|
|
func (c *MdbxCursor) lastDup() ([]byte, error) {
|
|
_, v, err := c.c.Get(nil, nil, mdbx.LastDup)
|
|
return v, err
|
|
}
|
|
|
|
func (c *MdbxCursor) Count() (uint64, error) {
|
|
st, err := c.tx.tx.StatDBI(c.dbi)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return st.Entries, nil
|
|
}
|
|
|
|
func (c *MdbxCursor) First() ([]byte, []byte, error) {
|
|
return c.Seek(nil)
|
|
}
|
|
|
|
func (c *MdbxCursor) Last() ([]byte, []byte, error) {
|
|
k, v, err := c.last()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
err = fmt.Errorf("failed MdbxKV cursor.Last(): %w, bucket: %s", err, c.bucketName)
|
|
return []byte{}, nil, err
|
|
}
|
|
|
|
b := c.bucketCfg
|
|
if b.AutoDupSortKeysConversion && len(k) == b.DupToLen {
|
|
keyPart := b.DupFromLen - b.DupToLen
|
|
k = append(k, v[:keyPart]...)
|
|
v = v[keyPart:]
|
|
}
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxCursor) Seek(seek []byte) (k, v []byte, err error) {
|
|
if c.bucketCfg.AutoDupSortKeysConversion {
|
|
return c.seekDupSort(seek)
|
|
}
|
|
|
|
if len(seek) == 0 {
|
|
k, v, err = c.first()
|
|
} else {
|
|
k, v, err = c.setRange(seek)
|
|
}
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
err = fmt.Errorf("failed MdbxKV cursor.Seek(): %w, bucket: %s, key: %x", err, c.bucketName, seek)
|
|
return []byte{}, nil, err
|
|
}
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxCursor) seekDupSort(seek []byte) (k, v []byte, err error) {
|
|
b := c.bucketCfg
|
|
from, to := b.DupFromLen, b.DupToLen
|
|
if len(seek) == 0 {
|
|
k, v, err = c.first()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, err
|
|
}
|
|
|
|
if len(k) == to {
|
|
k2 := make([]byte, 0, len(k)+from-to)
|
|
k2 = append(append(k2, k...), v[:from-to]...)
|
|
v = v[from-to:]
|
|
k = k2
|
|
}
|
|
return k, v, nil
|
|
}
|
|
|
|
var seek1, seek2 []byte
|
|
if len(seek) > to {
|
|
seek1, seek2 = seek[:to], seek[to:]
|
|
} else {
|
|
seek1 = seek
|
|
}
|
|
k, v, err = c.setRange(seek1)
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
|
|
return []byte{}, nil, err
|
|
}
|
|
|
|
if seek2 != nil && bytes.Equal(seek1, k) {
|
|
v, err = c.getBothRange(seek1, seek2)
|
|
if err != nil && mdbx.IsNotFound(err) {
|
|
k, v, err = c.next()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, err
|
|
}
|
|
} else if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
}
|
|
if len(k) == to {
|
|
k2 := make([]byte, 0, len(k)+from-to)
|
|
k2 = append(append(k2, k...), v[:from-to]...)
|
|
v = v[from-to:]
|
|
k = k2
|
|
}
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxCursor) Next() (k, v []byte, err error) {
|
|
k, v, err = c.next()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, fmt.Errorf("failed MdbxKV cursor.Next(): %w", err)
|
|
}
|
|
|
|
b := c.bucketCfg
|
|
if b.AutoDupSortKeysConversion && len(k) == b.DupToLen {
|
|
keyPart := b.DupFromLen - b.DupToLen
|
|
k = append(k, v[:keyPart]...)
|
|
v = v[keyPart:]
|
|
}
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxCursor) Prev() (k, v []byte, err error) {
|
|
k, v, err = c.prev()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, fmt.Errorf("failed MdbxKV cursor.Prev(): %w", err)
|
|
}
|
|
|
|
b := c.bucketCfg
|
|
if b.AutoDupSortKeysConversion && len(k) == b.DupToLen {
|
|
keyPart := b.DupFromLen - b.DupToLen
|
|
k = append(k, v[:keyPart]...)
|
|
v = v[keyPart:]
|
|
}
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
// Current - return key/data at current cursor position
|
|
func (c *MdbxCursor) Current() ([]byte, []byte, error) {
|
|
k, v, err := c.getCurrent()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, err
|
|
}
|
|
|
|
b := c.bucketCfg
|
|
if b.AutoDupSortKeysConversion && len(k) == b.DupToLen {
|
|
keyPart := b.DupFromLen - b.DupToLen
|
|
k = append(k, v[:keyPart]...)
|
|
v = v[keyPart:]
|
|
}
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxCursor) Delete(k []byte) error {
|
|
if c.bucketCfg.AutoDupSortKeysConversion {
|
|
return c.deleteDupSort(k)
|
|
}
|
|
|
|
_, _, err := c.set(k)
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
if c.bucketCfg.Flags&mdbx.DupSort != 0 {
|
|
return c.delAllDupData()
|
|
}
|
|
return c.delCurrent()
|
|
}
|
|
|
|
// DeleteCurrent This function deletes the key/data pair to which the cursor refers.
|
|
// This does not invalidate the cursor, so operations such as MDB_NEXT
|
|
// can still be used on it.
|
|
// Both MDB_NEXT and MDB_GET_CURRENT will return the same record after
|
|
// this operation.
|
|
func (c *MdbxCursor) DeleteCurrent() error {
|
|
return c.delCurrent()
|
|
}
|
|
|
|
func (c *MdbxCursor) deleteDupSort(key []byte) error {
|
|
b := c.bucketCfg
|
|
from, to := b.DupFromLen, b.DupToLen
|
|
if len(key) != from && len(key) >= to {
|
|
return fmt.Errorf("delete from dupsort bucket: %s, can have keys of len==%d and len<%d. key: %x,%d", c.bucketName, from, to, key, len(key))
|
|
}
|
|
|
|
if len(key) == from {
|
|
v, err := c.getBothRange(key[:to], key[to:])
|
|
if err != nil { // if key not found, or found another one - then nothing to delete
|
|
if mdbx.IsNotFound(err) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
if !bytes.Equal(v[:from-to], key[to:]) {
|
|
return nil
|
|
}
|
|
return c.delCurrent()
|
|
}
|
|
|
|
_, _, err := c.set(key)
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
return c.delCurrent()
|
|
}
|
|
|
|
func (c *MdbxCursor) PutNoOverwrite(key []byte, value []byte) error {
|
|
if c.bucketCfg.AutoDupSortKeysConversion {
|
|
panic("not implemented")
|
|
}
|
|
|
|
return c.putNoOverwrite(key, value)
|
|
}
|
|
|
|
func (c *MdbxCursor) Put(key []byte, value []byte) error {
|
|
b := c.bucketCfg
|
|
if b.AutoDupSortKeysConversion {
|
|
if err := c.putDupSort(key, value); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
if err := c.put(key, value); err != nil {
|
|
return fmt.Errorf("table: %s, err: %w", c.bucketName, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *MdbxCursor) putDupSort(key []byte, value []byte) error {
|
|
b := c.bucketCfg
|
|
from, to := b.DupFromLen, b.DupToLen
|
|
if len(key) != from && len(key) >= to {
|
|
return fmt.Errorf("put dupsort bucket: %s, can have keys of len==%d and len<%d. key: %x,%d", c.bucketName, from, to, key, len(key))
|
|
}
|
|
|
|
if len(key) != from {
|
|
err := c.putNoOverwrite(key, value)
|
|
if err != nil {
|
|
if mdbx.IsKeyExists(err) {
|
|
return c.putCurrent(key, value)
|
|
}
|
|
return fmt.Errorf("putNoOverwrite, bucket: %s, key: %x, val: %x, err: %w", c.bucketName, key, value, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
value = append(key[to:], value...)
|
|
key = key[:to]
|
|
v, err := c.getBothRange(key, value[:from-to])
|
|
if err != nil { // if key not found, or found another one - then just insert
|
|
if mdbx.IsNotFound(err) {
|
|
return c.put(key, value)
|
|
}
|
|
return err
|
|
}
|
|
|
|
if bytes.Equal(v[:from-to], value[:from-to]) {
|
|
if len(v) == len(value) { // in DupSort case mdbx.Current works only with values of same length
|
|
return c.putCurrent(key, value)
|
|
}
|
|
err = c.delCurrent()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return c.put(key, value)
|
|
}
|
|
|
|
func (c *MdbxCursor) SeekExact(key []byte) ([]byte, []byte, error) {
|
|
b := c.bucketCfg
|
|
if b.AutoDupSortKeysConversion && len(key) == b.DupFromLen {
|
|
from, to := b.DupFromLen, b.DupToLen
|
|
v, err := c.getBothRange(key[:to], key[to:])
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, err
|
|
}
|
|
if !bytes.Equal(key[to:], v[:from-to]) {
|
|
return nil, nil, nil
|
|
}
|
|
return key[:to], v[from-to:], nil
|
|
}
|
|
|
|
k, v, err := c.set(key)
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, err
|
|
}
|
|
return k, v, nil
|
|
}
|
|
|
|
// Append - speedy feature of mdbx which is not part of KV interface.
|
|
// Cast your cursor to *MdbxCursor to use this method.
|
|
// Return error - if provided data will not sorted (or bucket have old records which mess with new in sorting manner).
|
|
func (c *MdbxCursor) Append(k []byte, v []byte) error {
|
|
if len(k) == 0 {
|
|
return fmt.Errorf("mdbx doesn't support empty keys. bucket: %s", c.bucketName)
|
|
}
|
|
b := c.bucketCfg
|
|
if b.AutoDupSortKeysConversion {
|
|
from, to := b.DupFromLen, b.DupToLen
|
|
if len(k) != from && len(k) >= to {
|
|
return fmt.Errorf("append dupsort bucket: %s, can have keys of len==%d and len<%d. key: %x,%d", c.bucketName, from, to, k, len(k))
|
|
}
|
|
|
|
if len(k) == from {
|
|
v = append(k[to:], v...)
|
|
k = k[:to]
|
|
}
|
|
}
|
|
|
|
if b.Flags&mdbx.DupSort != 0 {
|
|
if err := c.appendDup(k, v); err != nil {
|
|
return fmt.Errorf("bucket: %s, %w", c.bucketName, err)
|
|
}
|
|
return nil
|
|
}
|
|
if err := c.append(k, v); err != nil {
|
|
return fmt.Errorf("bucket: %s, %w", c.bucketName, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *MdbxCursor) Close() {
|
|
if c.c != nil {
|
|
c.c.Close()
|
|
delete(c.tx.cursors, c.id)
|
|
c.c = nil
|
|
}
|
|
}
|
|
|
|
type MdbxDupSortCursor struct {
|
|
*MdbxCursor
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) Internal() *mdbx.Cursor {
|
|
return c.c
|
|
}
|
|
|
|
// DeleteExact - does delete
|
|
func (c *MdbxDupSortCursor) DeleteExact(k1, k2 []byte) error {
|
|
_, err := c.getBoth(k1, k2)
|
|
if err != nil { // if key not found, or found another one - then nothing to delete
|
|
if mdbx.IsNotFound(err) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
return c.delCurrent()
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) SeekBothExact(key, value []byte) ([]byte, []byte, error) {
|
|
v, err := c.getBoth(key, value)
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, fmt.Errorf("in SeekBothExact: %w", err)
|
|
}
|
|
return key, v, nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) SeekBothRange(key, value []byte) ([]byte, error) {
|
|
v, err := c.getBothRange(key, value)
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil
|
|
}
|
|
return nil, fmt.Errorf("in SeekBothRange: %w", err)
|
|
}
|
|
return v, nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) FirstDup() ([]byte, error) {
|
|
v, err := c.firstDup()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil
|
|
}
|
|
return nil, fmt.Errorf("in FirstDup: %w", err)
|
|
}
|
|
return v, nil
|
|
}
|
|
|
|
// NextDup - iterate only over duplicates of current key
|
|
func (c *MdbxDupSortCursor) NextDup() ([]byte, []byte, error) {
|
|
k, v, err := c.nextDup()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, fmt.Errorf("in NextDup: %w", err)
|
|
}
|
|
return k, v, nil
|
|
}
|
|
|
|
// NextNoDup - iterate with skipping all duplicates
|
|
func (c *MdbxDupSortCursor) NextNoDup() ([]byte, []byte, error) {
|
|
k, v, err := c.nextNoDup()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, fmt.Errorf("in NextNoDup: %w", err)
|
|
}
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) PrevDup() ([]byte, []byte, error) {
|
|
k, v, err := c.prevDup()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, fmt.Errorf("in PrevDup: %w", err)
|
|
}
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) PrevNoDup() ([]byte, []byte, error) {
|
|
k, v, err := c.prevNoDup()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil, nil
|
|
}
|
|
return []byte{}, nil, fmt.Errorf("in PrevNoDup: %w", err)
|
|
}
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) LastDup() ([]byte, error) {
|
|
v, err := c.lastDup()
|
|
if err != nil {
|
|
if mdbx.IsNotFound(err) {
|
|
return nil, nil
|
|
}
|
|
return nil, fmt.Errorf("in LastDup: %w", err)
|
|
}
|
|
return v, nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) Append(k []byte, v []byte) error {
|
|
if err := c.c.Put(k, v, mdbx.Append|mdbx.AppendDup); err != nil {
|
|
return fmt.Errorf("in Append: bucket=%s, %w", c.bucketName, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) AppendDup(k []byte, v []byte) error {
|
|
if err := c.appendDup(k, v); err != nil {
|
|
return fmt.Errorf("in AppendDup: bucket=%s, %w", c.bucketName, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *MdbxDupSortCursor) PutNoDupData(key, value []byte) error {
|
|
if err := c.putNoDupData(key, value); err != nil {
|
|
return fmt.Errorf("in PutNoDupData: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// DeleteCurrentDuplicates - delete all of the data items for the current key.
|
|
func (c *MdbxDupSortCursor) DeleteCurrentDuplicates() error {
|
|
if err := c.delAllDupData(); err != nil {
|
|
return fmt.Errorf("in DeleteCurrentDuplicates: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// CountDuplicates returns the number of duplicates for the current key. See mdb_cursor_count
|
|
func (c *MdbxDupSortCursor) CountDuplicates() (uint64, error) {
|
|
res, err := c.c.Count()
|
|
if err != nil {
|
|
return 0, fmt.Errorf("in CountDuplicates: %w", err)
|
|
}
|
|
return res, nil
|
|
}
|
|
|
|
func bucketSlice(b kv.TableCfg) []string {
|
|
buckets := make([]string, 0, len(b))
|
|
for name := range b {
|
|
buckets = append(buckets, name)
|
|
}
|
|
sort.Slice(buckets, func(i, j int) bool {
|
|
return strings.Compare(buckets[i], buckets[j]) < 0
|
|
})
|
|
return buckets
|
|
}
|