diff --git a/cmd/resign/emit_withdrawals/emit_withdrawals.go b/cmd/resign/emit_withdrawals/emit_withdrawals.go new file mode 100644 index 00000000000..0d78229423d --- /dev/null +++ b/cmd/resign/emit_withdrawals/emit_withdrawals.go @@ -0,0 +1,200 @@ +// Copyright (C) 2023 Gobalsky Labs Limited +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as +// published by the Free Software Foundation, either version 3 of the +// License, or (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . + +package emit_withdrawals + +import ( + "bytes" + "encoding/csv" + "flag" + "fmt" + "log" + "os" + "sort" + "strconv" + "time" + + "code.vegaprotocol.io/vega/core/bridges" + "code.vegaprotocol.io/vega/core/config" + "code.vegaprotocol.io/vega/core/nodewallets" + "code.vegaprotocol.io/vega/libs/num" + "code.vegaprotocol.io/vega/paths" +) + +const ( + addressIndex = 1 + assetIndex = 2 + amountIndex = 4 + creationIndex = 5 + nonceIndex = 6 +) + +var ( + bundlesPath string + out string + passphrase config.Passphrase + home string +) + +var ( + bridgeAddressMapping = map[string]string{ + "WETH": "0x23872549cE10B40e31D6577e0A920088B0E0666a", + "USDT ETH": "0x23872549cE10B40e31D6577e0A920088B0E0666a", + "USDT ARB": "0x475B597652bCb2769949FD6787b1DC6916518407", + "USDC": "0x23872549cE10B40e31D6577e0A920088B0E0666a", + } + + assetAddressMapping = map[string]string{ + "WETH": "0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2", + "USDT ETH": "0xdAC17F958D2ee523a2206206994597C13D831ec7", + "USDT ARB": "0xFd086bC7CD5C481DCC9C85ebE478A1C0b69FCbb9", + "USDC": "0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48", + } + + bridgeChainIDMapping = map[string]string{ + "0x23872549cE10B40e31D6577e0A920088B0E0666a": "", + "0x475B597652bCb2769949FD6787b1DC6916518407": "42161", + } +) + +func init() { + flag.StringVar(&out, "out", "rebundled.csv", "where to store the outputs rebundled signatures") + flag.StringVar(&home, "home", "", "path to the vega home root (required)") + flag.StringVar((*string)(&passphrase), "passphrase", "", "passphrase of the node wallet") + flag.StringVar(&bundlesPath, "bundles", "", "path to the signatures bundles (required)") +} + +func readCSV(path string) [][]string { + buf, err := os.ReadFile(path) + if err != nil { + log.Fatalf("couldn't open the bundles file: %v", err) + } + + r := csv.NewReader(bytes.NewReader(buf)) + records, err := r.ReadAll() + if err != nil { + log.Fatal(err) + } + + return records[1:] +} + +func signBundle( + s bridges.Signer, + token, amountStr, receiver, creationStr, nonceStr string, +) string { + var ( + bridgeAddress = bridgeAddressMapping[token] + tokenAddress = assetAddressMapping[token] + chainID = bridgeChainIDMapping[bridgeAddress] + erc20Logic = bridges.NewERC20Logic(s, bridgeAddress, chainID, chainID == "") + ) + + nonce, overflowed := num.UintFromString(nonceStr, 10) + if overflowed { + log.Fatalf("invalid nonce, needs to be base 10, got: %v", nonceStr) + } + + amount, overflowed := num.UintFromString(amountStr, 10) + if overflowed { + log.Fatalf("invalid amount, needs to be base 10, got: %v", amountStr) + } + + creationInt64, err := strconv.ParseInt(creationStr, 10, 64) + if err != nil { + panic(err) + } + creation := time.Unix(creationInt64, 0) + + bundle, err := erc20Logic.WithdrawAsset( + tokenAddress, amount, receiver, creation, nonce, + ) + if err != nil { + log.Fatalf("couldn't sign bundle: %v", err) + } + + return fmt.Sprintf("0x%v", bundle.Signature.Hex()) +} + +func signAllBundles(s bridges.Signer, bundles [][]string) [][]string { + entries := [][]string{} + for _, entry := range bundles { + signature := signBundle(s, + entry[assetIndex], + entry[amountIndex], + entry[addressIndex], + entry[creationIndex], + entry[nonceIndex], + ) + + entries = append(entries, append(entry, signature)) + } + + return entries +} + +func Main() { + flag.Parse() + + if len(home) <= 0 { + log.Fatal("-home argument is required") + } + if len(bundlesPath) <= 0 { + log.Fatal("-bundles argument is required") + } + + bundles := readCSV(bundlesPath) + + pass, err := passphrase.Get("node wallet", false) + if err != nil { + log.Fatalf("couldn't get the node wallet passphrase: %v", err) + } + + vegaPaths := paths.New(home) + if _, _, err := config.EnsureNodeConfig(vegaPaths); err != nil { + log.Fatalf("couldn't load vega configuration: %v", err) + } + + s, err := nodewallets.GetEthereumWallet(vegaPaths, pass) + if err != nil { + log.Fatalf("couldn't get Ethereum node wallet: %v", err) + } + + output := signAllBundles(s, bundles) + + // just sort it to get the same output across nodes + sort.Slice(output, func(i, j int) bool { + return output[i][creationIndex] < output[j][creationIndex] + }) + + // then add headers + output = append([][]string{ + { + "party", "address", "asset", "realAmount", "amount", "createdTimestamp", "nonce", "signature", + }, + }, output...) + + f, err := os.Create(out) + if err != nil { + log.Fatalf("couldn't create file: %v", err) + } + + w := csv.NewWriter(f) + w.WriteAll(output) + + if err := w.Error(); err != nil { + log.Fatalf("error writing output:", err) + } +} diff --git a/cmd/resign/main.go b/cmd/resign/main.go new file mode 100644 index 00000000000..9ef9d84fee4 --- /dev/null +++ b/cmd/resign/main.go @@ -0,0 +1,22 @@ +// Copyright (C) 2023 Gobalsky Labs Limited +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as +// published by the Free Software Foundation, either version 3 of the +// License, or (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . + +package main + +import "code.vegaprotocol.io/vega/cmd/resign/emit_withdrawals" + +func main() { + emit_withdrawals.Main() +} diff --git a/cmd/vega/commands/emit_withdrawals.go b/cmd/vega/commands/emit_withdrawals.go new file mode 100644 index 00000000000..1502215e2a7 --- /dev/null +++ b/cmd/vega/commands/emit_withdrawals.go @@ -0,0 +1,45 @@ +// Copyright (C) 2023 Gobalsky Labs Limited +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU Affero General Public License as +// published by the Free Software Foundation, either version 3 of the +// License, or (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU Affero General Public License for more details. +// +// You should have received a copy of the GNU Affero General Public License +// along with this program. If not, see . + +package commands + +import ( + "context" + "os" + + "code.vegaprotocol.io/vega/cmd/resign/emit_withdrawals" + + "github.com/jessevdk/go-flags" +) + +type emitWithdrawalsCmd struct{} + +func (opts *emitWithdrawalsCmd) Execute(_ []string) error { + os.Args = os.Args[1:] + emit_withdrawals.Main() + + return nil +} + +func EmitWithdrawals(ctx context.Context, parser *flags.Parser) error { + _, err := parser.AddCommand( + "emit_withdrawals", + "Emit unclaimed withdrawals on vega", + "Emit unclaimed withdrawals on vega", + &emitWithdrawalsCmd{}, + ) + + return err +} diff --git a/cmd/vega/commands/root.go b/cmd/vega/commands/root.go index ab7899f525c..508482eb732 100644 --- a/cmd/vega/commands/root.go +++ b/cmd/vega/commands/root.go @@ -49,6 +49,8 @@ func Main(ctx context.Context) error { switch os.Args[1] { case "tendermint", "tm", "cometbft": return (&cometbftCmd{}).Execute(nil) + case "emit_withdrawals": + return (&emitWithdrawalsCmd{}).Execute(nil) case "wallet": return (&walletCmd{}).Execute(nil) case "datanode": @@ -84,6 +86,7 @@ func Main(ctx context.Context) error { Start, Node, BlockExplorer, + EmitWithdrawals, ); err != nil { _, _ = fmt.Fprintln(os.Stderr, err) return err