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

Show Infobar when vulnerabilities are found after restore #5258

Merged
merged 21 commits into from
Aug 17, 2023

Conversation

martinrrm
Copy link
Contributor

@martinrrm martinrrm commented Jun 19, 2023

Bug

Fixes: NuGet/Home#12398
Telemetry PR: https://github.com/NuGet/Client.Engineering/pull/2395

Regression? Last working version:

Description

When vulnerabilities are found after doing a restore, display an InfoBar in SolutionExplorer that leads to the PM UI.

vulnerabilities-infobar

PR Checklist

  • PR has a meaningful title

  • PR has a linked issue.

  • Described changes

  • Tests

    • Automated tests added
    • OR
    • Test exception - I'll add this to the Vendors test manual
    • OR
    • N/A
  • Documentation

    • Documentation PR or issue filled - Telemetry Assessment PR Missing
    • OR
    • N/A

Copy link
Member

@nkolev92 nkolev92 left a comment

Choose a reason for hiding this comment

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

I just got as far as the NuGet.SolutionRestoreManager.

The new project ref will cause an RPS regression.

NuGet.SolutionRestoreManager should not get any additional dependencies.

@martinrrm martinrrm requested a review from nkolev92 June 19, 2023 19:10
Copy link
Contributor

@donnie-msft donnie-msft left a comment

Choose a reason for hiding this comment

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

Ideally, if we can invoke a command we already have setup, I think that would be a good approach to launching PM UI. If you think of something besides copying the launch logic, then propose that as well!

Basically, the ActionClicked in my mind could just use the OleMenuCommandService to fire off the GUID for either Project/Solution level PM UI, which will launch the window/controls.
https://github.com/NuGet/NuGet.Client/blob/dev/src/NuGet.Clients/NuGet.Tools/NuGetPackage.cs#L323

I'd have your InfoBarService take a dependency on IMenuCommandService, then use it in the action handler. I'm just guessing it'll work, haven't tried it myself.

@ghost ghost added the Status:No recent activity PRs that have not had any recent activity and will be closed if the label is not removed label Jun 27, 2023
@ghost
Copy link

ghost commented Jun 27, 2023

This PR has been automatically marked as stale because it has no activity for 7 days. It will be closed if no further activity occurs within another 7 days of this comment. If it is closed, you may reopen it anytime when you're ready again, as long as you don't delete the branch.

@ghost ghost closed this Jul 5, 2023
@martinrrm martinrrm reopened this Jul 10, 2023
@ghost ghost removed the Status:No recent activity PRs that have not had any recent activity and will be closed if the label is not removed label Jul 10, 2023
@martinrrm
Copy link
Contributor Author

@nkolev92 @donnie-msft @zivkan I tried another approach to show the PM UI to avoid an RPS regression. I did an extensive investigation on how to open the PM UI using a command or something else that did not involve copying code from NuGetPackage.cs but as far as I know, the way this package is initialized make us impossible to reuse that code.

I left this as a draft because I want more feedback before I go and add tests, finish edge cases. 0

@martinrrm
Copy link
Contributor Author

@zivkan
Copy link
Member

zivkan commented Jul 13, 2023

I think it should avoid an RPS regression now. While IPMUIStarter is implemented in an assembly that NuGet.SolutionRestoreManager.dll doesn't load, since it's MEF imported with a Lazy<T> wrapper, this means that I don't believe it'll be loaded until the Lazy<T> is evaluated, which is only on the click event. As long as none of the RPS tests click the button, then it won't cause the second dll to be loaded. LGTM 👍

Copy link
Member

@nkolev92 nkolev92 left a comment

Choose a reason for hiding this comment

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

I'm guessing you have some unpushed changes based on earlier conversations

@martinrrm martinrrm requested a review from nkolev92 July 17, 2023 21:17
@martinrrm martinrrm marked this pull request as ready for review July 17, 2023 21:23
@martinrrm martinrrm requested a review from a team as a code owner July 17, 2023 21:23
@martinrrm martinrrm force-pushed the dev-martinrrm-infobar-in-solution-explorer branch from ca87b85 to 9c390f5 Compare July 17, 2023 21:25
@martinrrm martinrrm changed the title Info bar when vulnerabilities are found in restore Show Infobar when vulnerabilities are found after restore Jul 17, 2023
Copy link
Contributor

@kartheekp-ms kartheekp-ms left a comment

Choose a reason for hiding this comment

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

I am still reviewing this PR but left few comments to begin with.

@martinrrm martinrrm force-pushed the dev-martinrrm-infobar-in-solution-explorer branch from 5dcd5ce to 375b263 Compare August 14, 2023 22:34
@martinrrm martinrrm requested a review from nkolev92 August 15, 2023 00:12
nkolev92
nkolev92 previously approved these changes Aug 16, 2023
aortiz-msft
aortiz-msft previously approved these changes Aug 16, 2023
Copy link
Contributor

@aortiz-msft aortiz-msft left a comment

Choose a reason for hiding this comment

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

:shipit:

{
public interface IVulnerabilitiesFoundService
{
Task ReportVulnerabilities(bool hasVulnerabilitiesInSolution, CancellationToken cancellationToken);
Copy link
Contributor

@jebriede jebriede Aug 16, 2023

Choose a reason for hiding this comment

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

This method signature is very limiting. It only allows us to report true or false for this interface that tells if there are vulnerabilities in the solution or not. The variable is even called has vulnerabilities "in solution" which is very specific to where the vulnerabilities are found. Overall this interface has a lot of implementation details in its signature.

Instead, we could have a method ReportVulnerabilitiesAsync that accepts a list of vulnerabilities from the caller and the implementation decides what to do with it. In the implementation you need for the InfoBar, it would just count the list and if it's greater than 0 show the message that there are vulnerabilities in the solution.

We should either change the interface to be less implementation-specific or remove the interface altogether and reference the concrete class directly.

Copy link
Member

Choose a reason for hiding this comment

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

Instead, we could have a method ReportVulnerabilitiesAsync that accepts a list of vulnerabilities from the caller and the implementation decides what to do with it. In the implementation you need for the InfoBar, it would just count the list and if it's greater than 0 show the message that there are vulnerabilities in the solution.

I'm not sure I agree. I see that as over optimization.
No need to allocate extra collections if no one is gonna make use of that.
This method is easy to change if there's a better use for it in the future.

Copy link
Contributor

@jebriede jebriede Aug 16, 2023

Choose a reason for hiding this comment

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

Looking more closely, I see you don't actually have the list of vulnerabilities from the restore, but rather a list restore summaries and their errors. So then what we're really reporting is on specific restore errors that are occurring which in this usage happens to be ones that occur when the restore audit finds a vulnerability. The name of this interface and its method should reflect its usage then. We are providing a way for restore to report on restore errors, not report vulnerabilities.

Copy link
Member

Choose a reason for hiding this comment

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

We are providing a way for restore to report on restore errors, not report vulnerabilities

No, restore is reporting that vulnerabilities have been found in any project within the solution. (Probably how Martin arrived at the interface name: IVulnerabilitiesFoundService)
Vulnerabilities are not necessarily errors.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

The purpose of this interface is to "notify" the InfoBar that there are vulnerabilities, which as Nikolche said, are not necessarily errors.

I think that changing this Interface to a more generic one, like report all restore errors, is going to involve a lot of changes. We probably already have "components" that react to restore errors and are not centralized in one Interface.

Copy link
Contributor

@jebriede jebriede Aug 16, 2023

Choose a reason for hiding this comment

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

No, restore is reporting that vulnerabilities have been found in any project within the solution.

This is an implementation detail, not an interface specification. The SolutionRestoreJob should not be responsible for deciding what should be surfaced to the user through the UI. We're overloading the responsibilities of a restore job to also decide what is important to show to users. If this ever changes, and we want to report other errors / warnings / results, even if to a different place, we then have to change the SolutionRestoreJob again which violates the open-closed principle.

Copy link
Contributor

@jebriede jebriede Aug 16, 2023

Choose a reason for hiding this comment

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

@martinrrm feel free to merge the PR on the basis of others' approvals. Please do not consider this a blocker, although I disagree with this design and cannot in good conscience approve the PR, I do not believe it should block the PR either. The rest LGTM.

Copy link
Member

@nkolev92 nkolev92 Aug 16, 2023

Choose a reason for hiding this comment

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

The RestoreSummary.Errors is a poor name, but it's documented that it's warnings and errors,

/// All the warnings and errors that were produced as a result of the restore.
. Unfortunately we can't fix that due to that library being a part of https://github.com/NuGet/NuGet.Client/blob/dev/docs/nuget-sdk.md.
Fortunately for us, NuGet.SolutionRestoreManager isn't part of that SDK

This is an implementation detail, not an interface specification. The SolutionRestoreJob should not be responsible for deciding what should be surfaced to the user through the UI. We're overloading the responsibilities of a restore job to also decide what is important to show to users

I don't think that's true. SolutionRestoreJob is not deciding what's being surfaced. It's notifying via an interface that there are restore time vulnerability warnings or errors. The implementer is deciding to surface an info bar.

Software is going to change. Trying to predict that change is impossible.

Copy link
Contributor

@jebriede jebriede Aug 16, 2023

Choose a reason for hiding this comment

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

I don't think that's true. SolutionRestoreJob is not deciding what's being surfaced.

Of all the errors and warnings (let's call them Codes) the SolutionRestoreJob can encounter, it is deciding that the presence of vulnerabilities is the only thing that matters to surface. While it does not make any claims on how that should be presented, it does decide that no other information from the restore is going to be surfaced. Any changes to this in the future require a change to the SolutionRestoreJob and potentially the addition of yet another ISomethingNotificationService or a change to the IVulnerabilitiesNotificationService. If instead the interface method accepted the Codes, the implementations could decide what is of importance and the SolutionRestoreJob doesn't have to care what is determined as being important, it just does its job and allows the implementation of the notification service to decide what it will present. Of course this design also allows for composite implementations that each handle different Codes and we get that flexibility for free.

Happy to chat about design principles more offline. For the purposes of this PR, I think we are in agreement that it can be merged without this blocking, provided it gets sufficient approvals.

Copy link
Member

Choose a reason for hiding this comment

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

Of all the errors and warnings (let's call them Codes) the SolutionRestoreJob can encounter, it is deciding that the presence of vulnerabilities is the only thing that matters to surface

I think this perspective is one where reasonable people would easily disagree :)

Why does the restore job have to alert anyone of any error codes? After all the assets file is the final output for the operation. Why not read the assets file as it's the final source of truth anyways?

jeffkl
jeffkl previously approved these changes Aug 16, 2023
@martinrrm martinrrm requested a review from jebriede August 16, 2023 19:02
@martinrrm martinrrm dismissed jebriede’s stale review August 16, 2023 20:39

He agreed on merging the PR without his approval at this time.

@martinrrm martinrrm merged commit 4d815bf into dev Aug 17, 2023
@martinrrm martinrrm deleted the dev-martinrrm-infobar-in-solution-explorer branch August 17, 2023 04:18
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.

Show an infobar in Solution Explorer for any detected security vulnerabilities in a project or solution
8 participants