k3s/vendor/github.com/rancher/wrangler/pkg/controller-gen/main.go

387 lines
11 KiB
Go
Raw Normal View History

2019-05-09 22:03:45 +00:00
package controllergen
import (
"fmt"
2019-05-09 22:03:45 +00:00
"io"
"io/ioutil"
"os"
"path/filepath"
"sort"
"strings"
"k8s.io/gengo/generator"
2019-05-09 22:03:45 +00:00
cgargs "github.com/rancher/wrangler/pkg/controller-gen/args"
"github.com/rancher/wrangler/pkg/controller-gen/generators"
2019-12-12 01:27:03 +00:00
"github.com/sirupsen/logrus"
2019-05-09 22:03:45 +00:00
"k8s.io/apimachinery/pkg/runtime/schema"
csargs "k8s.io/code-generator/cmd/client-gen/args"
clientgenerators "k8s.io/code-generator/cmd/client-gen/generators"
cs "k8s.io/code-generator/cmd/client-gen/generators"
types2 "k8s.io/code-generator/cmd/client-gen/types"
dpargs "k8s.io/code-generator/cmd/deepcopy-gen/args"
infargs "k8s.io/code-generator/cmd/informer-gen/args"
inf "k8s.io/code-generator/cmd/informer-gen/generators"
lsargs "k8s.io/code-generator/cmd/lister-gen/args"
ls "k8s.io/code-generator/cmd/lister-gen/generators"
"k8s.io/gengo/args"
dp "k8s.io/gengo/examples/deepcopy-gen/generators"
"k8s.io/gengo/types"
)
2019-12-12 01:27:03 +00:00
var (
t = true
)
2019-05-09 22:03:45 +00:00
func Run(opts cgargs.Options) {
customArgs := &cgargs.CustomArgs{
Options: opts,
TypesByGroup: map[schema.GroupVersion][]*types.Name{},
Package: opts.OutputPackage,
}
genericArgs := args.Default().WithoutDefaultFlagParsing()
genericArgs.CustomArgs = customArgs
genericArgs.GoHeaderFilePath = opts.Boilerplate
genericArgs.InputDirs = parseTypes(customArgs)
if genericArgs.OutputBase == "./" { //go modules
tempDir, err := ioutil.TempDir("", "")
if err != nil {
return
}
genericArgs.OutputBase = tempDir
defer os.RemoveAll(tempDir)
}
customArgs.OutputBase = genericArgs.OutputBase
clientGen := generators.NewClientGenerator()
if err := genericArgs.Execute(
2020-03-26 21:07:15 +00:00
clientgenerators.NameSystems(nil),
2019-05-09 22:03:45 +00:00
clientgenerators.DefaultNameSystem(),
clientGen.Packages,
); err != nil {
logrus.Fatalf("Error: %v", err)
}
groups := map[string]bool{}
listerGroups := map[string]bool{}
informerGroups := map[string]bool{}
2019-12-12 01:27:03 +00:00
deepCopygroups := map[string]bool{}
2019-05-09 22:03:45 +00:00
for groupName, group := range customArgs.Options.Groups {
if group.GenerateTypes {
2019-12-12 01:27:03 +00:00
deepCopygroups[groupName] = true
}
if group.GenerateClients {
2019-05-09 22:03:45 +00:00
groups[groupName] = true
}
if group.GenerateListers {
listerGroups[groupName] = true
}
if group.GenerateInformers {
informerGroups[groupName] = true
}
2019-05-09 22:03:45 +00:00
}
if len(deepCopygroups) == 0 && len(groups) == 0 && len(listerGroups) == 0 && len(informerGroups) == 0 {
2019-05-09 22:03:45 +00:00
if err := copyGoPathToModules(customArgs); err != nil {
logrus.Fatalf("go modules copy failed: %v", err)
}
return
}
if err := copyGoPathToModules(customArgs); err != nil {
logrus.Fatalf("go modules copy failed: %v", err)
}
2019-12-12 01:27:03 +00:00
if err := generateDeepcopy(deepCopygroups, customArgs); err != nil {
2019-05-09 22:03:45 +00:00
logrus.Fatalf("deepcopy failed: %v", err)
}
if err := generateClientset(groups, customArgs); err != nil {
logrus.Fatalf("clientset failed: %v", err)
}
if err := generateListers(listerGroups, customArgs); err != nil {
2019-05-09 22:03:45 +00:00
logrus.Fatalf("listers failed: %v", err)
}
if err := generateInformers(informerGroups, customArgs); err != nil {
2019-05-09 22:03:45 +00:00
logrus.Fatalf("informers failed: %v", err)
}
if err := copyGoPathToModules(customArgs); err != nil {
logrus.Fatalf("go modules copy failed: %v", err)
}
}
func sourcePackagePath(customArgs *cgargs.CustomArgs, pkgName string) string {
pkgSplit := strings.Split(pkgName, string(os.PathSeparator))
pkg := filepath.Join(customArgs.OutputBase, strings.Join(pkgSplit[:3], string(os.PathSeparator)))
return pkg
}
//until k8s code-gen supports gopath
func copyGoPathToModules(customArgs *cgargs.CustomArgs) error {
pathsToCopy := map[string]bool{}
for _, types := range customArgs.TypesByGroup {
for _, names := range types {
pkg := sourcePackagePath(customArgs, names.Package)
pathsToCopy[pkg] = true
}
}
pkg := sourcePackagePath(customArgs, customArgs.Package)
pathsToCopy[pkg] = true
for pkg, _ := range pathsToCopy {
if _, err := os.Stat(pkg); os.IsNotExist(err) {
continue
}
return filepath.Walk(pkg, func(path string, info os.FileInfo, err error) error {
newPath := strings.Replace(path, pkg, ".", 1)
if info.IsDir() {
return os.MkdirAll(newPath, info.Mode())
2019-05-09 22:03:45 +00:00
}
return copyFile(path, newPath)
2019-05-09 22:03:45 +00:00
})
}
return nil
}
func copyFile(src, dst string) error {
var err error
var srcfd *os.File
var dstfd *os.File
var srcinfo os.FileInfo
if srcfd, err = os.Open(src); err != nil {
return err
}
defer srcfd.Close()
if dstfd, err = os.Create(dst); err != nil {
return err
}
defer dstfd.Close()
if _, err = io.Copy(dstfd, srcfd); err != nil {
return err
}
if srcinfo, err = os.Stat(src); err != nil {
return err
}
return os.Chmod(dst, srcinfo.Mode())
}
func generateDeepcopy(groups map[string]bool, customArgs *cgargs.CustomArgs) error {
if len(groups) == 0 {
return nil
}
2019-05-09 22:03:45 +00:00
deepCopyCustomArgs := &dpargs.CustomArgs{}
args := args.Default().WithoutDefaultFlagParsing()
args.CustomArgs = deepCopyCustomArgs
args.OutputBase = customArgs.OutputBase
args.OutputFileBaseName = "zz_generated_deepcopy"
args.GoHeaderFilePath = customArgs.Options.Boilerplate
for gv, names := range customArgs.TypesByGroup {
if !groups[gv.Group] {
continue
}
args.InputDirs = append(args.InputDirs, names[0].Package)
deepCopyCustomArgs.BoundingDirs = append(deepCopyCustomArgs.BoundingDirs, names[0].Package)
}
return args.Execute(dp.NameSystems(),
dp.DefaultNameSystem(),
dp.Packages)
}
func generateClientset(groups map[string]bool, customArgs *cgargs.CustomArgs) error {
if len(groups) == 0 {
return nil
}
2019-05-09 22:03:45 +00:00
args, clientSetArgs := csargs.NewDefaults()
clientSetArgs.ClientsetName = "versioned"
args.OutputBase = customArgs.OutputBase
args.OutputPackagePath = filepath.Join(customArgs.Package, "clientset")
args.GoHeaderFilePath = customArgs.Options.Boilerplate
var order []schema.GroupVersion
for gv := range customArgs.TypesByGroup {
if !groups[gv.Group] {
continue
}
order = append(order, gv)
}
sort.Slice(order, func(i, j int) bool {
return order[i].Group < order[j].Group
})
for _, gv := range order {
2019-12-12 01:27:03 +00:00
packageName := customArgs.Options.Groups[gv.Group].PackageName
if packageName == "" {
packageName = gv.Group
}
2019-05-09 22:03:45 +00:00
names := customArgs.TypesByGroup[gv]
args.InputDirs = append(args.InputDirs, names[0].Package)
clientSetArgs.Groups = append(clientSetArgs.Groups, types2.GroupVersions{
2019-12-12 01:27:03 +00:00
PackageName: packageName,
2019-05-09 22:03:45 +00:00
Group: types2.Group(gv.Group),
Versions: []types2.PackageVersion{
{
Version: types2.Version(gv.Version),
Package: names[0].Package,
},
},
})
}
2020-03-26 21:07:15 +00:00
return args.Execute(cs.NameSystems(nil),
2019-05-09 22:03:45 +00:00
cs.DefaultNameSystem(),
setGenClient(groups, customArgs.TypesByGroup, cs.Packages))
}
func setGenClient(groups map[string]bool, typesByGroup map[schema.GroupVersion][]*types.Name, f func(*generator.Context, *args.GeneratorArgs) generator.Packages) func(*generator.Context, *args.GeneratorArgs) generator.Packages {
return func(context *generator.Context, generatorArgs *args.GeneratorArgs) generator.Packages {
for gv, names := range typesByGroup {
if !groups[gv.Group] {
continue
}
for _, name := range names {
var (
p = context.Universe.Package(name.Package)
t = p.Type(name.Name)
status bool
nsed bool
kubebuilder bool
)
for _, line := range append(t.SecondClosestCommentLines, t.CommentLines...) {
switch {
case strings.Contains(line, "+kubebuilder:object:root=true"):
kubebuilder = true
t.SecondClosestCommentLines = append(t.SecondClosestCommentLines, "+genclient")
case strings.Contains(line, "+kubebuilder:subresource:status"):
status = true
case strings.Contains(line, "+kubebuilder:resource:") && strings.Contains(line, "scope=Namespaced"):
nsed = true
}
}
if kubebuilder {
if !nsed {
t.SecondClosestCommentLines = append(t.SecondClosestCommentLines, "+genclient:nonNamespaced")
}
if !status {
t.SecondClosestCommentLines = append(t.SecondClosestCommentLines, "+genclient:noStatus")
}
foundGroup := false
for _, comment := range p.DocComments {
if strings.Contains(comment, "+groupName=") {
foundGroup = true
break
}
}
if !foundGroup {
p.DocComments = append(p.DocComments, "+groupName="+gv.Group)
p.Comments = append(p.Comments, "+groupName="+gv.Group)
fmt.Println(gv.Group, p.DocComments, p.Comments, p.Path)
}
}
}
}
return f(context, generatorArgs)
}
2019-05-09 22:03:45 +00:00
}
func generateInformers(groups map[string]bool, customArgs *cgargs.CustomArgs) error {
if len(groups) == 0 {
return nil
}
2019-05-09 22:03:45 +00:00
args, clientSetArgs := infargs.NewDefaults()
clientSetArgs.VersionedClientSetPackage = filepath.Join(customArgs.Package, "clientset/versioned")
clientSetArgs.ListersPackage = filepath.Join(customArgs.Package, "listers")
args.OutputBase = customArgs.OutputBase
args.OutputPackagePath = filepath.Join(customArgs.Package, "informers")
args.GoHeaderFilePath = customArgs.Options.Boilerplate
for gv, names := range customArgs.TypesByGroup {
if !groups[gv.Group] {
continue
}
args.InputDirs = append(args.InputDirs, names[0].Package)
}
2020-03-26 21:07:15 +00:00
return args.Execute(inf.NameSystems(nil),
2019-05-09 22:03:45 +00:00
inf.DefaultNameSystem(),
setGenClient(groups, customArgs.TypesByGroup, inf.Packages))
2019-05-09 22:03:45 +00:00
}
func generateListers(groups map[string]bool, customArgs *cgargs.CustomArgs) error {
if len(groups) == 0 {
return nil
}
2019-05-09 22:03:45 +00:00
args, _ := lsargs.NewDefaults()
args.OutputBase = customArgs.OutputBase
args.OutputPackagePath = filepath.Join(customArgs.Package, "listers")
args.GoHeaderFilePath = customArgs.Options.Boilerplate
for gv, names := range customArgs.TypesByGroup {
if !groups[gv.Group] {
continue
}
args.InputDirs = append(args.InputDirs, names[0].Package)
}
2020-03-26 21:07:15 +00:00
return args.Execute(ls.NameSystems(nil),
2019-05-09 22:03:45 +00:00
ls.DefaultNameSystem(),
setGenClient(groups, customArgs.TypesByGroup, ls.Packages))
2019-05-09 22:03:45 +00:00
}
func parseTypes(customArgs *cgargs.CustomArgs) []string {
for groupName, group := range customArgs.Options.Groups {
2019-12-12 01:27:03 +00:00
if group.GenerateTypes || group.GenerateClients {
if group.InformersPackage == "" {
group.InformersPackage = filepath.Join(customArgs.Package, "informers/externalversions")
}
if group.ClientSetPackage == "" {
group.ClientSetPackage = filepath.Join(customArgs.Package, "clientset/versioned")
}
if group.ListersPackage == "" {
group.ListersPackage = filepath.Join(customArgs.Package, "listers")
}
2019-05-09 22:03:45 +00:00
customArgs.Options.Groups[groupName] = group
}
}
for groupName, group := range customArgs.Options.Groups {
if err := cgargs.ObjectsToGroupVersion(groupName, group.Types, customArgs.TypesByGroup); err != nil {
// sorry, should really handle this better
panic(err)
}
2019-05-09 22:03:45 +00:00
}
var inputDirs []string
for _, names := range customArgs.TypesByGroup {
inputDirs = append(inputDirs, names[0].Package)
}
return inputDirs
}