|
| 1 | +package remediation |
| 2 | + |
| 3 | +import ( |
| 4 | + "github.com/snyk/cli-extension-os-flows/internal/semver" |
| 5 | +) |
| 6 | + |
| 7 | +func FixesToRemediationSummary(findings []Finding) (Summary, error) { |
| 8 | + pins, err := calculatePins(findings) |
| 9 | + if err != nil { |
| 10 | + return Summary{}, err |
| 11 | + } |
| 12 | + upgrades, unresolved, err := calculateUpgrades(findings) |
| 13 | + if err != nil { |
| 14 | + return Summary{}, err |
| 15 | + } |
| 16 | + return Summary{Pins: pins, Upgrades: upgrades, Unresolved: unresolved}, err |
| 17 | +} |
| 18 | + |
| 19 | +func calculateUpgrades(findings []Finding) (upgrades []Upgrade, unresolved []VulnerabilityInPackage, err error) { |
| 20 | + for _, finding := range findings { |
| 21 | + switch fix := finding.Fix.(type) { |
| 22 | + case UpgradeFix: |
| 23 | + matchingPaths := getMatchingPaths(finding, fix) |
| 24 | + |
| 25 | + upgradablePaths := matchingPaths.upgradablePaths |
| 26 | + for i, _ := range upgradablePaths.upgradePath { |
| 27 | + vulnerabilityInPackage := newVulnerabilityInPackage(finding) |
| 28 | + vulnerabilityInPackage.IntroducedThrough = upgradablePaths.dependencyPath |
| 29 | + |
| 30 | + upgrades = append(upgrades, Upgrade{ |
| 31 | + From: upgradablePaths.dependencyPath[i][1], |
| 32 | + To: upgradablePaths.upgradePath[i][0], |
| 33 | + Fixes: []VulnerabilityInPackage{vulnerabilityInPackage}, |
| 34 | + }) |
| 35 | + } |
| 36 | + |
| 37 | + if len(matchingPaths.nonUpgradablePaths) > 0 { |
| 38 | + vulnerabilityInPackage := newVulnerabilityInPackage(finding) |
| 39 | + vulnerabilityInPackage.IntroducedThrough = matchingPaths.nonUpgradablePaths |
| 40 | + unresolved = append(unresolved, vulnerabilityInPackage) |
| 41 | + } |
| 42 | + default: |
| 43 | + continue |
| 44 | + } |
| 45 | + } |
| 46 | + return upgrades, unresolved, nil |
| 47 | +} |
| 48 | + |
| 49 | +/* |
| 50 | +* |
| 51 | +Returns equally ordered arrays of original dependencyPaths and their upgrade counterparts. |
| 52 | +
|
| 53 | +For those upgrades/dep paths that don't have a counterpart, no entries will be found. |
| 54 | +*/ |
| 55 | +type paths struct { |
| 56 | + upgradablePaths upgradablePathss |
| 57 | + nonUpgradablePaths []DependencyPath |
| 58 | +} |
| 59 | + |
| 60 | +type upgradablePathss struct { |
| 61 | + dependencyPath []DependencyPath |
| 62 | + upgradePath []DependencyPath |
| 63 | +} |
| 64 | + |
| 65 | +func getMatchingPaths(finding Finding, fix UpgradeFix) paths { |
| 66 | + var upgradePaths []DependencyPath |
| 67 | + var dependencyPaths []DependencyPath |
| 68 | + var nonUpgradablePaths []DependencyPath |
| 69 | + |
| 70 | + for _, depPath := range finding.DependencyPaths { |
| 71 | + var found = false |
| 72 | + for _, upath := range fix.UpgradeAction.UpgradePaths { |
| 73 | + if !isMatchingUpgradePath(upath, depPath) { |
| 74 | + continue |
| 75 | + } |
| 76 | + found = true |
| 77 | + upgradePaths = append(upgradePaths, upath) |
| 78 | + dependencyPaths = append(dependencyPaths, depPath) |
| 79 | + break |
| 80 | + } |
| 81 | + if !found { |
| 82 | + nonUpgradablePaths = append(nonUpgradablePaths, depPath) |
| 83 | + } |
| 84 | + } |
| 85 | + return paths{ |
| 86 | + upgradablePaths: upgradablePathss{ |
| 87 | + dependencyPath: dependencyPaths, |
| 88 | + upgradePath: upgradePaths, |
| 89 | + }, |
| 90 | + nonUpgradablePaths: nonUpgradablePaths, |
| 91 | + } |
| 92 | +} |
| 93 | + |
| 94 | +func calculatePins(findings []Finding) ([]Upgrade, error) { |
| 95 | + var output []Upgrade |
| 96 | + var pinMap = make(map[string]*[]*Upgrade) |
| 97 | + for _, finding := range findings { |
| 98 | + switch finding.Fix.(type) { |
| 99 | + case PinFix: |
| 100 | + vulnerablePackage := finding.Package |
| 101 | + key := finding.Package.Name |
| 102 | + |
| 103 | + // We'll be grouping by package name |
| 104 | + upgrade, upgradeExists := pinMap[key] |
| 105 | + |
| 106 | + if upgradeExists { |
| 107 | + highestVersion, err := getMaxVersion(finding.PackageManager, (*upgrade)[0].To.Version, finding.Fix.(PinFix).PinAction.Package.Version) |
| 108 | + if err != nil { |
| 109 | + return nil, err |
| 110 | + } |
| 111 | + |
| 112 | + // As we process individual pins, we'll get the highest version of a given package across vulns & package version |
| 113 | + // Goal is to obtain a single pin that fixes as many problems as possible |
| 114 | + versionAlreadyExists := false |
| 115 | + for _, u := range *upgrade { |
| 116 | + u.To.Version = highestVersion |
| 117 | + if u.From.Version == finding.Package.Version { |
| 118 | + u.Fixes = append(u.Fixes, newVulnerabilityInPackage(finding)) |
| 119 | + versionAlreadyExists = true |
| 120 | + break |
| 121 | + } |
| 122 | + } |
| 123 | + if !versionAlreadyExists { |
| 124 | + *upgrade = append(*upgrade, &Upgrade{ |
| 125 | + From: vulnerablePackage, |
| 126 | + To: finding.Fix.(PinFix).PinAction.Package, |
| 127 | + Fixes: []VulnerabilityInPackage{newVulnerabilityInPackage(finding)}, |
| 128 | + }) |
| 129 | + } |
| 130 | + } else { |
| 131 | + pinMap[key] = &[]*Upgrade{{ |
| 132 | + From: vulnerablePackage, |
| 133 | + To: finding.Fix.(PinFix).PinAction.Package, |
| 134 | + Fixes: []VulnerabilityInPackage{newVulnerabilityInPackage(finding)}, |
| 135 | + }} |
| 136 | + } |
| 137 | + |
| 138 | + default: |
| 139 | + continue |
| 140 | + } |
| 141 | + } |
| 142 | + |
| 143 | + for _, versionGroupedPins := range pinMap { |
| 144 | + for _, pin := range *versionGroupedPins { |
| 145 | + output = append(output, *pin) |
| 146 | + } |
| 147 | + } |
| 148 | + return output, nil |
| 149 | +} |
| 150 | + |
| 151 | +func newVulnerabilityInPackage(finding Finding) VulnerabilityInPackage { |
| 152 | + return VulnerabilityInPackage{ |
| 153 | + FixedInVersions: finding.FixedInVersions, |
| 154 | + VulnerablePackage: finding.Package, |
| 155 | + Vulnerability: finding.Vulnerability, |
| 156 | + IntroducedThrough: finding.DependencyPaths, |
| 157 | + } |
| 158 | +} |
| 159 | + |
| 160 | +func isMatchingUpgradePath(upath DependencyPath, depPath DependencyPath) bool { |
| 161 | + if len(depPath) != len(upath)+1 { |
| 162 | + return false |
| 163 | + } |
| 164 | + for i, uDep := range upath { |
| 165 | + if uDep.Name != depPath[i+1].Name { |
| 166 | + return false |
| 167 | + } |
| 168 | + } |
| 169 | + return true |
| 170 | +} |
| 171 | + |
| 172 | +func getMaxVersion(packageManager PackageManager, v1 string, v2 string) (string, error) { |
| 173 | + semverResolver, err := semver.GetSemver(string(packageManager)) |
| 174 | + if err != nil { |
| 175 | + return "", err |
| 176 | + } |
| 177 | + var version string |
| 178 | + compare, err := semverResolver.Compare(v1, v2) |
| 179 | + if err != nil { |
| 180 | + return "", err |
| 181 | + } |
| 182 | + if compare >= 0 { |
| 183 | + version = v1 |
| 184 | + } else { |
| 185 | + version = v2 |
| 186 | + } |
| 187 | + return version, nil |
| 188 | +} |
0 commit comments