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

Resetting cross-signing leaves account in a broken state if you abort halfway #13338

Open
bwindels opened this issue Apr 23, 2020 · 18 comments
Open
Labels
A-E2EE-Cross-Signing A-E2EE-Key-Backup O-Uncommon Most users are unlikely to come across this or unexpected workflow S-Critical Prevents work, causes data loss and/or has no workaround T-Defect

Comments

@bwindels
Copy link
Contributor

bwindels commented Apr 23, 2020

  • Open Security & Privacy settings
  • Under "Cross-signing", click "Reset"
  • Click "Clear cross-signing keys"
  • Click "Reset all"
  • Click "Reset"
  • Choose "Enter a security phase", click "continue", enter a phrase, click "continue", confirm, click "continue"
    OR: Choose "Generate a Security Key", click "continue", copy the generated key
  • Click "continue"
  • Click "X", or the background, or refresh
  • Attempt to log in again
  • Click "Verify with Security Key or Phrase"
  • Enter new key
  • Click "continue"

Expected: new device is verified
Actual: back to the "Verify this device" screen.


Original report While testing retrying account data requests during cross-signing reset, I managed to get reset cross-signing in a completely broken state. Even on a fresh login with that account, I can't reset the cross-signing keys anymore.

During testing, I purposefully made set account data requests fail, and I refreshed while not having completed the reset operation. In any case, refreshing the app, or having requests fail should not leave you in a state where cross-signing is just completely broken and there is no way to fix it. I just get this dialog when trying:

image

When trying to verify somebody in this state, I get a passphrase dialog, and when entering my passphrase nothing happens with the following error in the console:

CrossSigning.js?c1e7:122 Uncaught (in promise) Error: Key type master from getCrossSigningKey callback did not match
    at CrossSigningInfo.getCrossSigningKey (CrossSigning.js?c1e7:122)
    at async Crypto.checkOwnCrossSigningTrust (index.js?89dd:1314)
    at async Crypto.bootstrapSecretStorage (index.js?89dd:719)
    at async accessSecretStorage (CrossSigningManager.js?bc8a:238)
    at async enable4SIfNeeded (verification.js?ee94:35)
    at async verifyUser (verification.js?ee94:117)

The security settings show this:
image
image

This was on a local synapse that I last updated on Monday Apr 20, last commit f5ea8b48bd57c19dae126a5cc631dc79cf5ce332.

This is the account data of the broken account:

{
  "m.secret_storage.key.NqawOlJcxXqeVRWYcgV9h6SMb89TNOe0": {
    "type": "m.secret_storage.key.NqawOlJcxXqeVRWYcgV9h6SMb89TNOe0",
    "content": {
      "algorithm": "m.secret_storage.v1.curve25519-aes-sha2",
      "passphrase": {
        "algorithm": "m.pbkdf2",
        "iterations": 500000,
        "salt": "y0UEjkoLynttHNQAAStmEe5aQKmit0IW"
      },
      "pubkey": "R9kN1CKC6uo8PicfPl7l+PmvxJRirLLVzDmBR3/hd08",
      "signatures": {
        "@bruno-cs2:localhost": {
          "ed25519:j6vzDKIjYJFWxeWVIOBYi3Z0NZYpchgzrbx0vKYOWIA": "9bnkJVfnoM6OeVdXBWo+TX36Q8lAm7Y6ynOfAqPuv46L/sYlzknUT20fazyPyHsehxKHQdQk618YOcfP49YFBw"
        }
      }
    },
    "unsigned": {}
  },
  "m.secret_storage.default_key": {
    "type": "m.secret_storage.default_key",
    "content": {
      "key": "UDEgkBfKmKP9IZIgeYkb3GqcpSqSDlUz"
    },
    "unsigned": {}
  },
  "m.secret_storage.key.UDEgkBfKmKP9IZIgeYkb3GqcpSqSDlUz": {
    "type": "m.secret_storage.key.UDEgkBfKmKP9IZIgeYkb3GqcpSqSDlUz",
    "content": {
      "algorithm": "m.secret_storage.v1.aes-hmac-sha2",
      "passphrase": {
        "algorithm": "m.pbkdf2",
        "iterations": 500000,
        "salt": "y0UEjkoLynttHNQAAStmEe5aQKmit0IW"
      },
      "iv": "RWyi8OHhl+0YdWAc1c1d3g==",
      "mac": "g5tEycHZgFMRCA5n2v5dgKeGQDHD0XoKeCKmDBOa1nc=",
      "signatures": {
        "@bruno-cs2:localhost": {
          "ed25519:j6vzDKIjYJFWxeWVIOBYi3Z0NZYpchgzrbx0vKYOWIA": "uDTSYBEsBGk7FFDg3oMz4r2tQ7gheBCIohdYBeKH5jfvyph7t0edTtrZlG3jHywFE58XS+liFqYRQ0uWHJBBDA",
          "ed25519:J+RSIygjIzFbtvmrnJ6hplo/wtbd/gVJsw0Bg24B/Kc": "92dB/bLhnrgzK7rsYh+5fh8m55CGGVmvReLSKftXs/UMXRfVNh/BsnujGhKJgkH+j+lem838rlmlsZC3gphaDA",
          "ed25519:o6KENMS74YIKmQdyFuRkxMYBhxgvnyn1FScHAEuCuSk": "p7SAlqZC+BvCzSIMODWr1DhtTGedln7pLorqqJQnDlD9x7nNtagEtYMDs8B+lmNPtZ23cw5huE0QpRSnQJ2ZCA",
          "ed25519:mmjnpW61nos/UMdbnB5xDA4LYq3uWhG2UWm8sTwzmKs": "BfQwWAcwh5hqF10aylQXyMDqFdAVR6VAnlBwgmlCz+JCoqaiBC42k4AEXAuaQJLmKlFNkCT+/D+a7HLLBfJTBA"
        }
      }
    },
    "unsigned": {}
  },
  "m.cross_signing.master": {
    "type": "m.cross_signing.master",
    "content": {
      "encrypted": {
        "UDEgkBfKmKP9IZIgeYkb3GqcpSqSDlUz": {
          "iv": "eA48yeLZnb474LvIrsPWyg==",
          "ciphertext": "Inep5o8uPSJ3JSQtdvn5llt8LHQb7EJBg50gCbbMCjZ3mM5hDzjGyqBIme4=",
          "mac": "qQyC1CIs/LBfsPS/UrhNf5Wt8F+aCxkgiO1v+tOoLVM="
        }
      }
    },
    "unsigned": {}
  },
  "m.cross_signing.self_signing": {
    "type": "m.cross_signing.self_signing",
    "content": {
      "encrypted": {
        "UDEgkBfKmKP9IZIgeYkb3GqcpSqSDlUz": {
          "iv": "9/rBfZ2ulK0tkhDmiR4IkQ==",
          "ciphertext": "BCWiQJK/IThOHvokatv3P7A6/D78uHb5UOKBykZkFiDjluixuOCLIzDWBE8=",
          "mac": "ny7Fdim7aqAkdJG95Ks4z3PPygoa3dkf3lIbFneaLhE="
        }
      }
    },
    "unsigned": {}
  },
  "m.cross_signing.user_signing": {
    "type": "m.cross_signing.user_signing",
    "content": {
      "encrypted": {
        "UDEgkBfKmKP9IZIgeYkb3GqcpSqSDlUz": {
          "iv": "GQU2r9LDQAIXTr3wxSzo8Q==",
          "ciphertext": "+EGISntCnPi9PFrjrDGKxElkKDH3G8sPzl0QuK0r+VqexWOcB/9s4AfXDjI=",
          "mac": "976KULDLCXApHmCjUocdKANzt2G8NhEWSmoOALxvGJY="
        }
      }
    },
    "unsigned": {}
  },
  "m.megolm_backup.v1": {
    "type": "m.megolm_backup.v1",
    "content": {
      "encrypted": {
        "UDEgkBfKmKP9IZIgeYkb3GqcpSqSDlUz": {
          "iv": "Qa/pR5CxKEUfupnCQYBipA==",
          "ciphertext": "zviX2Yf4nAs5lN34ZjsJwmGEgGZqF13GY8YW4vJfhEQ0cGH4AP1PofpU+bc=",
          "mac": "0/TASoY80XD9zT5kkdNo2jukQ8lA+6msUedjldVhReQ="
        }
      }
    },
    "unsigned": {}
  },
}

@jryans: Edited to remove unrelated account data

@bwindels bwindels changed the title resetting cross-signing can get completely broken if you abort a halfway failed attempt Resetting cross-signing can get completely broken if you abort a halfway failed attempt Apr 23, 2020
@jryans
Copy link
Collaborator

jryans commented Apr 23, 2020

Agreed this is not good, as reset should always work. It is a bit artificial, but still seems like it could happen in reality too if the connection drops out. Nice to have a fix for release if possible, but not a blocker.

@superboum
Copy link

I encountered this error on my production instance. What should I do now?

@jluebbe
Copy link

jluebbe commented May 5, 2020

I seem to have triggered this issue on riot.im. Is there any way to recover?

@dbkr dbkr self-assigned this May 5, 2020
@dbkr
Copy link
Member

dbkr commented May 5, 2020

Looks like the problem here is that if you have a key backup,CreateSecretStorageDialog will try to migrate it even if it's supposed to be force-resetting.

@dbkr
Copy link
Member

dbkr commented May 5, 2020

No wait, it skips straight to PHASE_PASSPHRASE if the force prop is true.

@dbkr
Copy link
Member

dbkr commented May 5, 2020

I tried to repro this by blocking the upload of the master key and then refreshing while it spins retrying to upload it, but resetting still works fine after refreshing and unblocking.

I also can't see any way the bootstrap function would print, "Cross-signing private keys found in secret storage" when the reset button was pressed: it should never enter that block. Were you definitely pressing the bootstrap button rather than the reset button?

Anyone else, if you think you're hitting this, please submit your logs (settings -> help & about -> submit debug logs).

@superboum

This comment has been minimized.

@dbkr

This comment has been minimized.

@superboum

This comment has been minimized.

@bwindels
Copy link
Contributor Author

Looks like the problem here is that if you have a key backup,CreateSecretStorageDialog will try to migrate it even if it's supposed to be force-resetting.

I linked another rageshake that looks to be an instance of CreateSecretStorageDialog, we seem to decide that there is a key backup we should migrate the passphrase from (not entirely sure yet how and where that decision is made though) and then when we actually look for the megolm backup key, it doesn't seem to be there and throws an exception.

@jryans jryans added phase:4 and removed phase:3 labels May 13, 2020
@jryans
Copy link
Collaborator

jryans commented May 14, 2020

In my own testing, I was able to create a situation like this by doing:

  1. Change setAccountData to throw an error
  2. Bootstrap cross-signing for a fresh account
  3. Sign out and back in so there's no locally cached cross-signing state, and skip verification during sign in
  4. Try to bootstrap in Settings

In this path, we'll enter checkOwnCrossSigningTrust from bootstrap, which tries to get the master cross-signing private key from 4S account data, but it won't match the newer public key we successfully uploaded to the server, so we get a semi-cryptic error:

2020-05-14 at 18 03

In such a situation, we need to reset cross-signing anyway so that our public and private keys are both the same "generation" and back in sync. If you unblock account data, resetting appears to work, so people shouldn't be stuck in such a situation, as they can still reset.

@jryans
Copy link
Collaborator

jryans commented May 14, 2020

I linked another rageshake that looks to be an instance of CreateSecretStorageDialog, we seem to decide that there is a key backup we should migrate the passphrase from (not entirely sure yet how and where that decision is made though) and then when we actually look for the megolm backup key, it doesn't seem to be there and throws an exception.

I'll continue chasing this theory tomorrow.

@bwindels
Copy link
Contributor Author

Right, that makes sense @jryans, and should be somewhat helped with the robustness work. I'm not sure why it would have ended up calling checkOwnCrossSigningTrust during the bootstrap, I'd expect you to end up in the first case because setupNewSecretStorage is true when triggered from a reset.

@bwindels
Copy link
Contributor Author

There also seems to be another reset blocking issue that seems related to key backup, where you get this error:

2020-05-08T16:10:24.368Z I restoreWithCachedKey failed: Couldn't get key
Error: Couldn't get key
    at I.restoreKeyBackupWithCache (vector://vector/webapp/bundles/9f769ac7a648e2ec9507/vendors~init.js:2:552106)
    at async hc._restoreWithCachedKey (vector://vector/webapp/bundles/9f769ac7a648e2ec9507/vendors~init.js:2:4016156)
    at async hc._loadBackupStatus (vector://vector/webapp/bundles/9f769ac7a648e2ec9507/vendors~init.js:2:4016575)

Probably best to break out in a separate issue.

@jryans
Copy link
Collaborator

jryans commented May 18, 2020

I'm not sure why it would have ended up calling checkOwnCrossSigningTrust during the bootstrap, I'd expect you to end up in the first case because setupNewSecretStorage is true when triggered from a reset.

Ah, I meant that pressing the green bootstrap (so not a reset) descends down to checkOwnCrossSigningTrust and hits the error in the screenshot above: we have something in 4S already (it's just wrong generation of keys), so currently we go to the Cross-signing private keys found in secret storage branch and then into checkOwnCrossSigningTrust.

@jryans
Copy link
Collaborator

jryans commented May 18, 2020

There also seems to be another reset blocking issue that seems related to key backup, where you get this error:

2020-05-08T16:10:24.368Z I restoreWithCachedKey failed: Couldn't get key
Error: Couldn't get key
    at I.restoreKeyBackupWithCache (vector://vector/webapp/bundles/9f769ac7a648e2ec9507/vendors~init.js:2:552106)
    at async hc._restoreWithCachedKey (vector://vector/webapp/bundles/9f769ac7a648e2ec9507/vendors~init.js:2:4016156)
    at async hc._loadBackupStatus (vector://vector/webapp/bundles/9f769ac7a648e2ec9507/vendors~init.js:2:4016575)

Probably best to break out in a separate issue.

Yes, let's break this out to a separate issue. I think that (somewhat confusing) log message mainly indicates we just don't have key backup key cached, so we'd need to read it from 4S instead.

@jryans
Copy link
Collaborator

jryans commented May 18, 2020

Overall, I haven't been able to isolate actionable steps here, and my assumption is that @bwindels bootstrapping robustness will help with much / all of these issues.

For now, I'm returning this to the backlog, and we can re-test again once more robustness work has merged.

@jryans jryans removed their assignment May 18, 2020
@MadLittleMods MadLittleMods added S-Critical Prevents work, causes data loss and/or has no workaround O-Uncommon Most users are unlikely to come across this or unexpected workflow A-E2EE-Key-Backup labels May 18, 2022
@Johennes Johennes removed the Z-Fire label Oct 2, 2023
@richvdh
Copy link
Member

richvdh commented Oct 8, 2024

This is still very much a thing; I just closed #26322 as a duplicate

@richvdh richvdh changed the title Resetting cross-signing can get completely broken if you abort a halfway failed attempt Resetting cross-signing leaves account in a broken state if you abort halfway Dec 2, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-E2EE-Cross-Signing A-E2EE-Key-Backup O-Uncommon Most users are unlikely to come across this or unexpected workflow S-Critical Prevents work, causes data loss and/or has no workaround T-Defect
Projects
None yet
Development

No branches or pull requests

9 participants