From 01c78144e55e62c11f816ebe28cc87e5221fd17c Mon Sep 17 00:00:00 2001 From: walkowif <59475134+walkowif@users.noreply.github.com> Date: Wed, 13 Dec 2023 15:22:35 +0100 Subject: [PATCH] Log only unique set of missing packages (#15) --- cmd/construct.go | 80 ++++++++++++++++++++++++++++++++++++------- cmd/construct_test.go | 9 ++++- cmd/root.go | 16 ++++----- cmd/structs.go | 5 +++ cmd/testdata/PACKAGES | 2 ++ 5 files changed, 91 insertions(+), 21 deletions(-) diff --git a/cmd/construct.go b/cmd/construct.go index eb00f1f..7b945c2 100644 --- a/cmd/construct.go +++ b/cmd/construct.go @@ -20,14 +20,16 @@ import ( "strings" ) +const lowestPossiblePackageVersion = "0.0.0.0" + // ConstructOutputPackageList generates a list of all packages and their dependencies // which should be included in the output renv.lock file, // based on the list of package descriptions, and information contained in the PACKAGES files. func ConstructOutputPackageList(packages []PackageDescription, packagesFiles map[string]PackagesFile, repositoryList []string, allowedMissingDependencyTypes []string) []PackageDescription { var outputPackageList []PackageDescription - var fatalErrors string - var nonFatalErrors string + fatalMissingPackageVersions := make(map[string]DependencyVersion) + nonFatalMissingPackageVersions := make(map[string]DependencyVersion) // Add all input packages to output list, as the packages should be downloaded from git repositories. for _, p := range packages { outputPackageList = append(outputPackageList, PackageDescription{ @@ -46,17 +48,38 @@ func ConstructOutputPackageList(packages []PackageDescription, packagesFiles map ResolveDependenciesRecursively( &outputPackageList, d.DependencyName, d.VersionOperator, d.VersionValue, d.DependencyType, allowedMissingDependencyTypes, - repositoryList, packagesFiles, 1, &fatalErrors, &nonFatalErrors, + repositoryList, packagesFiles, 1, fatalMissingPackageVersions, + nonFatalMissingPackageVersions, ) } } } } - if fatalErrors != "" { - log.Fatal(fatalErrors) + errorsString := "Packages not found in any repository:\n" + for packageName, versionConstraint := range nonFatalMissingPackageVersions { + var versionConstraintString string + if versionConstraint.VersionValue != lowestPossiblePackageVersion { + versionConstraintString = (" " + + versionConstraint.VersionOperator + " " + + versionConstraint.VersionValue) + } + errorsString += packageName + versionConstraintString + "\n" + } + if len(errorsString) > len("Packages not found in any repository:\n") { + log.Error(errorsString) } - if nonFatalErrors != "" { - log.Error(nonFatalErrors) + errorsString = "Packages not found in any repository:\n" + for packageName, versionConstraint := range fatalMissingPackageVersions { + var versionConstraintString string + if versionConstraint.VersionValue != lowestPossiblePackageVersion { + versionConstraintString = (" " + + versionConstraint.VersionOperator + " " + + versionConstraint.VersionValue) + } + errorsString += packageName + versionConstraintString + "\n" + } + if len(errorsString) > len("Packages not found in any repository:\n") { + log.Fatal(errorsString) } return outputPackageList } @@ -68,7 +91,8 @@ func ConstructOutputPackageList(packages []PackageDescription, packagesFiles map func ResolveDependenciesRecursively(outputList *[]PackageDescription, name string, versionOperator string, versionValue string, dependencyType string, allowedMissingDependencyTypes []string, repositoryList []string, packagesFiles map[string]PackagesFile, recursionLevel int, - fatalErrors *string, nonFatalErrors *string) { + fatalMissingPackageVersions map[string]DependencyVersion, + nonFatalMissingPackageVersions map[string]DependencyVersion) { var indentation string for i := 0; i < recursionLevel; i++ { indentation += " " @@ -110,7 +134,8 @@ func ResolveDependenciesRecursively(outputList *[]PackageDescription, name strin ResolveDependenciesRecursively( outputList, d.DependencyName, d.VersionOperator, d.VersionValue, d.DependencyType, allowedMissingDependencyTypes, repositoryList, - packagesFiles, recursionLevel+1, fatalErrors, nonFatalErrors, + packagesFiles, recursionLevel+1, fatalMissingPackageVersions, + nonFatalMissingPackageVersions, ) } } @@ -120,17 +145,48 @@ func ResolveDependenciesRecursively(outputList *[]PackageDescription, name strin } } } + ProcessMissingPackage( + indentation, name, versionOperator, versionValue, dependencyType, + allowedMissingDependencyTypes, fatalMissingPackageVersions, + nonFatalMissingPackageVersions, + ) +} + +// ProcessMissingPackage saves information about missing packages (dependencies) and their versions. +// This information is later reported to the user, together with optionally exiting the application +// with failed status, depending on the types of missing dependencies and the configuration +// provided by --allowIncompleteRenvLock flag. +func ProcessMissingPackage(indentation string, packageName string, versionOperator string, + versionValue string, dependencyType string, allowedMissingDependencyTypes []string, + fatalMissingPackageVersions map[string]DependencyVersion, + nonFatalMissingPackageVersions map[string]DependencyVersion) { var versionConstraint string if versionOperator != "" && versionValue != "" { versionConstraint = " (version " + versionOperator + " " + versionValue + ")" + } else { + versionOperator = ">=" + versionValue = lowestPossiblePackageVersion } - message := "Could not find package " + name + versionConstraint + " in any of the repositories.\n" + message := "Could not find package " + packageName + versionConstraint + " in any of the repositories.\n" if stringInSlice(dependencyType, allowedMissingDependencyTypes) { log.Warn(indentation + message) - *nonFatalErrors += message + val, ok := nonFatalMissingPackageVersions[packageName] + if !ok || (ok && !CheckIfVersionSufficient(val.VersionValue, versionOperator, versionValue)) { + // This is the first time we see this package as a missing dependency, or + // some other package already requires this missing dependency in a lower version + // and the currently processed package requires it in a higher version, + // so the current requirement is more important. + nonFatalMissingPackageVersions[packageName] = DependencyVersion{versionOperator, versionValue} + log.Trace("Adding package ", packageName, " ", versionOperator, " ", versionValue, " to missing packages list.") + } } else { log.Error(indentation + message) - *fatalErrors += message + val, ok := fatalMissingPackageVersions[packageName] + if !ok || (ok && !CheckIfVersionSufficient(val.VersionValue, versionOperator, versionValue)) { + // See a comment above for explanation of this condition. + fatalMissingPackageVersions[packageName] = DependencyVersion{versionOperator, versionValue} + log.Trace("Adding package ", packageName, " ", versionOperator, " ", versionValue, " to missing packages list.") + } } } diff --git a/cmd/construct_test.go b/cmd/construct_test.go index e76c86c..188c44a 100644 --- a/cmd/construct_test.go +++ b/cmd/construct_test.go @@ -332,6 +332,12 @@ func Test_ConstructOutputPackageList(t *testing.T) { "", "", }, + { + "LinkingTo", + "nonExistentPackage2", + ">=", + "1.0.0", + }, }, "", "", "", "", "", "", "", }, @@ -389,7 +395,8 @@ func Test_ConstructOutputPackageList(t *testing.T) { }, packagesFiles, repositoryList, // Let the generation of renv.lock proceed, despite 'nonExistentPackage' - // (dependency type LinkingTo) not being found in any repository. + // and 'nonExistentPackage2' (dependency type LinkingTo) not being found + // in any repository. []string{"LinkingTo"}, ) assert.Equal(t, outputPackageList, diff --git a/cmd/root.go b/cmd/root.go index 6282e90..90169a9 100644 --- a/cmd/root.go +++ b/cmd/root.go @@ -106,21 +106,21 @@ in an renv.lock-compatible file.`, writeJSON(outputRenvLock, renvLock) }, } - rootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", + rootCmd.PersistentFlags().StringVarP(&cfgFile, "config", "c", "", "config file (default is $HOME/.locksmith.yaml)") - rootCmd.PersistentFlags().StringVar(&logLevel, "logLevel", "info", + rootCmd.PersistentFlags().StringVarP(&logLevel, "logLevel", "l", "info", "Logging level (trace, debug, info, warn, error). ") - rootCmd.PersistentFlags().StringVar(&inputPackageList, "inputPackageList", "", + rootCmd.PersistentFlags().StringVarP(&inputPackageList, "inputPackageList", "p", "", "Comma-separated list of URLs for raw DESCRIPTION files in git repositories for input packages.") - rootCmd.PersistentFlags().StringVar(&inputRepositoryList, "inputRepositoryList", "", + rootCmd.PersistentFlags().StringVarP(&inputRepositoryList, "inputRepositoryList", "r", "", "Comma-separated list of package repositories URLs, sorted according to their priorities (descending).") - rootCmd.PersistentFlags().StringVar(&gitHubToken, "gitHubToken", "", + rootCmd.PersistentFlags().StringVarP(&gitHubToken, "gitHubToken", "t", "", "Token to download non-public files from GitHub.") - rootCmd.PersistentFlags().StringVar(&gitLabToken, "gitLabToken", "", + rootCmd.PersistentFlags().StringVarP(&gitLabToken, "gitLabToken", "g", "", "Token to download non-public files from GitLab.") - rootCmd.PersistentFlags().StringVar(&outputRenvLock, "outputRenvLock", "renv.lock", + rootCmd.PersistentFlags().StringVarP(&outputRenvLock, "outputRenvLock", "o", "renv.lock", "File name to save the output renv.lock file.") - rootCmd.PersistentFlags().StringVar(&allowIncompleteRenvLock, "allowIncompleteRenvLock", "", + rootCmd.PersistentFlags().StringVarP(&allowIncompleteRenvLock, "allowIncompleteRenvLock", "i", "", "Locksmith will fail if any of dependencies of input packages cannot be found in the repositories. "+ "However, it will not fail for comma-separated dependency types listed in this argument, e.g.: "+ "'Imports,Depends,Suggests,LinkingTo'") diff --git a/cmd/structs.go b/cmd/structs.go index 01865ee..14be0f6 100644 --- a/cmd/structs.go +++ b/cmd/structs.go @@ -97,3 +97,8 @@ type Dependency struct { VersionOperator string `json:"operator"` VersionValue string `json:"value"` } + +type DependencyVersion struct { + VersionOperator string `json:"operator"` + VersionValue string `json:"value"` +} diff --git a/cmd/testdata/PACKAGES b/cmd/testdata/PACKAGES index ac01d4b..5f56be7 100644 --- a/cmd/testdata/PACKAGES +++ b/cmd/testdata/PACKAGES @@ -13,6 +13,8 @@ License: GPL-3 MD5sum: bbb222333444555666 NeedsCompilation: no + + Package: skippedPackage Version: 5.0.0 Depends: R (>= 3.6.0)