-
-
Notifications
You must be signed in to change notification settings - Fork 3.2k
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
fix: multisig stale data after failed refresh #9863
base: master
Are you sure you want to change the base?
fix: multisig stale data after failed refresh #9863
Conversation
f6096cb
to
10c6802
Compare
Edit: this should fix it. Revert previous commit that leads to failing functional test. If anyone knows if we really need this, please let me know how to use it.
|
10c6802
to
f492351
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This looks like an improvement to me as is. Nice work!
Would be nice to have @woodser test this out as well :)
Couple things to take into consideration
- If refresh fails while
m_multisig_rescan_k
andm_multisig_rescan_info
are still set, and the wallet then closes before refreshing again, it looks like that would resurface this bug.- One way to solve this would be to persist
m_multisig_rescan_k
andm_multisig_rescan_info
to the wallet cache.
- One way to solve this would be to persist
- If
m_multisig_rescan_k
andm_multisig_rescan_info
are still set after returning fromimport_multisig
(sincerefresh
fails to execute), wallets should probably be sure to complete a refresh before allowing the user to do any other multisig actions.- Looking at the multisig code, you'll see there is
get_multisig_status
which all multisig functions rely on. - We could add a boolean
needs_refresh
tomultisig_account_status
that gets set to true inget_multisig_status
ifm_multisig_rescan_k
andm_multisig_rescan_info
are non-empty. - Then all those functions that check
get_multisig_status
could also check for a falseyneeds_refresh
boolean where necessary.
- Looking at the multisig code, you'll see there is
I think this PR as is works as an improvement (because wallets should have UX that refreshes the wallet when it isn't synced or encourages refresh), and the above would be ok for a future PR.
f492351
to
c848e21
Compare
Thank you for the review. One thing I noticed, unrelated to this PR, I had the functional tests fail with |
I tried to reproduce again, and failed again. I was careful to follow exactly the steps you outlined. No idea why it works for me. Would you predict that the old code fails reliably, not only intermittently, or only rarely by sheer bad luck? Maybe my failure to reproduce is unimportant, and a total fluke, but still mildly worries me. Looking at your modifications, I understood technically what you did, but unfortunately I still don't understand yet what the core problem was, and why switching from pointers to some vectors to vectors as direct class members would change anything important. Can you maybe drop a few high-level hints? |
AFAIUI, in The problem is, if The change to class members was made to extend the scope, because
From all the testing I did I would say it fails reliably if you're quick enough to call
I assume if you have a fast wallet <-> daemon connection it may be harder to reproduce, because the wallet connects and uses auto-refresh. It seems |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, @SNeedlewoods, for the detailed explanation. Things make sense now.
I found that preventing refresh before the second import_multisig_info
(step 7) is indeed crucial with my setup to reproduce the problem. After I did a set auto-refresh 0
in the wallet as a step 0 I was finally successful.
So I could reproduce the error with the master code, could not reproduce it anymore with the code from this branch, and also reviewed the code.
Please don't wait for my signoff, as my one comment has been addressed, and I don't have time to review the PR in full. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A good long term fix would be to modify the m_transfers
in-place, instead of detaching the blockchain and requiring a live refresh, but the behavior introduced in this PR is significantly better than current. Good job!
It looks like the live refresh is there because it's calculating key images for receives, and then refreshing from the first point of a calculated key image to identify respective spends (without revealing to the daemon which key images are the user's). Sync via full balance view key would negate the need for the live refresh. With the legacy view key, another option to avoid revealing the user's key images to a daemon would for multisig wallets to use "background sync" to make sure the wallet picks up all receives and potential spends, and then upon calculating key images via |
Without this PR you should be able to reproduce the error with the following steps:
export_multisig_info
for both walletsimport_multisig_info
for W1import_multisig_info
for W2 (it fails as expected)import_multisig_info
for W2 (it looks like it worked, but it actually dropped enotes with partial key images fromm_transfers
, this becomes a problem e.g. if you try to sign a tx from this wallet)transfer
from W1sign_multisig
from W2, here we getError: Multisig error: This signature was made with stale data: export fresh multisig data, which other participants must then use