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

Expose APIs for working with transaction proposals #1382

Merged
merged 10 commits into from
Mar 8, 2024
Merged

Conversation

str4d
Copy link
Collaborator

@str4d str4d commented Feb 21, 2024

Closes #1343.

This code review checklist is intended to serve as a starting point for the author and reviewer, although it may not be appropriate for all types of changes (e.g. fixing a spelling typo in documentation). For more in-depth discussion of how we think about code review, please see Code Review Guidelines.

Author

  • Self-review: Did you review your own code in GitHub's web interface? Code often looks different when reviewing the diff in a browser, making it easier to spot potential bugs.
  • Automated tests: Did you add appropriate automated tests for any code changes?
  • Code coverage: Did you check the code coverage report for the automated tests? While we are not looking for perfect coverage, the tool can point out potential cases that have been missed.
  • Documentation: Did you update Docs as appropiate? (E.g README.md, etc.)
  • Run the app: Did you run the app and try the changes?
  • Did you provide Screenshots of what the App looks like before and after your changes as part of the description of this PR? (only applicable to UI Changes)
  • Rebase and squash: Did you pull in the latest changes from the main branch and squash your commits before assigning a reviewer? Having your code up to date and squashed will make it easier for others to review. Use best judgement when squashing commits, as some changes (such as refactoring) might be easier to review as a separate commit.

Reviewer

  • Checklist review: Did you go through the code with the Code Review Guidelines checklist?
  • Ad hoc review: Did you perform an ad hoc review? In addition to a first pass using the code review guidelines, do a second pass using your best judgement and experience which may identify additional questions or comments. Research shows that code review is most effective when done in multiple passes, where reviewers look for different things through each pass.
  • Automated tests: Did you review the automated tests?
  • Manual tests: Did you review the manual tests?You will find manual testing guidelines under our manual testing section
  • How is Code Coverage affected by this PR? We encourage you to compare coverage befor and after your changes and when possible, leave it in a better place. Learn More...
  • Documentation: Did you review Docs, README.md, LICENSE.md, and Architecture.md as appropriate?
  • Run the app: Did you run the app and try the changes? While the CI server runs the app to look for build failures or crashes, humans running the app are more likely to notice unexpected log messages, UI inconsistencies, or bad output data.

@str4d str4d force-pushed the 1204-expose-proposals branch from 9afed2a to d4b0306 Compare February 21, 2024 01:14
@str4d str4d added the S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. label Feb 21, 2024
@str4d str4d marked this pull request as ready for review February 21, 2024 01:15
@str4d str4d force-pushed the 1204-expose-proposals branch from 35cc57b to 93e9954 Compare February 21, 2024 01:18
@str4d str4d marked this pull request as draft February 21, 2024 01:24
@str4d str4d force-pushed the 1204-expose-proposals branch from 93e9954 to 4d4d749 Compare February 22, 2024 01:09
@str4d str4d marked this pull request as ready for review February 22, 2024 01:09
Copy link
Contributor

@nuttycom nuttycom left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

utACK with questions that we should probably create follow-up issues to address.

usk: spendingKey
)

return txId
logger.debug("transaction id: \(txId)")
return [try await repository.find(rawID: txId)]
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does repository.find throw if it cannot find the txid? I would think that ideally that would return an optional result, so that "not found" is distinguished from "error" (though in this particular case you'd want to throw an error anyway, so it's nonblocking for now.)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We shouldn't encounter this case (successful wallet DB lookup that returns no result) because if we reach here, the transaction was successfully created by the Rust backend (and didn't itself result in an exception). Any exception thrown here would instead be due to database access failures, which are indeed exceptions.

Copy link
Contributor

@daira daira left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

utACK modulo comments.

@str4d str4d force-pushed the 1204-expose-proposals branch from 4d4d749 to cf6d4a1 Compare February 27, 2024 16:05
@str4d
Copy link
Collaborator Author

str4d commented Feb 27, 2024

Force-pushed to address review comments.

}
}

public extension Proposal {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would advise not to expose this publicly.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you have a suggestion for how to avoid exposing this publicly, I'm all ears. The constraint here is Zashi iOS using TCA, which needs to be able to construct mock data to be returned from its mock implementation of the Synchronizer protocol for use in e.g. preview of UIs (so we can't just return something that would be an error case).

Copy link
Contributor

@pacu pacu Feb 29, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

NOTE: I don't know which things you've tried and which ones you haven't, but there are a few things I can think from the top of my head. I promise I'll look this deeper shortly. I might be still lacking context and this could be somewhat inaccurate or insufficient.

A. "Uncle bob's Clean code" approach

Treat the whole SDK as an external dependency under the principle of Bob Martin's clean code architecture. Look at the problem as you were Unstoppable or EDGE and you don't control the dependency. Wrap around it and have a shim that you Do control and use that instead.

B. Pointfree.co / TCA approach.

Make proposal on the SDK Protocol. The SDK returns an implementation of it. Then use a protocol witness or a mock implementation on Zashi to provide the mocked version that you need.

A and B

Make a protocol on Zashi for what the app expects of a proposal. This propocol contains a throwing toProposal() function.
Make an Adapter of the SDK's Proposal type that conforms to Zashi's.
Make a Mock implementation for testing on Zashi that throws whenever toProposal() is invoked. If by mistake this ever happens and a mocked proposal is sent in production you can see the error and flag it as critical in Crash reporting platform.

Or better, the testing code is never available in outside of the testing target.

str4d and others added 6 commits March 6, 2024 02:53
- Returns `null` when there are no funds to shield or the shielding
  threshold is not met.
- Throws an exception if there are funds to shield in more than one
  transparent receiver within the account.
- Has an optional parameter for specifying which transparent receiver
  to shield funds from.

This commit only alters the API to support the above; the functional
changes require modifying the FFI and Rust backend, which will happen
in a separate commit.
@str4d str4d force-pushed the 1204-expose-proposals branch from 67d564f to f617f17 Compare March 6, 2024 02:54
@str4d
Copy link
Collaborator Author

str4d commented Mar 6, 2024

Force-pushed to clean up the commits.

Includes:
- Multi-step transaction proposals.
- Changes to support `Synchronizer.proposeShielding` API changes.
@LukasKorba
Copy link
Collaborator

LukasKorba commented Mar 6, 2024

I tested all 5 APIs while working on adoption of this PR (e9177a2) in Zashi. I can confirm

  • deprecated Send API still works ok
  • deprecated shileding API still works ok
  • new proposeTransfer API works ok
  • new proposeShielding API works ok
  • createProposedTransactions API works ok

@str4d
Copy link
Collaborator Author

str4d commented Mar 7, 2024

Before we merge this, I'm going to add a helper method to Proposal to expose the number of steps, so that callers can anticipate how many events they will get from the async stream returned from createProposedTransactions.

Copy link
Contributor

@nuttycom nuttycom left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

cursory utACK, modulo @pacu's comments

var iterator = transactions.makeIterator()
var submitFailed = false

return AsyncThrowingStream() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't understand exactly how AsyncThrowingStream works - is it something whereby each time you poll it it invokes this block?

Copy link
Collaborator

@LukasKorba LukasKorba Mar 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's async stream and there's a very specific way how to work with it. The stream emits values until it finishes (nil value is emitted). And a client awaits it until the nil is emitted (or an error in case of AsyncThrowingStream - throwing means this stream can emit an error, another stream is AsyncStream which ends only when nil is emitted).

So technically speaking, typically it's not known how many values are emitted, it usually even doesn't matter because the use of stream is implemented as

for try await value in stream { }

This for will end at some point in the future, the stream can emit 1, 5 or 5M values. The processing lives in an async context so it's not blocking.

Copy link
Contributor

@nuttycom nuttycom left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

re-utACK 72acbc6

@str4d str4d force-pushed the 1204-expose-proposals branch from 72acbc6 to 129ac43 Compare March 8, 2024 14:49
@str4d
Copy link
Collaborator Author

str4d commented Mar 8, 2024

Force-pushed to remove the 2.0.11 changelog bump, as @LukasKorba will make a few other release updates before the actual release commit.

@str4d str4d merged commit 7fcf1fa into main Mar 8, 2024
1 of 2 checks passed
@str4d str4d deleted the 1204-expose-proposals branch March 8, 2024 14:50
@true-jared true-jared removed the S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. label Mar 11, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Replace ZcashRustBackend.{createToAddress, shieldFunds} with proposal-based methods
6 participants