Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Release inputs if a RPC fails after the txn is funded and before it is broadcast #147

Merged
merged 2 commits into from
Jan 10, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
default: patch
---

# Release locked UTXOs if contract formation, renewal, or refresh fails
17 changes: 17 additions & 0 deletions rhp/v4/rpc.go
Original file line number Diff line number Diff line change
Expand Up @@ -568,11 +568,13 @@ func RPCFormContract(ctx context.Context, t TransportClient, tp TxPool, signer F
RenterParents: formationSet,
}
if err := rhp4.WriteRequest(s, rhp4.RPCFormContractID, &req); err != nil {
signer.ReleaseInputs([]types.V2Transaction{formationTxn})
return RPCFormContractResult{}, fmt.Errorf("failed to write request: %w", err)
}

var hostInputsResp rhp4.RPCFormContractResponse
if err := rhp4.ReadResponse(s, &hostInputsResp); err != nil {
signer.ReleaseInputs([]types.V2Transaction{formationTxn})
return RPCFormContractResult{}, fmt.Errorf("failed to read host inputs response: %w", err)
}

Expand All @@ -584,6 +586,7 @@ func RPCFormContract(ctx context.Context, t TransportClient, tp TxPool, signer F
}

if n := hostInputSum.Cmp(fc.TotalCollateral); n < 0 {
signer.ReleaseInputs([]types.V2Transaction{formationTxn})
return RPCFormContractResult{}, fmt.Errorf("expected host to fund at least %v, got %v", fc.TotalCollateral, hostInputSum)
} else if n > 0 {
// add change output
Expand All @@ -606,12 +609,14 @@ func RPCFormContract(ctx context.Context, t TransportClient, tp TxPool, signer F
}
// send the renter signatures
if err := rhp4.WriteResponse(s, &renterPolicyResp); err != nil {
signer.ReleaseInputs([]types.V2Transaction{formationTxn})
return RPCFormContractResult{}, fmt.Errorf("failed to write signature response: %w", err)
}

// read the finalized transaction set
var hostTransactionSetResp rhp4.RPCFormContractThirdResponse
if err := rhp4.ReadResponse(s, &hostTransactionSetResp); err != nil {
// at this point the formation txn must already have been broadcast, so no need to release inputs
return RPCFormContractResult{}, fmt.Errorf("failed to read final response: %w", err)
}

Expand Down Expand Up @@ -681,6 +686,7 @@ func RPCRenewContract(ctx context.Context, t TransportClient, tp TxPool, signer

req.Basis, req.RenterParents, err = tp.V2TransactionSet(basis, renewalTxn)
if err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRenewContractResult{}, fmt.Errorf("failed to get transaction set: %w", err)
}
for _, si := range renewalTxn.SiacoinInputs {
Expand All @@ -695,11 +701,13 @@ func RPCRenewContract(ctx context.Context, t TransportClient, tp TxPool, signer
defer s.Close()

if err := rhp4.WriteRequest(s, rhp4.RPCRenewContractID, &req); err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRenewContractResult{}, fmt.Errorf("failed to write request: %w", err)
}

var hostInputsResp rhp4.RPCRenewContractResponse
if err := rhp4.ReadResponse(s, &hostInputsResp); err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRenewContractResult{}, fmt.Errorf("failed to read host inputs response: %w", err)
}

Expand All @@ -712,6 +720,7 @@ func RPCRenewContract(ctx context.Context, t TransportClient, tp TxPool, signer

// verify the host added enough inputs
if n := hostInputSum.Cmp(hostCost); n < 0 {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRenewContractResult{}, fmt.Errorf("expected host to fund %v, got %v", hostCost, hostInputSum)
} else if n > 0 {
// add change output
Expand Down Expand Up @@ -739,12 +748,14 @@ func RPCRenewContract(ctx context.Context, t TransportClient, tp TxPool, signer
renterPolicyResp.RenterSatisfiedPolicies = append(renterPolicyResp.RenterSatisfiedPolicies, si.SatisfiedPolicy)
}
if err := rhp4.WriteResponse(s, &renterPolicyResp); err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRenewContractResult{}, fmt.Errorf("failed to write signature response: %w", err)
}

// read the finalized transaction set
var hostTransactionSetResp rhp4.RPCRenewContractThirdResponse
if err := rhp4.ReadResponse(s, &hostTransactionSetResp); err != nil {
// at this point the renewal txn must already have been broadcast, so no need to release inputs
return RPCRenewContractResult{}, fmt.Errorf("failed to read final response: %w", err)
}

Expand Down Expand Up @@ -812,6 +823,7 @@ func RPCRefreshContract(ctx context.Context, t TransportClient, tp TxPool, signe

req.Basis, req.RenterParents, err = tp.V2TransactionSet(basis, renewalTxn)
if err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRefreshContractResult{}, fmt.Errorf("failed to get transaction set: %w", err)
}
for _, si := range renewalTxn.SiacoinInputs {
Expand All @@ -826,11 +838,13 @@ func RPCRefreshContract(ctx context.Context, t TransportClient, tp TxPool, signe
defer s.Close()

if err := rhp4.WriteRequest(s, rhp4.RPCRefreshContractID, &req); err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRefreshContractResult{}, fmt.Errorf("failed to write request: %w", err)
}

var hostInputsResp rhp4.RPCRefreshContractResponse
if err := rhp4.ReadResponse(s, &hostInputsResp); err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRefreshContractResult{}, fmt.Errorf("failed to read host inputs response: %w", err)
}

Expand All @@ -843,6 +857,7 @@ func RPCRefreshContract(ctx context.Context, t TransportClient, tp TxPool, signe

// verify the host added enough inputs
if n := hostInputSum.Cmp(hostCost); n < 0 {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRefreshContractResult{}, fmt.Errorf("expected host to fund %v, got %v", hostCost, hostInputSum)
} else if n > 0 {
// add change output
Expand Down Expand Up @@ -870,12 +885,14 @@ func RPCRefreshContract(ctx context.Context, t TransportClient, tp TxPool, signe
renterPolicyResp.RenterSatisfiedPolicies = append(renterPolicyResp.RenterSatisfiedPolicies, si.SatisfiedPolicy)
}
if err := rhp4.WriteResponse(s, &renterPolicyResp); err != nil {
signer.ReleaseInputs([]types.V2Transaction{renewalTxn})
return RPCRefreshContractResult{}, fmt.Errorf("failed to write signature response: %w", err)
}

// read the finalized transaction set
var hostTransactionSetResp rhp4.RPCRefreshContractThirdResponse
if err := rhp4.ReadResponse(s, &hostTransactionSetResp); err != nil {
// at this point the renewal txn must already have been broadcast, so no need to release inputs
return RPCRefreshContractResult{}, fmt.Errorf("failed to read final response: %w", err)
}

Expand Down
Loading