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

feat: persist UserStorage e2e content keys using an encrypted keyStore #5129

Draft
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

mirceanis
Copy link

@mirceanis mirceanis commented Jan 10, 2025

⚠️ Tentative PR for early feedback gathering

Explanation

The UserStorageController e2e encryption keys are derived from a storageKey that is specific to the user profile. The key derivation function used is scrypt, with parameters recommended for password inputs. This means that it's a very costly operation (on the order of seconds on a 2024 mobile device).
These derived keys are cached in memory for the lifetime of the controller instance, but a better approach would be to use a Key Store, to persist the derived keys in a safe manner. This would avoid the rerun of the costly key derivation operation on every app restart.

This set of changes also comes at a time when we're preparing for a multi-device/multi-srp user profile, so the implementation of the KeyStore relies on the new encryption capabilities of the message-signing-snap. The snap is already used for auth and is a key component of the UserStorageController, but we are introducing some defaults here that create even more coupling between the controller and the snap.
The proposed KeyStore implementation relies on the persistable state properties and the messagingSystem available to MM controllers, so externalizing this would create some overhead, but feedback is very welcome about the approach.

References

fixes #5128

Changelog

@metamask/profile-sync-controller

  • : KeyStore persistence for e2ee keys

Checklist

  • I've updated the test suite for new or updated code as appropriate
  • I've updated documentation (JSDoc, Markdown, etc.) for new or updated code as appropriate
  • I've highlighted breaking changes using the "BREAKING" category above as appropriate
  • I've prepared draft pull requests for clients and consumer packages to resolve any breaking changes

@mirceanis mirceanis added the team-identity Identity Team changes. https://github.com/orgs/MetaMask/teams/identity label Jan 10, 2025
Copy link

socket-security bot commented Jan 10, 2025

New dependencies detected. Learn more about Socket for GitHub ↗︎

Package New capabilities Transitives Size Publisher
npm/@noble/[email protected] None 0 859 kB paulmillr

View full report↗︎

@mirceanis mirceanis force-pushed the 5128-persist-user-storage-e2e-keys branch from be2f7ca to 62a9880 Compare January 10, 2025 13:58
Comment on lines +386 to +393
storeWrappedKey: (keyRef: string, wrappedKey: string): Promise<void> => {
return new Promise((resolve) => {
this.update((state) => {
state.encryptedContentKeys[keyRef] = wrappedKey;
resolve();
});
});
},
Copy link
Contributor

Choose a reason for hiding this comment

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

Do these need to be promises?

If we do want to make then pseudo promises, we can make the functions async and avoid return explicit promises.

// By using the `async` function syntax, the return type will be inferred as a promise.
storeWrappedKey: async (...): Promise<void> => {
  this.update(...)
}

@@ -54,12 +56,14 @@ class EncryptorDecryptor {
plaintext: string,
password: string,
nativeScryptCrypto?: NativeScrypt,
keyStore?: KeyStore,
Copy link
Contributor

Choose a reason for hiding this comment

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

long term (far future) - I don't really like how many layers we have to drill in this context (both for NativeScrypt, and for this).

We can think of some diff approach later.

const keyRef = `${hashedPassword}${bytesToHex(newSalt)}`;
if (keyStore) {
try {
const storedKey = await keyStore.loadKey(keyRef);
Copy link
Contributor

Choose a reason for hiding this comment

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

Oooh very cool, so this keyStore will be used to store the encrypted keys to be used by other potential platforms?

As we persist this on the extension, would we need a way for this to be accessible outside of the extension? Reason I'm asking is could we place this keystore into the snap storage? That way sites can access this from the snap rather than the extension?

nonce: bytesToBase64(nonce),
ephemPublicKey: bytesToBase64(ephemeralPublic),
ciphertext: bytesToBase64(encryptedMessage),
} as Eip1024EncryptedData;
Copy link
Contributor

Choose a reason for hiding this comment

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

What fields are missing for this to be be type asserted?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
team-identity Identity Team changes. https://github.com/orgs/MetaMask/teams/identity
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[Proposal] Persist UserStorageController e2ee content keys
2 participants