Merge pull request #2 from mumoshu/runnerdeployment
feat: RunnerDeployments
This commit is contained in:
		
						commit
						0edf0d59f7
					
				|  | @ -107,7 +107,7 @@ type RunnerSet struct { | ||||||
| type RunnerSetSpec struct { | type RunnerSetSpec struct { | ||||||
| 	Replicas *int `json:"replicas"` | 	Replicas *int `json:"replicas"` | ||||||
| 
 | 
 | ||||||
| 	Template RunnerSpec `json:"template"` | 	Template RunnerTemplate `json:"template"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type RunnerSetStatus struct { | type RunnerSetStatus struct { | ||||||
|  | @ -115,6 +115,12 @@ type RunnerSetStatus struct { | ||||||
| 	ReadyReplicas     int `json:"readyReplicas"` | 	ReadyReplicas     int `json:"readyReplicas"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | type RunnerTemplate struct { | ||||||
|  | 	metav1.ObjectMeta `json:"metadata,omitempty"` | ||||||
|  | 
 | ||||||
|  | 	Spec RunnerSpec `json:"spec,omitempty"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // +kubebuilder:object:root=true
 | // +kubebuilder:object:root=true
 | ||||||
| 
 | 
 | ||||||
| // RunnerList contains a list of Runner
 | // RunnerList contains a list of Runner
 | ||||||
|  | @ -124,6 +130,42 @@ type RunnerSetList struct { | ||||||
| 	Items           []RunnerSet `json:"items"` | 	Items           []RunnerSet `json:"items"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func init() { | // +kubebuilder:object:root=true
 | ||||||
| 	SchemeBuilder.Register(&Runner{}, &RunnerList{}, &RunnerSet{}, &RunnerSetList{}) | // +kubebuilder:subresource:status
 | ||||||
|  | // +kubebuilder:printcolumn:JSONPath=".spec.replicas",name=Desired,type=number
 | ||||||
|  | // +kubebuilder:printcolumn:JSONPath=".status.availableReplicas",name=Current,type=number
 | ||||||
|  | // +kubebuilder:printcolumn:JSONPath=".status.readyReplicas",name=Ready,type=number
 | ||||||
|  | 
 | ||||||
|  | // RunnerSet is the Schema for the runnersets API
 | ||||||
|  | type RunnerDeployment struct { | ||||||
|  | 	metav1.TypeMeta   `json:",inline"` | ||||||
|  | 	metav1.ObjectMeta `json:"metadata,omitempty"` | ||||||
|  | 
 | ||||||
|  | 	Spec   RunnerDeploymentSpec   `json:"spec,omitempty"` | ||||||
|  | 	Status RunnerDeploymentStatus `json:"status,omitempty"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // RunnerSetSpec defines the desired state of RunnerDeployment
 | ||||||
|  | type RunnerDeploymentSpec struct { | ||||||
|  | 	Replicas *int `json:"replicas"` | ||||||
|  | 
 | ||||||
|  | 	Template RunnerTemplate `json:"template"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type RunnerDeploymentStatus struct { | ||||||
|  | 	AvailableReplicas int `json:"availableReplicas"` | ||||||
|  | 	ReadyReplicas     int `json:"readyReplicas"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // +kubebuilder:object:root=true
 | ||||||
|  | 
 | ||||||
|  | // RunnerList contains a list of Runner
 | ||||||
|  | type RunnerDeploymentList struct { | ||||||
|  | 	metav1.TypeMeta `json:",inline"` | ||||||
|  | 	metav1.ListMeta `json:"metadata,omitempty"` | ||||||
|  | 	Items           []RunnerDeployment `json:"items"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func init() { | ||||||
|  | 	SchemeBuilder.Register(&Runner{}, &RunnerList{}, &RunnerSet{}, &RunnerSetList{}, &RunnerDeployment{}, &RunnerDeploymentList{}) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -52,6 +52,101 @@ func (in *Runner) DeepCopyObject() runtime.Object { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | ||||||
|  | func (in *RunnerDeployment) DeepCopyInto(out *RunnerDeployment) { | ||||||
|  | 	*out = *in | ||||||
|  | 	out.TypeMeta = in.TypeMeta | ||||||
|  | 	in.ObjectMeta.DeepCopyInto(&out.ObjectMeta) | ||||||
|  | 	in.Spec.DeepCopyInto(&out.Spec) | ||||||
|  | 	out.Status = in.Status | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunnerDeployment.
 | ||||||
|  | func (in *RunnerDeployment) DeepCopy() *RunnerDeployment { | ||||||
|  | 	if in == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	out := new(RunnerDeployment) | ||||||
|  | 	in.DeepCopyInto(out) | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | ||||||
|  | func (in *RunnerDeployment) DeepCopyObject() runtime.Object { | ||||||
|  | 	if c := in.DeepCopy(); c != nil { | ||||||
|  | 		return c | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | ||||||
|  | func (in *RunnerDeploymentList) DeepCopyInto(out *RunnerDeploymentList) { | ||||||
|  | 	*out = *in | ||||||
|  | 	out.TypeMeta = in.TypeMeta | ||||||
|  | 	in.ListMeta.DeepCopyInto(&out.ListMeta) | ||||||
|  | 	if in.Items != nil { | ||||||
|  | 		in, out := &in.Items, &out.Items | ||||||
|  | 		*out = make([]RunnerDeployment, len(*in)) | ||||||
|  | 		for i := range *in { | ||||||
|  | 			(*in)[i].DeepCopyInto(&(*out)[i]) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunnerDeploymentList.
 | ||||||
|  | func (in *RunnerDeploymentList) DeepCopy() *RunnerDeploymentList { | ||||||
|  | 	if in == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	out := new(RunnerDeploymentList) | ||||||
|  | 	in.DeepCopyInto(out) | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
 | ||||||
|  | func (in *RunnerDeploymentList) DeepCopyObject() runtime.Object { | ||||||
|  | 	if c := in.DeepCopy(); c != nil { | ||||||
|  | 		return c | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | ||||||
|  | func (in *RunnerDeploymentSpec) DeepCopyInto(out *RunnerDeploymentSpec) { | ||||||
|  | 	*out = *in | ||||||
|  | 	if in.Replicas != nil { | ||||||
|  | 		in, out := &in.Replicas, &out.Replicas | ||||||
|  | 		*out = new(int) | ||||||
|  | 		**out = **in | ||||||
|  | 	} | ||||||
|  | 	in.Template.DeepCopyInto(&out.Template) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunnerDeploymentSpec.
 | ||||||
|  | func (in *RunnerDeploymentSpec) DeepCopy() *RunnerDeploymentSpec { | ||||||
|  | 	if in == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	out := new(RunnerDeploymentSpec) | ||||||
|  | 	in.DeepCopyInto(out) | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | ||||||
|  | func (in *RunnerDeploymentStatus) DeepCopyInto(out *RunnerDeploymentStatus) { | ||||||
|  | 	*out = *in | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunnerDeploymentStatus.
 | ||||||
|  | func (in *RunnerDeploymentStatus) DeepCopy() *RunnerDeploymentStatus { | ||||||
|  | 	if in == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	out := new(RunnerDeploymentStatus) | ||||||
|  | 	in.DeepCopyInto(out) | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | ||||||
| func (in *RunnerList) DeepCopyInto(out *RunnerList) { | func (in *RunnerList) DeepCopyInto(out *RunnerList) { | ||||||
| 	*out = *in | 	*out = *in | ||||||
|  | @ -232,3 +327,20 @@ func (in *RunnerStatusRegistration) DeepCopy() *RunnerStatusRegistration { | ||||||
| 	in.DeepCopyInto(out) | 	in.DeepCopyInto(out) | ||||||
| 	return out | 	return out | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
 | ||||||
|  | func (in *RunnerTemplate) DeepCopyInto(out *RunnerTemplate) { | ||||||
|  | 	*out = *in | ||||||
|  | 	in.ObjectMeta.DeepCopyInto(&out.ObjectMeta) | ||||||
|  | 	in.Spec.DeepCopyInto(&out.Spec) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunnerTemplate.
 | ||||||
|  | func (in *RunnerTemplate) DeepCopy() *RunnerTemplate { | ||||||
|  | 	if in == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	out := new(RunnerTemplate) | ||||||
|  | 	in.DeepCopyInto(out) | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,199 @@ | ||||||
|  | 
 | ||||||
|  | --- | ||||||
|  | apiVersion: apiextensions.k8s.io/v1beta1 | ||||||
|  | kind: CustomResourceDefinition | ||||||
|  | metadata: | ||||||
|  |   annotations: | ||||||
|  |     controller-gen.kubebuilder.io/version: v0.2.4 | ||||||
|  |   creationTimestamp: null | ||||||
|  |   name: runnerdeployments.actions.summerwind.dev | ||||||
|  | spec: | ||||||
|  |   additionalPrinterColumns: | ||||||
|  |   - JSONPath: .spec.replicas | ||||||
|  |     name: Desired | ||||||
|  |     type: number | ||||||
|  |   - JSONPath: .status.availableReplicas | ||||||
|  |     name: Current | ||||||
|  |     type: number | ||||||
|  |   - JSONPath: .status.readyReplicas | ||||||
|  |     name: Ready | ||||||
|  |     type: number | ||||||
|  |   group: actions.summerwind.dev | ||||||
|  |   names: | ||||||
|  |     kind: RunnerDeployment | ||||||
|  |     listKind: RunnerDeploymentList | ||||||
|  |     plural: runnerdeployments | ||||||
|  |     singular: runnerdeployment | ||||||
|  |   scope: Namespaced | ||||||
|  |   subresources: | ||||||
|  |     status: {} | ||||||
|  |   validation: | ||||||
|  |     openAPIV3Schema: | ||||||
|  |       description: RunnerSet is the Schema for the runnersets API | ||||||
|  |       properties: | ||||||
|  |         apiVersion: | ||||||
|  |           description: 'APIVersion defines the versioned schema of this representation | ||||||
|  |             of an object. Servers should convert recognized schemas to the latest | ||||||
|  |             internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' | ||||||
|  |           type: string | ||||||
|  |         kind: | ||||||
|  |           description: 'Kind is a string value representing the REST resource this | ||||||
|  |             object represents. Servers may infer this from the endpoint the client | ||||||
|  |             submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' | ||||||
|  |           type: string | ||||||
|  |         metadata: | ||||||
|  |           type: object | ||||||
|  |         spec: | ||||||
|  |           description: RunnerSetSpec defines the desired state of RunnerDeployment | ||||||
|  |           properties: | ||||||
|  |             replicas: | ||||||
|  |               type: integer | ||||||
|  |             template: | ||||||
|  |               properties: | ||||||
|  |                 metadata: | ||||||
|  |                   type: object | ||||||
|  |                 spec: | ||||||
|  |                   description: RunnerSpec defines the desired state of Runner | ||||||
|  |                   properties: | ||||||
|  |                     env: | ||||||
|  |                       items: | ||||||
|  |                         description: EnvVar represents an environment variable present | ||||||
|  |                           in a Container. | ||||||
|  |                         properties: | ||||||
|  |                           name: | ||||||
|  |                             description: Name of the environment variable. Must be | ||||||
|  |                               a C_IDENTIFIER. | ||||||
|  |                             type: string | ||||||
|  |                           value: | ||||||
|  |                             description: 'Variable references $(VAR_NAME) are expanded | ||||||
|  |                               using the previous defined environment variables in | ||||||
|  |                               the container and any service environment variables. | ||||||
|  |                               If a variable cannot be resolved, the reference in the | ||||||
|  |                               input string will be unchanged. The $(VAR_NAME) syntax | ||||||
|  |                               can be escaped with a double $$, ie: $$(VAR_NAME). Escaped | ||||||
|  |                               references will never be expanded, regardless of whether | ||||||
|  |                               the variable exists or not. Defaults to "".' | ||||||
|  |                             type: string | ||||||
|  |                           valueFrom: | ||||||
|  |                             description: Source for the environment variable's value. | ||||||
|  |                               Cannot be used if value is not empty. | ||||||
|  |                             properties: | ||||||
|  |                               configMapKeyRef: | ||||||
|  |                                 description: Selects a key of a ConfigMap. | ||||||
|  |                                 properties: | ||||||
|  |                                   key: | ||||||
|  |                                     description: The key to select. | ||||||
|  |                                     type: string | ||||||
|  |                                   name: | ||||||
|  |                                     description: 'Name of the referent. More info: | ||||||
|  |                                       https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names | ||||||
|  |                                       TODO: Add other useful fields. apiVersion, kind, | ||||||
|  |                                       uid?' | ||||||
|  |                                     type: string | ||||||
|  |                                   optional: | ||||||
|  |                                     description: Specify whether the ConfigMap or | ||||||
|  |                                       its key must be defined | ||||||
|  |                                     type: boolean | ||||||
|  |                                 required: | ||||||
|  |                                 - key | ||||||
|  |                                 type: object | ||||||
|  |                               fieldRef: | ||||||
|  |                                 description: 'Selects a field of the pod: supports | ||||||
|  |                                   metadata.name, metadata.namespace, metadata.labels, | ||||||
|  |                                   metadata.annotations, spec.nodeName, spec.serviceAccountName, | ||||||
|  |                                   status.hostIP, status.podIP.' | ||||||
|  |                                 properties: | ||||||
|  |                                   apiVersion: | ||||||
|  |                                     description: Version of the schema the FieldPath | ||||||
|  |                                       is written in terms of, defaults to "v1". | ||||||
|  |                                     type: string | ||||||
|  |                                   fieldPath: | ||||||
|  |                                     description: Path of the field to select in the | ||||||
|  |                                       specified API version. | ||||||
|  |                                     type: string | ||||||
|  |                                 required: | ||||||
|  |                                 - fieldPath | ||||||
|  |                                 type: object | ||||||
|  |                               resourceFieldRef: | ||||||
|  |                                 description: 'Selects a resource of the container: | ||||||
|  |                                   only resources limits and requests (limits.cpu, | ||||||
|  |                                   limits.memory, limits.ephemeral-storage, requests.cpu, | ||||||
|  |                                   requests.memory and requests.ephemeral-storage) | ||||||
|  |                                   are currently supported.' | ||||||
|  |                                 properties: | ||||||
|  |                                   containerName: | ||||||
|  |                                     description: 'Container name: required for volumes, | ||||||
|  |                                       optional for env vars' | ||||||
|  |                                     type: string | ||||||
|  |                                   divisor: | ||||||
|  |                                     description: Specifies the output format of the | ||||||
|  |                                       exposed resources, defaults to "1" | ||||||
|  |                                     type: string | ||||||
|  |                                   resource: | ||||||
|  |                                     description: 'Required: resource to select' | ||||||
|  |                                     type: string | ||||||
|  |                                 required: | ||||||
|  |                                 - resource | ||||||
|  |                                 type: object | ||||||
|  |                               secretKeyRef: | ||||||
|  |                                 description: Selects a key of a secret in the pod's | ||||||
|  |                                   namespace | ||||||
|  |                                 properties: | ||||||
|  |                                   key: | ||||||
|  |                                     description: The key of the secret to select from.  Must | ||||||
|  |                                       be a valid secret key. | ||||||
|  |                                     type: string | ||||||
|  |                                   name: | ||||||
|  |                                     description: 'Name of the referent. More info: | ||||||
|  |                                       https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names | ||||||
|  |                                       TODO: Add other useful fields. apiVersion, kind, | ||||||
|  |                                       uid?' | ||||||
|  |                                     type: string | ||||||
|  |                                   optional: | ||||||
|  |                                     description: Specify whether the Secret or its | ||||||
|  |                                       key must be defined | ||||||
|  |                                     type: boolean | ||||||
|  |                                 required: | ||||||
|  |                                 - key | ||||||
|  |                                 type: object | ||||||
|  |                             type: object | ||||||
|  |                         required: | ||||||
|  |                         - name | ||||||
|  |                         type: object | ||||||
|  |                       type: array | ||||||
|  |                     image: | ||||||
|  |                       type: string | ||||||
|  |                     repository: | ||||||
|  |                       minLength: 3 | ||||||
|  |                       pattern: ^[^/]+/[^/]+$ | ||||||
|  |                       type: string | ||||||
|  |                   required: | ||||||
|  |                   - repository | ||||||
|  |                   type: object | ||||||
|  |               type: object | ||||||
|  |           required: | ||||||
|  |           - replicas | ||||||
|  |           - template | ||||||
|  |           type: object | ||||||
|  |         status: | ||||||
|  |           properties: | ||||||
|  |             availableReplicas: | ||||||
|  |               type: integer | ||||||
|  |             readyReplicas: | ||||||
|  |               type: integer | ||||||
|  |           required: | ||||||
|  |           - availableReplicas | ||||||
|  |           - readyReplicas | ||||||
|  |           type: object | ||||||
|  |       type: object | ||||||
|  |   version: v1alpha1 | ||||||
|  |   versions: | ||||||
|  |   - name: v1alpha1 | ||||||
|  |     served: true | ||||||
|  |     storage: true | ||||||
|  | status: | ||||||
|  |   acceptedNames: | ||||||
|  |     kind: "" | ||||||
|  |     plural: "" | ||||||
|  |   conditions: [] | ||||||
|  |   storedVersions: [] | ||||||
|  | @ -49,117 +49,127 @@ spec: | ||||||
|             replicas: |             replicas: | ||||||
|               type: integer |               type: integer | ||||||
|             template: |             template: | ||||||
|               description: RunnerSpec defines the desired state of Runner |  | ||||||
|               properties: |               properties: | ||||||
|                 env: |                 metadata: | ||||||
|                   items: |                   type: object | ||||||
|                     description: EnvVar represents an environment variable present |                 spec: | ||||||
|                       in a Container. |                   description: RunnerSpec defines the desired state of Runner | ||||||
|                     properties: |                   properties: | ||||||
|                       name: |                     env: | ||||||
|                         description: Name of the environment variable. Must be a C_IDENTIFIER. |                       items: | ||||||
|                         type: string |                         description: EnvVar represents an environment variable present | ||||||
|                       value: |                           in a Container. | ||||||
|                         description: 'Variable references $(VAR_NAME) are expanded |  | ||||||
|                           using the previous defined environment variables in the |  | ||||||
|                           container and any service environment variables. If a variable |  | ||||||
|                           cannot be resolved, the reference in the input string will |  | ||||||
|                           be unchanged. The $(VAR_NAME) syntax can be escaped with |  | ||||||
|                           a double $$, ie: $$(VAR_NAME). Escaped references will never |  | ||||||
|                           be expanded, regardless of whether the variable exists or |  | ||||||
|                           not. Defaults to "".' |  | ||||||
|                         type: string |  | ||||||
|                       valueFrom: |  | ||||||
|                         description: Source for the environment variable's value. |  | ||||||
|                           Cannot be used if value is not empty. |  | ||||||
|                         properties: |                         properties: | ||||||
|                           configMapKeyRef: |                           name: | ||||||
|                             description: Selects a key of a ConfigMap. |                             description: Name of the environment variable. Must be | ||||||
|  |                               a C_IDENTIFIER. | ||||||
|  |                             type: string | ||||||
|  |                           value: | ||||||
|  |                             description: 'Variable references $(VAR_NAME) are expanded | ||||||
|  |                               using the previous defined environment variables in | ||||||
|  |                               the container and any service environment variables. | ||||||
|  |                               If a variable cannot be resolved, the reference in the | ||||||
|  |                               input string will be unchanged. The $(VAR_NAME) syntax | ||||||
|  |                               can be escaped with a double $$, ie: $$(VAR_NAME). Escaped | ||||||
|  |                               references will never be expanded, regardless of whether | ||||||
|  |                               the variable exists or not. Defaults to "".' | ||||||
|  |                             type: string | ||||||
|  |                           valueFrom: | ||||||
|  |                             description: Source for the environment variable's value. | ||||||
|  |                               Cannot be used if value is not empty. | ||||||
|                             properties: |                             properties: | ||||||
|                               key: |                               configMapKeyRef: | ||||||
|                                 description: The key to select. |                                 description: Selects a key of a ConfigMap. | ||||||
|                                 type: string |                                 properties: | ||||||
|                               name: |                                   key: | ||||||
|                                 description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names |                                     description: The key to select. | ||||||
|                                   TODO: Add other useful fields. apiVersion, kind, |                                     type: string | ||||||
|                                   uid?' |                                   name: | ||||||
|                                 type: string |                                     description: 'Name of the referent. More info: | ||||||
|                               optional: |                                       https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names | ||||||
|                                 description: Specify whether the ConfigMap or its |                                       TODO: Add other useful fields. apiVersion, kind, | ||||||
|                                   key must be defined |                                       uid?' | ||||||
|                                 type: boolean |                                     type: string | ||||||
|                             required: |                                   optional: | ||||||
|                             - key |                                     description: Specify whether the ConfigMap or | ||||||
|                             type: object |                                       its key must be defined | ||||||
|                           fieldRef: |                                     type: boolean | ||||||
|                             description: 'Selects a field of the pod: supports metadata.name, |                                 required: | ||||||
|                               metadata.namespace, metadata.labels, metadata.annotations, |                                 - key | ||||||
|                               spec.nodeName, spec.serviceAccountName, status.hostIP, |                                 type: object | ||||||
|                               status.podIP.' |                               fieldRef: | ||||||
|                             properties: |                                 description: 'Selects a field of the pod: supports | ||||||
|                               apiVersion: |                                   metadata.name, metadata.namespace, metadata.labels, | ||||||
|                                 description: Version of the schema the FieldPath is |                                   metadata.annotations, spec.nodeName, spec.serviceAccountName, | ||||||
|                                   written in terms of, defaults to "v1". |                                   status.hostIP, status.podIP.' | ||||||
|                                 type: string |                                 properties: | ||||||
|                               fieldPath: |                                   apiVersion: | ||||||
|                                 description: Path of the field to select in the specified |                                     description: Version of the schema the FieldPath | ||||||
|                                   API version. |                                       is written in terms of, defaults to "v1". | ||||||
|                                 type: string |                                     type: string | ||||||
|                             required: |                                   fieldPath: | ||||||
|                             - fieldPath |                                     description: Path of the field to select in the | ||||||
|                             type: object |                                       specified API version. | ||||||
|                           resourceFieldRef: |                                     type: string | ||||||
|                             description: 'Selects a resource of the container: only |                                 required: | ||||||
|                               resources limits and requests (limits.cpu, limits.memory, |                                 - fieldPath | ||||||
|                               limits.ephemeral-storage, requests.cpu, requests.memory |                                 type: object | ||||||
|                               and requests.ephemeral-storage) are currently supported.' |                               resourceFieldRef: | ||||||
|                             properties: |                                 description: 'Selects a resource of the container: | ||||||
|                               containerName: |                                   only resources limits and requests (limits.cpu, | ||||||
|                                 description: 'Container name: required for volumes, |                                   limits.memory, limits.ephemeral-storage, requests.cpu, | ||||||
|                                   optional for env vars' |                                   requests.memory and requests.ephemeral-storage) | ||||||
|                                 type: string |                                   are currently supported.' | ||||||
|                               divisor: |                                 properties: | ||||||
|                                 description: Specifies the output format of the exposed |                                   containerName: | ||||||
|                                   resources, defaults to "1" |                                     description: 'Container name: required for volumes, | ||||||
|                                 type: string |                                       optional for env vars' | ||||||
|                               resource: |                                     type: string | ||||||
|                                 description: 'Required: resource to select' |                                   divisor: | ||||||
|                                 type: string |                                     description: Specifies the output format of the | ||||||
|                             required: |                                       exposed resources, defaults to "1" | ||||||
|                             - resource |                                     type: string | ||||||
|                             type: object |                                   resource: | ||||||
|                           secretKeyRef: |                                     description: 'Required: resource to select' | ||||||
|                             description: Selects a key of a secret in the pod's namespace |                                     type: string | ||||||
|                             properties: |                                 required: | ||||||
|                               key: |                                 - resource | ||||||
|                                 description: The key of the secret to select from.  Must |                                 type: object | ||||||
|                                   be a valid secret key. |                               secretKeyRef: | ||||||
|                                 type: string |                                 description: Selects a key of a secret in the pod's | ||||||
|                               name: |                                   namespace | ||||||
|                                 description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names |                                 properties: | ||||||
|                                   TODO: Add other useful fields. apiVersion, kind, |                                   key: | ||||||
|                                   uid?' |                                     description: The key of the secret to select from.  Must | ||||||
|                                 type: string |                                       be a valid secret key. | ||||||
|                               optional: |                                     type: string | ||||||
|                                 description: Specify whether the Secret or its key |                                   name: | ||||||
|                                   must be defined |                                     description: 'Name of the referent. More info: | ||||||
|                                 type: boolean |                                       https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names | ||||||
|                             required: |                                       TODO: Add other useful fields. apiVersion, kind, | ||||||
|                             - key |                                       uid?' | ||||||
|  |                                     type: string | ||||||
|  |                                   optional: | ||||||
|  |                                     description: Specify whether the Secret or its | ||||||
|  |                                       key must be defined | ||||||
|  |                                     type: boolean | ||||||
|  |                                 required: | ||||||
|  |                                 - key | ||||||
|  |                                 type: object | ||||||
|                             type: object |                             type: object | ||||||
|  |                         required: | ||||||
|  |                         - name | ||||||
|                         type: object |                         type: object | ||||||
|                     required: |                       type: array | ||||||
|                     - name |                     image: | ||||||
|                     type: object |                       type: string | ||||||
|                   type: array |                     repository: | ||||||
|                 image: |                       minLength: 3 | ||||||
|                   type: string |                       pattern: ^[^/]+/[^/]+$ | ||||||
|                 repository: |                       type: string | ||||||
|                   minLength: 3 |                   required: | ||||||
|                   pattern: ^[^/]+/[^/]+$ |                   - repository | ||||||
|                   type: string |                   type: object | ||||||
|               required: |  | ||||||
|               - repository |  | ||||||
|               type: object |               type: object | ||||||
|           required: |           required: | ||||||
|           - replicas |           - replicas | ||||||
|  |  | ||||||
|  | @ -6,6 +6,26 @@ metadata: | ||||||
|   creationTimestamp: null |   creationTimestamp: null | ||||||
|   name: manager-role |   name: manager-role | ||||||
| rules: | rules: | ||||||
|  | - apiGroups: | ||||||
|  |   - actions.summerwind.dev | ||||||
|  |   resources: | ||||||
|  |   - runnerdeployments | ||||||
|  |   verbs: | ||||||
|  |   - create | ||||||
|  |   - delete | ||||||
|  |   - get | ||||||
|  |   - list | ||||||
|  |   - patch | ||||||
|  |   - update | ||||||
|  |   - watch | ||||||
|  | - apiGroups: | ||||||
|  |   - actions.summerwind.dev | ||||||
|  |   resources: | ||||||
|  |   - runnerdeployments/status | ||||||
|  |   verbs: | ||||||
|  |   - get | ||||||
|  |   - patch | ||||||
|  |   - update | ||||||
| - apiGroups: | - apiGroups: | ||||||
|   - actions.summerwind.dev |   - actions.summerwind.dev | ||||||
|   resources: |   resources: | ||||||
|  |  | ||||||
|  | @ -0,0 +1,260 @@ | ||||||
|  | /* | ||||||
|  | Copyright 2020 The actions-runner-controller authors. | ||||||
|  | 
 | ||||||
|  | Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  | you may not use this file except in compliance with the License. | ||||||
|  | You may obtain a copy of the License at | ||||||
|  | 
 | ||||||
|  |     http://www.apache.org/licenses/LICENSE-2.0
 | ||||||
|  | 
 | ||||||
|  | Unless required by applicable law or agreed to in writing, software | ||||||
|  | distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  | See the License for the specific language governing permissions and | ||||||
|  | limitations under the License. | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | package controllers | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"github.com/davecgh/go-spew/spew" | ||||||
|  | 	"github.com/go-logr/logr" | ||||||
|  | 	"hash/fnv" | ||||||
|  | 	"k8s.io/apimachinery/pkg/runtime" | ||||||
|  | 	"k8s.io/apimachinery/pkg/util/rand" | ||||||
|  | 	"k8s.io/client-go/tools/record" | ||||||
|  | 	ctrl "sigs.k8s.io/controller-runtime" | ||||||
|  | 	"sigs.k8s.io/controller-runtime/pkg/client" | ||||||
|  | 	"sort" | ||||||
|  | 
 | ||||||
|  | 	corev1 "k8s.io/api/core/v1" | ||||||
|  | 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||||||
|  | 
 | ||||||
|  | 	"github.com/summerwind/actions-runner-controller/api/v1alpha1" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	LabelKeyRunnerTemplateHash = "runner-template-hash" | ||||||
|  | 
 | ||||||
|  | 	runnerSetOwnerKey = ".metadata.controller" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // RunnerDeploymentReconciler reconciles a Runner object
 | ||||||
|  | type RunnerDeploymentReconciler struct { | ||||||
|  | 	client.Client | ||||||
|  | 	Log      logr.Logger | ||||||
|  | 	Recorder record.EventRecorder | ||||||
|  | 	Scheme   *runtime.Scheme | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // +kubebuilder:rbac:groups=actions.summerwind.dev,resources=runnerdeployments,verbs=get;list;watch;create;update;patch;delete
 | ||||||
|  | // +kubebuilder:rbac:groups=actions.summerwind.dev,resources=runnerdeployments/status,verbs=get;update;patch
 | ||||||
|  | // +kubebuilder:rbac:groups=actions.summerwind.dev,resources=runnersets,verbs=get;list;watch;create;update;patch;delete
 | ||||||
|  | // +kubebuilder:rbac:groups=actions.summerwind.dev,resources=runnersets/status,verbs=get;update;patch
 | ||||||
|  | 
 | ||||||
|  | func (r *RunnerDeploymentReconciler) Reconcile(req ctrl.Request) (ctrl.Result, error) { | ||||||
|  | 	ctx := context.Background() | ||||||
|  | 	log := r.Log.WithValues("runnerset", req.NamespacedName) | ||||||
|  | 
 | ||||||
|  | 	var rd v1alpha1.RunnerDeployment | ||||||
|  | 	if err := r.Get(ctx, req.NamespacedName, &rd); err != nil { | ||||||
|  | 		return ctrl.Result{}, client.IgnoreNotFound(err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if !rd.ObjectMeta.DeletionTimestamp.IsZero() { | ||||||
|  | 		return ctrl.Result{}, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	var myRunnerSetList v1alpha1.RunnerSetList | ||||||
|  | 	if err := r.List(ctx, &myRunnerSetList, client.InNamespace(req.Namespace), client.MatchingFields{runnerSetOwnerKey: req.Name}); err != nil { | ||||||
|  | 		return ctrl.Result{}, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	myRunnerSets := myRunnerSetList.Items | ||||||
|  | 
 | ||||||
|  | 	sort.Slice(myRunnerSets, func(i, j int) bool { | ||||||
|  | 		return myRunnerSets[i].GetCreationTimestamp().After(myRunnerSets[j].GetCreationTimestamp().Time) | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	var newestSet *v1alpha1.RunnerSet | ||||||
|  | 
 | ||||||
|  | 	var oldSets []v1alpha1.RunnerSet | ||||||
|  | 
 | ||||||
|  | 	if len(myRunnerSets) > 0 { | ||||||
|  | 		newestSet = &myRunnerSets[0] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if len(myRunnerSets) > 1 { | ||||||
|  | 		oldSets = myRunnerSets[1:] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	desiredRS, err := r.newRunnerSet(rd) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Error(err, "Could not create runnerset") | ||||||
|  | 
 | ||||||
|  | 		return ctrl.Result{}, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if newestSet == nil { | ||||||
|  | 		if err := r.Client.Create(ctx, &desiredRS); err != nil { | ||||||
|  | 			log.Error(err, "Failed to create runnerset resource") | ||||||
|  | 
 | ||||||
|  | 			return ctrl.Result{}, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return ctrl.Result{}, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	newestTemplateHash, ok := getTemplateHash(newestSet) | ||||||
|  | 	if !ok { | ||||||
|  | 		log.Info("Failed to get template hash of newest runnerset resource. It must be in an invalid state. Please manually delete the runnerset so that it is recreated") | ||||||
|  | 
 | ||||||
|  | 		return ctrl.Result{}, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	desiredTemplateHash, ok := getTemplateHash(&desiredRS) | ||||||
|  | 	if !ok { | ||||||
|  | 		log.Info("Failed to get template hash of desired runnerset resource. It must be in an invalid state. Please manually delete the runnerset so that it is recreated") | ||||||
|  | 
 | ||||||
|  | 		return ctrl.Result{}, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if newestTemplateHash != desiredTemplateHash { | ||||||
|  | 		if err := r.Client.Create(ctx, &desiredRS); err != nil { | ||||||
|  | 			log.Error(err, "Failed to create runnerset resource") | ||||||
|  | 
 | ||||||
|  | 			return ctrl.Result{}, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return ctrl.Result{}, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Please add more conditions that we can in-place update the newest runnerset without disruption
 | ||||||
|  | 	if newestSet.Spec.Replicas != desiredRS.Spec.Replicas { | ||||||
|  | 		newestSet.Spec.Replicas = desiredRS.Spec.Replicas | ||||||
|  | 
 | ||||||
|  | 		if err := r.Client.Update(ctx, newestSet); err != nil { | ||||||
|  | 			log.Error(err, "Failed to update runnerset resource") | ||||||
|  | 
 | ||||||
|  | 			return ctrl.Result{}, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return ctrl.Result{}, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for i := range oldSets { | ||||||
|  | 		rs := oldSets[i] | ||||||
|  | 
 | ||||||
|  | 		if err := r.Client.Delete(ctx, &rs); err != nil { | ||||||
|  | 			log.Error(err, "Failed to delete runner resource") | ||||||
|  | 
 | ||||||
|  | 			return ctrl.Result{}, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		r.Recorder.Event(&rd, corev1.EventTypeNormal, "RunnerSetDeleted", fmt.Sprintf("Deleted runnerset '%s'", rs.Name)) | ||||||
|  | 		log.Info("Deleted runnerset", "runnerdeployment", rd.ObjectMeta.Name, "runnerset", rs.Name) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ctrl.Result{}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func getTemplateHash(rs *v1alpha1.RunnerSet) (string, bool) { | ||||||
|  | 	hash, ok := rs.Labels[LabelKeyRunnerTemplateHash] | ||||||
|  | 
 | ||||||
|  | 	return hash, ok | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // ComputeHash returns a hash value calculated from pod template and
 | ||||||
|  | // a collisionCount to avoid hash collision. The hash will be safe encoded to
 | ||||||
|  | // avoid bad words.
 | ||||||
|  | //
 | ||||||
|  | // Proudly modified and adopted from k8s.io/kubernetes/pkg/util/hash.DeepHashObject and
 | ||||||
|  | // k8s.io/kubernetes/pkg/controller.ComputeHash.
 | ||||||
|  | func ComputeHash(template interface{}) string { | ||||||
|  | 	hasher := fnv.New32a() | ||||||
|  | 
 | ||||||
|  | 	hasher.Reset() | ||||||
|  | 
 | ||||||
|  | 	printer := spew.ConfigState{ | ||||||
|  | 		Indent:         " ", | ||||||
|  | 		SortKeys:       true, | ||||||
|  | 		DisableMethods: true, | ||||||
|  | 		SpewKeys:       true, | ||||||
|  | 	} | ||||||
|  | 	printer.Fprintf(hasher, "%#v", template) | ||||||
|  | 
 | ||||||
|  | 	return rand.SafeEncodeString(fmt.Sprint(hasher.Sum32())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Clones the given map and returns a new map with the given key and value added.
 | ||||||
|  | // Returns the given map, if labelKey is empty.
 | ||||||
|  | //
 | ||||||
|  | // Proudly copied from k8s.io/kubernetes/pkg/util/labels.CloneAndAddLabel
 | ||||||
|  | func CloneAndAddLabel(labels map[string]string, labelKey, labelValue string) map[string]string { | ||||||
|  | 	if labelKey == "" { | ||||||
|  | 		// Don't need to add a label.
 | ||||||
|  | 		return labels | ||||||
|  | 	} | ||||||
|  | 	// Clone.
 | ||||||
|  | 	newLabels := map[string]string{} | ||||||
|  | 	for key, value := range labels { | ||||||
|  | 		newLabels[key] = value | ||||||
|  | 	} | ||||||
|  | 	newLabels[labelKey] = labelValue | ||||||
|  | 	return newLabels | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (r *RunnerDeploymentReconciler) newRunnerSet(rd v1alpha1.RunnerDeployment) (v1alpha1.RunnerSet, error) { | ||||||
|  | 	newRSTemplate := *rd.Spec.Template.DeepCopy() | ||||||
|  | 	templateHash := ComputeHash(&newRSTemplate) | ||||||
|  | 	// Add template hash label to selector.
 | ||||||
|  | 	labels := CloneAndAddLabel(rd.Spec.Template.Labels, LabelKeyRunnerTemplateHash, templateHash) | ||||||
|  | 
 | ||||||
|  | 	newRSTemplate.Labels = labels | ||||||
|  | 
 | ||||||
|  | 	rs := v1alpha1.RunnerSet{ | ||||||
|  | 		TypeMeta: metav1.TypeMeta{}, | ||||||
|  | 		ObjectMeta: metav1.ObjectMeta{ | ||||||
|  | 			GenerateName: rd.ObjectMeta.Name, | ||||||
|  | 			Namespace:    rd.ObjectMeta.Namespace, | ||||||
|  | 			Labels:       labels, | ||||||
|  | 		}, | ||||||
|  | 		Spec: v1alpha1.RunnerSetSpec{ | ||||||
|  | 			Replicas: rd.Spec.Replicas, | ||||||
|  | 			Template: newRSTemplate, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if err := ctrl.SetControllerReference(&rd, &rs, r.Scheme); err != nil { | ||||||
|  | 		return rs, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return rs, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (r *RunnerDeploymentReconciler) SetupWithManager(mgr ctrl.Manager) error { | ||||||
|  | 	r.Recorder = mgr.GetEventRecorderFor("runnerdeployment-controller") | ||||||
|  | 
 | ||||||
|  | 	if err := mgr.GetFieldIndexer().IndexField(&v1alpha1.RunnerSet{}, runnerSetOwnerKey, func(rawObj runtime.Object) []string { | ||||||
|  | 		runnerSet := rawObj.(*v1alpha1.RunnerSet) | ||||||
|  | 		owner := metav1.GetControllerOf(runnerSet) | ||||||
|  | 		if owner == nil { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if owner.APIVersion != v1alpha1.GroupVersion.String() || owner.Kind != "RunnerSet" { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return []string{owner.Name} | ||||||
|  | 	}); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ctrl.NewControllerManagedBy(mgr). | ||||||
|  | 		For(&v1alpha1.RunnerDeployment{}). | ||||||
|  | 		Owns(&v1alpha1.RunnerSet{}). | ||||||
|  | 		Complete(r) | ||||||
|  | } | ||||||
|  | @ -0,0 +1,175 @@ | ||||||
|  | package controllers | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	corev1 "k8s.io/api/core/v1" | ||||||
|  | 	"k8s.io/apimachinery/pkg/types" | ||||||
|  | 	"k8s.io/client-go/kubernetes/scheme" | ||||||
|  | 	ctrl "sigs.k8s.io/controller-runtime" | ||||||
|  | 	logf "sigs.k8s.io/controller-runtime/pkg/log" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	. "github.com/onsi/ginkgo" | ||||||
|  | 	. "github.com/onsi/gomega" | ||||||
|  | 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||||||
|  | 	"sigs.k8s.io/controller-runtime/pkg/client" | ||||||
|  | 
 | ||||||
|  | 	actionsv1alpha1 "github.com/summerwind/actions-runner-controller/api/v1alpha1" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // SetupDeploymentTest will set up a testing environment.
 | ||||||
|  | // This includes:
 | ||||||
|  | // * creating a Namespace to be used during the test
 | ||||||
|  | // * starting the 'RunnerDeploymentReconciler'
 | ||||||
|  | // * stopping the 'RunnerDeploymentReconciler" after the test ends
 | ||||||
|  | // Call this function at the start of each of your tests.
 | ||||||
|  | func SetupDeploymentTest(ctx context.Context) *corev1.Namespace { | ||||||
|  | 	var stopCh chan struct{} | ||||||
|  | 	ns := &corev1.Namespace{} | ||||||
|  | 
 | ||||||
|  | 	BeforeEach(func() { | ||||||
|  | 		stopCh = make(chan struct{}) | ||||||
|  | 		*ns = corev1.Namespace{ | ||||||
|  | 			ObjectMeta: metav1.ObjectMeta{Name: "testns-" + randStringRunes(5)}, | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		err := k8sClient.Create(ctx, ns) | ||||||
|  | 		Expect(err).NotTo(HaveOccurred(), "failed to create test namespace") | ||||||
|  | 
 | ||||||
|  | 		mgr, err := ctrl.NewManager(cfg, ctrl.Options{}) | ||||||
|  | 		Expect(err).NotTo(HaveOccurred(), "failed to create manager") | ||||||
|  | 
 | ||||||
|  | 		controller := &RunnerDeploymentReconciler{ | ||||||
|  | 			Client:   mgr.GetClient(), | ||||||
|  | 			Scheme:   scheme.Scheme, | ||||||
|  | 			Log:      logf.Log, | ||||||
|  | 			Recorder: mgr.GetEventRecorderFor("runnerset-controller"), | ||||||
|  | 		} | ||||||
|  | 		err = controller.SetupWithManager(mgr) | ||||||
|  | 		Expect(err).NotTo(HaveOccurred(), "failed to setup controller") | ||||||
|  | 
 | ||||||
|  | 		go func() { | ||||||
|  | 			defer GinkgoRecover() | ||||||
|  | 
 | ||||||
|  | 			err := mgr.Start(stopCh) | ||||||
|  | 			Expect(err).NotTo(HaveOccurred(), "failed to start manager") | ||||||
|  | 		}() | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	AfterEach(func() { | ||||||
|  | 		close(stopCh) | ||||||
|  | 
 | ||||||
|  | 		err := k8sClient.Delete(ctx, ns) | ||||||
|  | 		Expect(err).NotTo(HaveOccurred(), "failed to delete test namespace") | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	return ns | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var _ = Context("Inside of a new namespace", func() { | ||||||
|  | 	ctx := context.TODO() | ||||||
|  | 	ns := SetupDeploymentTest(ctx) | ||||||
|  | 
 | ||||||
|  | 	Describe("when no existing resources exist", func() { | ||||||
|  | 
 | ||||||
|  | 		It("should create a new RunnerSet resource from the specified template, add a another RunnerSet on template modification, and eventually removes old runnersets", func() { | ||||||
|  | 			name := "example-runnerdeploy" | ||||||
|  | 
 | ||||||
|  | 			{ | ||||||
|  | 				rs := &actionsv1alpha1.RunnerDeployment{ | ||||||
|  | 					ObjectMeta: metav1.ObjectMeta{ | ||||||
|  | 						Name:      name, | ||||||
|  | 						Namespace: ns.Name, | ||||||
|  | 					}, | ||||||
|  | 					Spec: actionsv1alpha1.RunnerDeploymentSpec{ | ||||||
|  | 						Replicas: intPtr(1), | ||||||
|  | 						Template: actionsv1alpha1.RunnerTemplate{ | ||||||
|  | 							Spec: actionsv1alpha1.RunnerSpec{ | ||||||
|  | 								Repository: "foo/bar", | ||||||
|  | 								Image:      "bar", | ||||||
|  | 								Env: []corev1.EnvVar{ | ||||||
|  | 									{Name: "FOO", Value: "FOOVALUE"}, | ||||||
|  | 								}, | ||||||
|  | 							}, | ||||||
|  | 						}, | ||||||
|  | 					}, | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				err := k8sClient.Create(ctx, rs) | ||||||
|  | 
 | ||||||
|  | 				Expect(err).NotTo(HaveOccurred(), "failed to create test RunnerSet resource") | ||||||
|  | 
 | ||||||
|  | 				runnerSets := actionsv1alpha1.RunnerSetList{Items: []actionsv1alpha1.RunnerSet{}} | ||||||
|  | 
 | ||||||
|  | 				Eventually( | ||||||
|  | 					func() int { | ||||||
|  | 						err := k8sClient.List(ctx, &runnerSets, client.InNamespace(ns.Name)) | ||||||
|  | 						if err != nil { | ||||||
|  | 							logf.Log.Error(err, "list runner sets") | ||||||
|  | 						} | ||||||
|  | 
 | ||||||
|  | 						return len(runnerSets.Items) | ||||||
|  | 					}, | ||||||
|  | 					time.Second*5, time.Millisecond*500).Should(BeEquivalentTo(1)) | ||||||
|  | 
 | ||||||
|  | 				Eventually( | ||||||
|  | 					func() int { | ||||||
|  | 						err := k8sClient.List(ctx, &runnerSets, client.InNamespace(ns.Name)) | ||||||
|  | 						if err != nil { | ||||||
|  | 							logf.Log.Error(err, "list runner sets") | ||||||
|  | 						} | ||||||
|  | 
 | ||||||
|  | 						if len(runnerSets.Items) == 0 { | ||||||
|  | 							logf.Log.Info("No runnersets exist yet") | ||||||
|  | 							return -1 | ||||||
|  | 						} | ||||||
|  | 
 | ||||||
|  | 						return *runnerSets.Items[0].Spec.Replicas | ||||||
|  | 					}, | ||||||
|  | 					time.Second*5, time.Millisecond*500).Should(BeEquivalentTo(1)) | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			{ | ||||||
|  | 				// We wrap the update in the Eventually block to avoid the below error that occurs due to concurrent modification
 | ||||||
|  | 				// made by the controller to update .Status.AvailableReplicas and .Status.ReadyReplicas
 | ||||||
|  | 				//   Operation cannot be fulfilled on runnersets.actions.summerwind.dev "example-runnerset": the object has been modified; please apply your changes to the latest version and try again
 | ||||||
|  | 				Eventually(func() error { | ||||||
|  | 					var rd actionsv1alpha1.RunnerDeployment | ||||||
|  | 
 | ||||||
|  | 					err := k8sClient.Get(ctx, types.NamespacedName{Namespace: ns.Name, Name: name}, &rd) | ||||||
|  | 
 | ||||||
|  | 					Expect(err).NotTo(HaveOccurred(), "failed to get test RunnerSet resource") | ||||||
|  | 
 | ||||||
|  | 					rd.Spec.Replicas = intPtr(2) | ||||||
|  | 
 | ||||||
|  | 					return k8sClient.Update(ctx, &rd) | ||||||
|  | 				}, | ||||||
|  | 					time.Second*1, time.Millisecond*500).Should(BeNil()) | ||||||
|  | 
 | ||||||
|  | 				runnerSets := actionsv1alpha1.RunnerSetList{Items: []actionsv1alpha1.RunnerSet{}} | ||||||
|  | 
 | ||||||
|  | 				Eventually( | ||||||
|  | 					func() int { | ||||||
|  | 						err := k8sClient.List(ctx, &runnerSets, client.InNamespace(ns.Name)) | ||||||
|  | 						if err != nil { | ||||||
|  | 							logf.Log.Error(err, "list runner sets") | ||||||
|  | 						} | ||||||
|  | 
 | ||||||
|  | 						return len(runnerSets.Items) | ||||||
|  | 					}, | ||||||
|  | 					time.Second*5, time.Millisecond*500).Should(BeEquivalentTo(1)) | ||||||
|  | 
 | ||||||
|  | 				Eventually( | ||||||
|  | 					func() int { | ||||||
|  | 						err := k8sClient.List(ctx, &runnerSets, client.InNamespace(ns.Name)) | ||||||
|  | 						if err != nil { | ||||||
|  | 							logf.Log.Error(err, "list runner sets") | ||||||
|  | 						} | ||||||
|  | 
 | ||||||
|  | 						return *runnerSets.Items[0].Spec.Replicas | ||||||
|  | 					}, | ||||||
|  | 					time.Second*5, time.Millisecond*500).Should(BeEquivalentTo(2)) | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
|  | 	}) | ||||||
|  | }) | ||||||
|  | @ -138,13 +138,15 @@ func (r *RunnerSetReconciler) Reconcile(req ctrl.Request) (ctrl.Result, error) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (r *RunnerSetReconciler) newRunner(rs v1alpha1.RunnerSet) (v1alpha1.Runner, error) { | func (r *RunnerSetReconciler) newRunner(rs v1alpha1.RunnerSet) (v1alpha1.Runner, error) { | ||||||
|  | 	objectMeta := rs.Spec.Template.ObjectMeta.DeepCopy() | ||||||
|  | 
 | ||||||
|  | 	objectMeta.GenerateName = rs.ObjectMeta.Name | ||||||
|  | 	objectMeta.Namespace = rs.ObjectMeta.Namespace | ||||||
|  | 
 | ||||||
| 	runner := v1alpha1.Runner{ | 	runner := v1alpha1.Runner{ | ||||||
| 		TypeMeta: metav1.TypeMeta{}, | 		TypeMeta:   metav1.TypeMeta{}, | ||||||
| 		ObjectMeta: metav1.ObjectMeta{ | 		ObjectMeta: *objectMeta, | ||||||
| 			GenerateName: rs.ObjectMeta.Name, | 		Spec:       rs.Spec.Template.Spec, | ||||||
| 			Namespace:    rs.ObjectMeta.Namespace, |  | ||||||
| 		}, |  | ||||||
| 		Spec: rs.Spec.Template, |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if err := ctrl.SetControllerReference(&rs, &runner, r.Scheme); err != nil { | 	if err := ctrl.SetControllerReference(&rs, &runner, r.Scheme); err != nil { | ||||||
|  |  | ||||||
|  | @ -98,11 +98,13 @@ var _ = Context("Inside of a new namespace", func() { | ||||||
| 					}, | 					}, | ||||||
| 					Spec: actionsv1alpha1.RunnerSetSpec{ | 					Spec: actionsv1alpha1.RunnerSetSpec{ | ||||||
| 						Replicas: intPtr(1), | 						Replicas: intPtr(1), | ||||||
| 						Template: actionsv1alpha1.RunnerSpec{ | 						Template: actionsv1alpha1.RunnerTemplate{ | ||||||
| 							Repository: "foo/bar", | 							Spec: actionsv1alpha1.RunnerSpec{ | ||||||
| 							Image:      "bar", | 								Repository: "foo/bar", | ||||||
| 							Env: []corev1.EnvVar{ | 								Image:      "bar", | ||||||
| 								{Name: "FOO", Value: "FOOVALUE"}, | 								Env: []corev1.EnvVar{ | ||||||
|  | 									{Name: "FOO", Value: "FOOVALUE"}, | ||||||
|  | 								}, | ||||||
| 							}, | 							}, | ||||||
| 						}, | 						}, | ||||||
| 					}, | 					}, | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								go.mod
								
								
								
								
							
							
						
						
									
										1
									
								
								go.mod
								
								
								
								
							|  | @ -6,6 +6,7 @@ require ( | ||||||
| 	github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751 // indirect | 	github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751 // indirect | ||||||
| 	github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d // indirect | 	github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d // indirect | ||||||
| 	github.com/bradleyfalzon/ghinstallation v1.1.1 | 	github.com/bradleyfalzon/ghinstallation v1.1.1 | ||||||
|  | 	github.com/davecgh/go-spew v1.1.1 | ||||||
| 	github.com/go-logr/logr v0.1.0 | 	github.com/go-logr/logr v0.1.0 | ||||||
| 	github.com/google/go-github v17.0.0+incompatible | 	github.com/google/go-github v17.0.0+incompatible | ||||||
| 	github.com/google/go-github/v29 v29.0.2 | 	github.com/google/go-github/v29 v29.0.2 | ||||||
|  |  | ||||||
							
								
								
									
										11
									
								
								main.go
								
								
								
								
							
							
						
						
									
										11
									
								
								main.go
								
								
								
								
							|  | @ -121,6 +121,17 @@ func main() { | ||||||
| 		setupLog.Error(err, "unable to create controller", "controller", "RunnerSet") | 		setupLog.Error(err, "unable to create controller", "controller", "RunnerSet") | ||||||
| 		os.Exit(1) | 		os.Exit(1) | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	runnerDeploymentReconciler := &controllers.RunnerDeploymentReconciler{ | ||||||
|  | 		Client: mgr.GetClient(), | ||||||
|  | 		Log:    ctrl.Log.WithName("controllers").WithName("RunnerDeployment"), | ||||||
|  | 		Scheme: mgr.GetScheme(), | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if err = runnerDeploymentReconciler.SetupWithManager(mgr); err != nil { | ||||||
|  | 		setupLog.Error(err, "unable to create controller", "controller", "RunnerDeployment") | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
| 	// +kubebuilder:scaffold:builder
 | 	// +kubebuilder:scaffold:builder
 | ||||||
| 
 | 
 | ||||||
| 	setupLog.Info("starting manager") | 	setupLog.Info("starting manager") | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue