| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | package crd | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 	"github.com/rancher/norman/store/proxy" | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	"github.com/rancher/norman/types" | 
					
						
							|  |  |  | 	"github.com/sirupsen/logrus" | 
					
						
							| 
									
										
										
										
											2020-03-05 21:29:41 -07:00
										 |  |  | 	"golang.org/x/sync/errgroup" | 
					
						
							| 
									
										
										
										
											2021-06-21 16:08:58 -04:00
										 |  |  | 	apiext "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1" | 
					
						
							| 
									
										
										
										
											2018-01-17 15:42:31 -07:00
										 |  |  | 	"k8s.io/apiextensions-apiserver/pkg/client/clientset/clientset" | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	"k8s.io/apimachinery/pkg/api/errors" | 
					
						
							|  |  |  | 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | 
					
						
							|  |  |  | 	"k8s.io/apimachinery/pkg/util/wait" | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 	"k8s.io/client-go/rest" | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Factory struct { | 
					
						
							| 
									
										
										
										
											2020-03-05 21:29:41 -07:00
										 |  |  | 	eg           errgroup.Group | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 	ClientGetter proxy.ClientGetter | 
					
						
							| 
									
										
										
										
											2018-01-17 15:42:31 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | func NewFactoryFromClientGetter(clientGetter proxy.ClientGetter) *Factory { | 
					
						
							|  |  |  | 	return &Factory{ | 
					
						
							|  |  |  | 		ClientGetter: clientGetter, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NewFactoryFromClient(config rest.Config) (*Factory, error) { | 
					
						
							|  |  |  | 	getter, err := proxy.NewClientGetterFromConfig(config) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return &Factory{ | 
					
						
							|  |  |  | 		ClientGetter: getter, | 
					
						
							|  |  |  | 	}, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 21:29:41 -07:00
										 |  |  | func (f *Factory) BatchWait() error { | 
					
						
							|  |  |  | 	return f.eg.Wait() | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 14:02:30 -07:00
										 |  |  | func (f *Factory) BatchCreateCRDs(ctx context.Context, storageContext types.StorageContext, typer proxy.StoreTyper, schemas *types.Schemas, version *types.APIVersion, schemaIDs ...string) { | 
					
						
							| 
									
										
										
										
											2020-03-05 21:29:41 -07:00
										 |  |  | 	f.eg.Go(func() error { | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 		var schemasToCreate []*types.Schema | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for _, schemaID := range schemaIDs { | 
					
						
							|  |  |  | 			s := schemas.Schema(version, schemaID) | 
					
						
							|  |  |  | 			if s == nil { | 
					
						
							| 
									
										
										
										
											2020-03-05 21:29:41 -07:00
										 |  |  | 				return fmt.Errorf("can not find schema %s", schemaID) | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			schemasToCreate = append(schemasToCreate, s) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 14:02:30 -07:00
										 |  |  | 		err := f.AssignStores(ctx, storageContext, typer, schemasToCreate...) | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2020-03-05 21:29:41 -07:00
										 |  |  | 			return fmt.Errorf("creating CRD store %v", err) | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-03-05 21:29:41 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 14:02:30 -07:00
										 |  |  | func (f *Factory) AssignStores(ctx context.Context, storageContext types.StorageContext, typer proxy.StoreTyper, schemas ...*types.Schema) error { | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 	schemaStatus, err := f.CreateCRDs(ctx, storageContext, schemas...) | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2018-01-17 15:42:31 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 	for _, schema := range schemas { | 
					
						
							|  |  |  | 		crd, ok := schemaStatus[schema] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							|  |  |  | 			return fmt.Errorf("failed to create create/find CRD for %s", schema.ID) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 		schema.Store = proxy.NewProxyStore(ctx, f.ClientGetter, | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 			storageContext, | 
					
						
							| 
									
										
										
										
											2021-09-21 14:02:30 -07:00
										 |  |  | 			typer, | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 			[]string{"apis"}, | 
					
						
							|  |  |  | 			crd.Spec.Group, | 
					
						
							| 
									
										
										
										
											2021-06-21 16:08:58 -04:00
										 |  |  | 			// Even if CRD is created as v1beta1, it's served as v1 with a single element in Versions | 
					
						
							|  |  |  | 			crd.Spec.Versions[0].Name, | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 			crd.Status.AcceptedNames.Kind, | 
					
						
							|  |  |  | 			crd.Status.AcceptedNames.Plural) | 
					
						
							| 
									
										
										
										
											2018-01-17 15:42:31 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | func (f *Factory) CreateCRDs(ctx context.Context, storageContext types.StorageContext, schemas ...*types.Schema) (map[*types.Schema]*apiext.CustomResourceDefinition, error) { | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	schemaStatus := map[*types.Schema]*apiext.CustomResourceDefinition{} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 	apiClient, err := f.ClientGetter.APIExtClient(nil, storageContext) | 
					
						
							| 
									
										
										
										
											2018-02-09 13:31:12 -07:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 13:07:32 -07:00
										 |  |  | 	ready, err := f.getReadyCRDs(ctx, apiClient) | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, schema := range schemas { | 
					
						
							| 
									
										
										
										
											2020-03-26 13:07:32 -07:00
										 |  |  | 		crd, err := f.createCRD(ctx, apiClient, schema, ready) | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		schemaStatus[schema] = crd | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 13:07:32 -07:00
										 |  |  | 	ready, err = f.getReadyCRDs(ctx, apiClient) | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for schema, crd := range schemaStatus { | 
					
						
							|  |  |  | 		if readyCrd, ok := ready[crd.Name]; ok { | 
					
						
							|  |  |  | 			schemaStatus[schema] = readyCrd | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | 			if err := f.waitCRD(ctx, apiClient, crd.Name, schema, schemaStatus); err != nil { | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return schemaStatus, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 11:26:12 -07:00
										 |  |  | func (f *Factory) waitCRD(ctx context.Context, apiClient clientset.Interface, crdName string, schema *types.Schema, schemaStatus map[*types.Schema]*apiext.CustomResourceDefinition) error { | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	logrus.Infof("Waiting for CRD %s to become available", crdName) | 
					
						
							|  |  |  | 	defer logrus.Infof("Done waiting for CRD %s to become available", crdName) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	first := true | 
					
						
							| 
									
										
										
										
											2023-08-31 12:19:28 +05:30
										 |  |  | 	return wait.PollUntilContextTimeout(ctx, 500*time.Millisecond, 60*time.Second, false, func(ctx context.Context) (bool, error) { | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 		if !first { | 
					
						
							|  |  |  | 			logrus.Infof("Waiting for CRD %s to become available", crdName) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		first = false | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 16:08:58 -04:00
										 |  |  | 		crd, err := apiClient.ApiextensionsV1().CustomResourceDefinitions().Get(ctx, crdName, metav1.GetOptions{}) | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return false, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for _, cond := range crd.Status.Conditions { | 
					
						
							|  |  |  | 			switch cond.Type { | 
					
						
							|  |  |  | 			case apiext.Established: | 
					
						
							|  |  |  | 				if cond.Status == apiext.ConditionTrue { | 
					
						
							|  |  |  | 					schemaStatus[schema] = crd | 
					
						
							|  |  |  | 					return true, err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			case apiext.NamesAccepted: | 
					
						
							|  |  |  | 				if cond.Status == apiext.ConditionFalse { | 
					
						
							|  |  |  | 					logrus.Infof("Name conflict on %s: %v\n", crdName, cond.Reason) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return false, ctx.Err() | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 13:07:32 -07:00
										 |  |  | func (f *Factory) createCRD(ctx context.Context, apiClient clientset.Interface, schema *types.Schema, ready map[string]*apiext.CustomResourceDefinition) (*apiext.CustomResourceDefinition, error) { | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	plural := strings.ToLower(schema.PluralName) | 
					
						
							|  |  |  | 	name := strings.ToLower(plural + "." + schema.Version.Group) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	crd, ok := ready[name] | 
					
						
							|  |  |  | 	if ok { | 
					
						
							|  |  |  | 		return crd, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	crd = &apiext.CustomResourceDefinition{ | 
					
						
							|  |  |  | 		ObjectMeta: metav1.ObjectMeta{ | 
					
						
							|  |  |  | 			Name: name, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		Spec: apiext.CustomResourceDefinitionSpec{ | 
					
						
							| 
									
										
										
										
											2021-06-21 16:08:58 -04:00
										 |  |  | 			Group: schema.Version.Group, | 
					
						
							|  |  |  | 			Versions: []apiext.CustomResourceDefinitionVersion{ | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Name:    schema.Version.Version, | 
					
						
							|  |  |  | 					Served:  true, | 
					
						
							|  |  |  | 					Storage: true, | 
					
						
							|  |  |  | 					// Using catch-all schema as it's required in v1 and we do not have enough info in `schema` | 
					
						
							|  |  |  | 					// Note catch-all schema used in Wrangler (open schema for "spec" and "status") is not good enough | 
					
						
							|  |  |  | 					// here as Norman CRDs often define direct fields | 
					
						
							|  |  |  | 					Schema: &apiext.CustomResourceValidation{ | 
					
						
							|  |  |  | 						OpenAPIV3Schema: &apiext.JSONSchemaProps{ | 
					
						
							|  |  |  | 							Type:                   "object", | 
					
						
							|  |  |  | 							XPreserveUnknownFields: &[]bool{true}[0], | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 			Names: apiext.CustomResourceDefinitionNames{ | 
					
						
							|  |  |  | 				Plural: plural, | 
					
						
							| 
									
										
										
										
											2020-05-15 17:38:28 -07:00
										 |  |  | 				Kind:   schema.CodeName, | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if schema.Scope == types.NamespaceScope { | 
					
						
							|  |  |  | 		crd.Spec.Scope = apiext.NamespaceScoped | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		crd.Spec.Scope = apiext.ClusterScoped | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	logrus.Infof("Creating CRD %s", name) | 
					
						
							| 
									
										
										
										
											2021-06-21 16:08:58 -04:00
										 |  |  | 	crd2, err := apiClient.ApiextensionsV1().CustomResourceDefinitions().Create(ctx, crd, metav1.CreateOptions{}) | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	if errors.IsAlreadyExists(err) { | 
					
						
							|  |  |  | 		return crd, nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-30 20:35:31 +09:00
										 |  |  | 	return crd2, err | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-26 13:07:32 -07:00
										 |  |  | func (f *Factory) getReadyCRDs(ctx context.Context, apiClient clientset.Interface) (map[string]*apiext.CustomResourceDefinition, error) { | 
					
						
							| 
									
										
										
										
											2021-06-21 16:08:58 -04:00
										 |  |  | 	list, err := apiClient.ApiextensionsV1().CustomResourceDefinitions().List(ctx, metav1.ListOptions{}) | 
					
						
							| 
									
										
										
										
											2017-12-22 23:13:59 -07:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	result := map[string]*apiext.CustomResourceDefinition{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i, crd := range list.Items { | 
					
						
							|  |  |  | 		for _, cond := range crd.Status.Conditions { | 
					
						
							|  |  |  | 			switch cond.Type { | 
					
						
							|  |  |  | 			case apiext.Established: | 
					
						
							|  |  |  | 				if cond.Status == apiext.ConditionTrue { | 
					
						
							|  |  |  | 					result[crd.Name] = &list.Items[i] | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return result, nil | 
					
						
							|  |  |  | } |