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

[PM-9111] Extension: persist add/edit form #12236

Open
wants to merge 23 commits into
base: main
Choose a base branch
from

Conversation

nick-livefront
Copy link
Collaborator

@nick-livefront nick-livefront commented Dec 4, 2024

🎟️ Tracking

PM-9111
Server PR (add feature flag - no logic)

📔 Objective

Use ViewCacheService to store the cipher that is being added/edited to repopulate the form when the extension loses focus and regains focus.

  • This stores the cipher in the signal as it is updated in the form.
    • The cache is discarded when a navigation occurs which keeps users from seeing stale values.
    • CipherFormCacheService is an in-between consuming components and the ViewCacheService
  • The main logic occurs in CipherFormComponent:
    • The updatedCipherView is set to the cached cipher if it is available.
    • getInitialCipherView allows each of the child components to get the cached cipher to populate their own forms with either the cached cipher or the updatedCipherView. This is a change from using originalCipherView
    • Cloning also needed a flag to check so - Clone is only appended on the initial load of the cloned cipher.
  • Both the PM9111ExtensionPersistAddEditForm and PersistPopupView feature flag should be enabled to use.
`PopupViewCacheService.formGroup`
  • PopupViewCacheService has a formGroup method that will store/create a formGroup directly and update the value as the form updates. This didn't work for the cipher form as each child component registers their own individual form as they are initialized in registerChildForm. I wasn't able to successfully get this to work with the dynamic & nested form.

  • Along the same lines, trying to store each child form in PopupViewCacheService.formGroup would require a cache entry for each individual form group alongside repopulating which strays from how the cipher form has a singular form.

  • Initialization with PopupViewCacheService.formGroup didn't support FormArrays we definitely could overcome that with some logic change

📸 Screenshots

Adding Ciphers

Login Card Identity Note
new-login-cached.mov
new-card-cached.mov
new-identity-cached.mov
new-note-cached.mov

Other scenarios

Editing a cipher Cloning a cipher
edit-cipher.mov
cloned-cipher.mov

🦮 Reviewer guidelines

  • 👍 (:+1:) or similar for great changes
  • 📝 (:memo:) or ℹ️ (:information_source:) for notes or general info
  • ❓ (:question:) for questions
  • 🤔 (:thinking:) or 💭 (:thought_balloon:) for more open inquiry that's not quite a confirmed issue and could potentially benefit from discussion
  • 🎨 (:art:) for suggestions / improvements
  • ❌ (:x:) or ⚠️ (:warning:) for more significant problems or concerns needing attention
  • 🌱 (:seedling:) or ♻️ (:recycle:) for future improvements or indications of technical debt
  • ⛏ (:pick:) for minor or nitpick changes

Copy link
Contributor

github-actions bot commented Dec 4, 2024

Logo
Checkmarx One – Scan Summary & Details21e943dc-108d-44fd-9653-701daeaf4336

Great job, no security vulnerabilities found in this Pull Request

Copy link

codecov bot commented Dec 4, 2024

Codecov Report

Attention: Patch coverage is 62.50000% with 33 lines in your changes missing coverage. Please review.

Project coverage is 34.29%. Comparing base (553d20f) to head (e8188f8).

✅ All tests successful. No failed tests found.

Files with missing lines Patch % Lines
...rc/cipher-form/components/cipher-form.component.ts 5.88% 16 Missing ⚠️
libs/vault/src/cipher-form/cipher-form.stories.ts 0.00% 7 Missing ⚠️
...her-form/components/identity/identity.component.ts 0.00% 4 Missing ⚠️
...nts/item-details/item-details-section.component.ts 69.23% 0 Missing and 4 partials ⚠️
...rm/services/popup-view-cache-background.service.ts 0.00% 2 Missing ⚠️
Additional details and impacted files
@@           Coverage Diff           @@
##             main   #12236   +/-   ##
=======================================
  Coverage   34.28%   34.29%           
=======================================
  Files        2935     2936    +1     
  Lines       90229    90291   +62     
  Branches    16943    16957   +14     
=======================================
+ Hits        30937    30965   +28     
- Misses      56828    56860   +32     
- Partials     2464     2466    +2     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

Copy link
Member

@shane-melton shane-melton left a comment

Choose a reason for hiding this comment

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

I think this approach worked out nicely, great work! I just noticed a few issues and had a few small suggestions.

name: name ? name : (this.initialValues?.name ?? ""),
organizationId: prefillCipher.organizationId, // We do not allow changing ownership of an existing cipher.
folderId: folderId ? folderId : (this.initialValues?.folderId ?? null),
collectionIds: prefillCipher.collectionIds,
Copy link
Member

Choose a reason for hiding this comment

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

⚠️ The collectionIds form control is actually the type SelectItemView[] so this will cause collectionsIds to have a map of undefined values (due to the missing .id property that is referenced in the itemDetailsForm.valueChanges subscription.

We can keep this with an empty [] and let the updateCollectionOptions() call down below (which you already updated).

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Updated! 7ef3a7b

this.initialValues?.collectionIds ??
(this.originalCipherView.collectionIds as CollectionId[]),
);
const prefillCollections = prefillCipher.collectionIds?.length
Copy link
Member

Choose a reason for hiding this comment

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

⚠️ I noticed when running this locally, the call above to itemDetailsForm.setValue() will actually update the reference to prefillCipher resulting in the loss of the prefillCipher.collectionIds value. If we pull out the collectionIds at the top of this method (with name and folderId) that should get us an unchanging copy of the ids.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Nice! Catch! This scenario totally slipped by me.

Updated! 7ef3a7b

return cachedCipherView;
}

return this.updatedCipherView;
Copy link
Member

Choose a reason for hiding this comment

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

⚠️ I think this should be this.originalCipherView?

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

This one is tripping me but I agree with you. I recall switching from originalCipherView to updatedCipherView during development but I can't for the life of me remember why.

  1. I should write better commit messages
  2. I'm betting it was necessary for one of my earlier solutions that didn't pan out. After testing originalCipherView works just fine and makes more sense. We're rolling.

77c0c19

}

// Create a new shallow reference to force the signal to update
this.cipherCache.set({ ...cipherView } as CipherView);
Copy link
Member

Choose a reason for hiding this comment

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

💭 I'm slightly worried this may not be a "deep" enough / separate clone.

For example the item details and auto fill option child forms update the form values during initialization that ends up making changes to the updatedCipherView which we're caching here which in turn may be affecting subsequent child forms during initialization.

I did see the item details section having an impact which I described in my other comment about pulling the collectionIds reference out. However, when I tried stepping through the auto fill options form with multiple URIs everything appeared to be working, which makes me suspicious that we may just be getting "lucky".

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

I did some digging, Signals use Object.is by default, any new object would fail that equality check update the signal value. Docs.

I think your concern is legitimate and I can't really find an answer to it. If we have a deeply nested object and there is a comparison done against and updated value would it still hold the same reference.

Do you think it's worth creating a deep clone for it? structuredClone is available but the browser support may be too narrow for our liking.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

For now, I added a comment about the default equality check but I'll update depending on which you think is a good path forward.

5c0a399

Copy link
Member

Choose a reason for hiding this comment

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

💭 We should double check that this behaves as expected in the Web client that uses a NoopViewCacheService.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

I haven't seen any preserved values or anything unexpected in the web 🚀

shane-melton
shane-melton previously approved these changes Jan 9, 2025
Copy link
Member

@shane-melton shane-melton left a comment

Choose a reason for hiding this comment

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

This is looking great!

I would also suggest we have QA run through this before we merge unless they or you think otherwise. My thinking is, while the bulk of the functionality is behind a feature flag, we are still making some adjustments to the initialization logic for the child form components.

Also apologies for the delay of this review!

Jingo88
Jingo88 previously approved these changes Jan 10, 2025
@nick-livefront nick-livefront dismissed stale reviews from Jingo88 and shane-melton via e8188f8 January 14, 2025 20:59
@nick-livefront
Copy link
Collaborator Author

@Jingo88 The Autofill on page load setting was being overwrote because of the URIs were first being added to the form. When the URIs were adding the value of on the form was still null causing the existing cipher to then have a null value on the next calls.

I added emitEvent false when adding existing URIs to stop this behavior. The valueChanges will be emitted when the next call to patch the existing autofill on load value. af9dc2e

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.

3 participants