mirror of
https://github.com/containers/podman.git
synced 2025-11-30 18:18:18 +08:00
pkg/libartifact has been moved to common and as such needs to be removed from podman and the new common vendored in along with required deps. https://issues.redhat.com/browse/RUN-3618 Signed-off-by: Brent Baude <bbaude@redhat.com>
134 lines
3.1 KiB
Go
134 lines
3.1 KiB
Go
// Copyright 2018 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package packages
|
|
|
|
import (
|
|
"cmp"
|
|
"fmt"
|
|
"iter"
|
|
"os"
|
|
"slices"
|
|
)
|
|
|
|
// Visit visits all the packages in the import graph whose roots are
|
|
// pkgs, calling the optional pre function the first time each package
|
|
// is encountered (preorder), and the optional post function after a
|
|
// package's dependencies have been visited (postorder).
|
|
// The boolean result of pre(pkg) determines whether
|
|
// the imports of package pkg are visited.
|
|
//
|
|
// Example:
|
|
//
|
|
// pkgs, err := Load(...)
|
|
// if err != nil { ... }
|
|
// Visit(pkgs, nil, func(pkg *Package) {
|
|
// log.Println(pkg)
|
|
// })
|
|
//
|
|
// In most cases, it is more convenient to use [Postorder]:
|
|
//
|
|
// for pkg := range Postorder(pkgs) {
|
|
// log.Println(pkg)
|
|
// }
|
|
func Visit(pkgs []*Package, pre func(*Package) bool, post func(*Package)) {
|
|
seen := make(map[*Package]bool)
|
|
var visit func(*Package)
|
|
visit = func(pkg *Package) {
|
|
if !seen[pkg] {
|
|
seen[pkg] = true
|
|
|
|
if pre == nil || pre(pkg) {
|
|
for _, imp := range sorted(pkg.Imports) { // for determinism
|
|
visit(imp)
|
|
}
|
|
}
|
|
|
|
if post != nil {
|
|
post(pkg)
|
|
}
|
|
}
|
|
}
|
|
for _, pkg := range pkgs {
|
|
visit(pkg)
|
|
}
|
|
}
|
|
|
|
// PrintErrors prints to os.Stderr the accumulated errors of all
|
|
// packages in the import graph rooted at pkgs, dependencies first.
|
|
// PrintErrors returns the number of errors printed.
|
|
func PrintErrors(pkgs []*Package) int {
|
|
var n int
|
|
errModules := make(map[*Module]bool)
|
|
for pkg := range Postorder(pkgs) {
|
|
for _, err := range pkg.Errors {
|
|
fmt.Fprintln(os.Stderr, err)
|
|
n++
|
|
}
|
|
|
|
// Print pkg.Module.Error once if present.
|
|
mod := pkg.Module
|
|
if mod != nil && mod.Error != nil && !errModules[mod] {
|
|
errModules[mod] = true
|
|
fmt.Fprintln(os.Stderr, mod.Error.Err)
|
|
n++
|
|
}
|
|
}
|
|
return n
|
|
}
|
|
|
|
// Postorder returns an iterator over the the packages in
|
|
// the import graph whose roots are pkg.
|
|
// Packages are enumerated in dependencies-first order.
|
|
func Postorder(pkgs []*Package) iter.Seq[*Package] {
|
|
return func(yield func(*Package) bool) {
|
|
seen := make(map[*Package]bool)
|
|
var visit func(*Package) bool
|
|
visit = func(pkg *Package) bool {
|
|
if !seen[pkg] {
|
|
seen[pkg] = true
|
|
for _, imp := range sorted(pkg.Imports) { // for determinism
|
|
if !visit(imp) {
|
|
return false
|
|
}
|
|
}
|
|
if !yield(pkg) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
for _, pkg := range pkgs {
|
|
if !visit(pkg) {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// -- copied from golang.org.x/tools/gopls/internal/util/moremaps --
|
|
|
|
// sorted returns an iterator over the entries of m in key order.
|
|
func sorted[M ~map[K]V, K cmp.Ordered, V any](m M) iter.Seq2[K, V] {
|
|
// TODO(adonovan): use maps.Sorted if proposal #68598 is accepted.
|
|
return func(yield func(K, V) bool) {
|
|
keys := keySlice(m)
|
|
slices.Sort(keys)
|
|
for _, k := range keys {
|
|
if !yield(k, m[k]) {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// KeySlice returns the keys of the map M, like slices.Collect(maps.Keys(m)).
|
|
func keySlice[M ~map[K]V, K comparable, V any](m M) []K {
|
|
r := make([]K, 0, len(m))
|
|
for k := range m {
|
|
r = append(r, k)
|
|
}
|
|
return r
|
|
}
|