From b3662779984a39c7cdb31b27ce9d0d63b3e930f4 Mon Sep 17 00:00:00 2001 From: Chris Doherty Date: Tue, 21 Feb 2023 22:09:12 -0600 Subject: [PATCH] Add v1alpha2 API Signed-off-by: Chris Doherty --- Makefile | 2 +- api/v1alpha2/conditions.go | 45 ++ api/v1alpha2/groupversion_info.go | 20 + api/v1alpha2/hardware.go | 178 +++++ api/v1alpha2/osie.go | 36 + api/v1alpha2/template.go | 96 +++ api/v1alpha2/workflow.go | 142 ++++ api/v1alpha2/zz_generated.deepcopy.go | 666 ++++++++++++++++++ config/crd/bases/tinkerbell.org_hardware.yaml | 133 +++- config/crd/bases/tinkerbell.org_osies.yaml | 57 ++ .../crd/bases/tinkerbell.org_templates.yaml | 71 +- .../crd/bases/tinkerbell.org_workflows.yaml | 171 ++++- 12 files changed, 1613 insertions(+), 4 deletions(-) create mode 100644 api/v1alpha2/conditions.go create mode 100644 api/v1alpha2/groupversion_info.go create mode 100644 api/v1alpha2/hardware.go create mode 100644 api/v1alpha2/osie.go create mode 100644 api/v1alpha2/template.go create mode 100644 api/v1alpha2/workflow.go create mode 100644 api/v1alpha2/zz_generated.deepcopy.go create mode 100644 config/crd/bases/tinkerbell.org_osies.yaml diff --git a/Makefile b/Makefile index 9c562fced..7c166d104 100644 --- a/Makefile +++ b/Makefile @@ -19,7 +19,7 @@ GOPROXY ?= $(shell go env GOPROXY) # Runnable tools GO ?= go BUF := $(GO) run github.com/bufbuild/buf/cmd/buf@v1.11 -CONTROLLER_GEN := $(GO) run sigs.k8s.io/controller-tools/cmd/controller-gen@v0.10 +CONTROLLER_GEN := $(GO) run sigs.k8s.io/controller-tools/cmd/controller-gen@v0.11 GOFUMPT := $(GO) run mvdan.cc/gofumpt@v0.4 KUSTOMIZE := $(GO) run sigs.k8s.io/kustomize/kustomize/v4@v4.5 SETUP_ENVTEST := $(GO) run sigs.k8s.io/controller-runtime/tools/setup-envtest@v0.0.0-20220304125252-9ee63fc65a97 diff --git a/api/v1alpha2/conditions.go b/api/v1alpha2/conditions.go new file mode 100644 index 000000000..e1e1e1b75 --- /dev/null +++ b/api/v1alpha2/conditions.go @@ -0,0 +1,45 @@ +package v1alpha2 + +import metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + +// ConditionType identifies the type of condition. +type ConditionType string + +// ConditionStatus expresses the current state of the condition. +type ConditionStatus string + +const ( + // ConditionStatusUnknown is the default status and indicates the condition cannot be + // evaluated as True or False. + ConditionStatusUnknown ConditionStatus = "Unknown" + + // ConditionStatusTrue indicates the condition has been evaluated as true. + ConditionStatusTrue ConditionStatus = "True" + + // ConditionStatusFalse indicates the condition has been evaluated as false. + ConditionStatusFalse ConditionStatus = "False" +) + +// Condition defines an observation on a resource that is generally attainable by inspecting +// other status fields. +type Condition struct { + // Type of condition. + Type ConditionType `json:"type"` + + // Status of the condition. + Status ConditionStatus `json:"status"` + + // LastTransition is the last time the condition transitioned from one status to another. + LastTransition *metav1.Time `json:"lastTransitionTime"` + + // Reason is a short CamelCase description for the conditions last transition. + // +optional + Reason *string `json:"reason,omitempty"` + + // Message is a human readable message indicating details about the last transition. + // +optional + Message *string `json:"message,omitempty"` +} + +// Conditions define a list of observations of a particular resource. +type Conditions []Condition diff --git a/api/v1alpha2/groupversion_info.go b/api/v1alpha2/groupversion_info.go new file mode 100644 index 000000000..e66aab567 --- /dev/null +++ b/api/v1alpha2/groupversion_info.go @@ -0,0 +1,20 @@ +// +groupName=tinkerbell.org +// +kubebuilder:object:generate=true + +package v1alpha2 + +import ( + "k8s.io/apimachinery/pkg/runtime/schema" + "sigs.k8s.io/controller-runtime/pkg/scheme" +) + +var ( + // GroupVersion is group version used to register these objects. + GroupVersion = schema.GroupVersion{Group: "tinkerbell.org", Version: "v1alpha2"} + + // SchemeBuilder is used to add go types to the GroupVersionKind scheme. + SchemeBuilder = &scheme.Builder{GroupVersion: GroupVersion} + + // AddToScheme adds the types in this group-version to the given scheme. + AddToScheme = SchemeBuilder.AddToScheme +) diff --git a/api/v1alpha2/hardware.go b/api/v1alpha2/hardware.go new file mode 100644 index 000000000..51beff7aa --- /dev/null +++ b/api/v1alpha2/hardware.go @@ -0,0 +1,178 @@ +package v1alpha2 + +import ( + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" +) + +type HardwareSpec struct { + // NetworkInterfaces defines the desired DHCP and netboot configuration for a network interface. + // +kubebuilder:validation:MinPoperties=1 + NetworkInterfaces NetworkInterfaces `json:"networkInterfaces,omitempty"` + + // IPXE provides iPXE script override fields. This is useful for debugging or netboot + // customization. + // +optional. + IPXE *IPXE `json:"ipxe,omitempty"` + + // OSIE describes the Operating System Installation Environment to be netbooted. + OSIE corev1.LocalObjectReference `json:"osie,omitempty"` + + // KernelParams passed to the kernel when launching the OSIE. Parameters are joined with a + // space. + // +optional + KernelParams []string `json:"kernelParams,omitempty"` + + // Instance describes instance specific data that is generally unused by Tinkerbell core. + // +optional + Instance *Instance `json:"instance,omitempty"` + + // StorageDevices is a list of storage devices that will be available in the OSIE. + // +optional. + StorageDevices []StorageDevice `json:"storageDevices,omitempty"` + + // BMCRef references a Rufio Machine object. + // +optional. + BMCRef *corev1.LocalObjectReference `json:"bmcRef,omitempty"` +} + +// NetworkInterfaces maps a MAC address to a NetworkInterface. +type NetworkInterfaces map[MAC]NetworkInterface + +// NetworkInterface is the desired configuration for a particular network interface. +type NetworkInterface struct { + // DHCP is the basic network information for serving DHCP requests. + DHCP DHCP `json:"dhcp,omitempty"` + + // DisableDHCP disables DHCP for this interface. Implies DisableNetboot. + // +kubebuilder:default=false + DisableDHCP bool `json:"disableDhcp,omitempty"` + + // DisableNetboot disables netbooting for this interface. The interface will still receive + // network information speified on by DHCP. + // +kubebuilder:default=false + DisableNetboot bool `json:"disableNetboot,omitempty"` +} + +// DHCP describes basic network configuration to be served in DHCP OFFER responses. It can be +// considered a DHCP reservation. +type DHCP struct { + // IP is an IPv4 address to serve. + // +kubebuilder:validation:Pattern=`(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}` + IP string `json:"ip,omitempty"` + + // Netmask is an IPv4 netmask to serve. + // +kubebuilder+validation:Pattern=`^(255)\.(0|128|192|224|240|248|252|254|255)\.(0|128|192|224|240|248|252|254|255)\.(0|128|192|224|240|248|252|254|255)` + Netmask string `json:"netmask,omitempty"` + + // Gateway is the default gateway address to serve. + // +kubebuilder:validation:Pattern=`(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}` + // +optional + Gateway *string `json:"gateway,omitempty"` + + // +kubebuilder:validation:Pattern=`^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9]"[A-Za-z0-9\-]*[A-Za-z0-9])$` + // +optional + Hostname *string `json:"hostname,omitempty"` + + // VLANID is a VLAN ID between 0 and 4096. + // +kubebuilder:validation:Pattern=`^(([0-9][0-9]{0,2}|[1-3][0-9][0-9][0-9]|40([0-8][0-9]|9[0-6]))(,[1-9][0-9]{0,2}|[1-3][0-9][0-9][0-9]|40([0-8][0-9]|9[0-6]))*)$` + // +optional + VLANID *string `json:"vlanId,omitempty"` + + // Nameservers to serve. + // +optional + Nameservers []Nameserver `json:"nameservers,omitempty"` + + // Timeservers to serve. + // +optional + Timeservers []Timeserver `json:"timeservers,omitempty"` + + // LeaseTimeSeconds to serve. 24h default. Maximum equates to max uint32 as defined by RFC 2132 + // § 9.2 (https://www.rfc-editor.org/rfc/rfc2132.html#section-9.2). + // +kubebuilder:default=86400 + // +kubebuilder:validation:Minimum=0 + // +kubebuilder:validation:Maximum=4294967295 + // +optional + LeaseTimeSeconds *int64 `json:"leaseTimeSeconds"` +} + +// IPXE describes overrides for IPXE scripts. At least 1 option must be specified. +type IPXE struct { + // Content is an inline iPXE script. + // +optional + Content *string `json:"inline,omitempty"` + + // URL is a URL to a hosted iPXE script. + // +optional + URL *string `json:"url,omitempty"` +} + +// Instance describes instance specific data. Instance specific data is typically dependent on the +// permanent OS that a piece of hardware runs. This data is often served by an instance metadata +// service such as Tinkerbell's Hegel. The core Tinkerbell stack does not leverage this data. +type Instance struct { + // Userdata is data with a structure understood by the producer and consumer of the data. + // +optional + Userdata *string `json:"userdata,omitempty"` + + // Vendordata is data with a structure understood by the producer and consumer of the data. + // +optional + Vendordata *string `json:"vendordata,omitempty"` +} + +// MAC is a Media Access Control address. MACs must use lower case letters. +// +kubebuilder:validation:Pattern=`^([0-9a-f]{2}:){5}([0-9a-f]{2})$` +type MAC string + +// Nameserver is an IP or hostname. +// +kubebuilder:validation:Pattern=`^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])$|^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$` +type Nameserver string + +// Timeserver is an IP or hostname. +// +kubebuilder:validation:Pattern=`^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])$|^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$` +type Timeserver string + +// StorageDevice describes a storage device path that will be present in the OSIE. +// StorageDevices must be valid Linux paths. They should not contain partitions. +// +// Good +// +// /dev/sda +// /dev/nvme0n1 +// +// Bad (contains partitions) +// +// /dev/sda1 +// /dev/nvme0n1p1 +// +// Bad (invalid Linux path) +// +// \dev\sda +// +// +kubebuilder:validation:Pattern=`^(/[^/ ]*)+/?$` +type StorageDevice string + +// +kubebuilder:object:root=true +// +kubebuilder:resource:categories=tinkerbell,path=hardware,shortName=hw +// +kubebuilder:printcolumn:name="BMC",type="string",JSONPath=".spec.bmcRef",description="Baseboard management computer attached to the Hardware" +// +kubebuilder:unservedversion + +// Hardware is a logical representation of a machine that can execute Workflows. +type Hardware struct { + metav1.TypeMeta `json:",inline"` + metav1.ObjectMeta `json:"metadata,omitempty"` + + Spec HardwareSpec `json:"spec,omitempty"` +} + +// +kubebuilder:object:root=true + +type HardwareList struct { + metav1.TypeMeta `json:",inline"` + metav1.ListMeta `json:"metadata,omitempty"` + Items []Hardware `json:"items"` +} + +func init() { + SchemeBuilder.Register(&Hardware{}, &HardwareList{}) +} diff --git a/api/v1alpha2/osie.go b/api/v1alpha2/osie.go new file mode 100644 index 000000000..e2c4c603d --- /dev/null +++ b/api/v1alpha2/osie.go @@ -0,0 +1,36 @@ +package v1alpha2 + +import metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + +type OSIESpec struct { + // KernelURL is a URL to a kernel image. + KernelURL string `json:"kernelUrl,omitempty"` + + // InitrdURL is a URL to an initrd image. + InitrdURL string `json:"initrdUrl,omitempty"` +} + +// +kubebuilder:object:root=true +// +kubebuilder:storageversion +// +kubebuilder:resource:categories=tinkerbell + +// OSIE describes an Operating System Installation Environment. It is used by Tinkerbell +// to provision machines and should launch the Tink Worker component. +type OSIE struct { + metav1.TypeMeta `json:",inline"` + metav1.ObjectMeta `json:"metadata,omitempty"` + + Spec OSIESpec `json:"spec,omitempty"` +} + +// +kubebuilder:object:root=true + +type OSIEList struct { + metav1.TypeMeta `json:",inline"` + metav1.ListMeta `json:"metadata,omitempty"` + Items []OSIE `json:"items"` +} + +func init() { + SchemeBuilder.Register(&OSIE{}, &OSIEList{}) +} diff --git a/api/v1alpha2/template.go b/api/v1alpha2/template.go new file mode 100644 index 000000000..f8bff8975 --- /dev/null +++ b/api/v1alpha2/template.go @@ -0,0 +1,96 @@ +package v1alpha2 + +import ( + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" +) + +type TemplateSpec struct { + // Actions defines the set of actions to be run on a target machine. Actions are run sequentially + // in the order they are specified. At least 1 action must be specified. Names of actions + // must be unique within a Template. + // +kubebuilder:validation:MinItems=1 + Actions []Action `json:"actions,omitempty"` + + // Volumes to be mounted on all actions. If an action specifies the same volume it will take + // precedence. + // +optional + Volumes []Volume `json:"volumes,omitempty"` + + // Env defines environment variables to be available in all actions. If an action specifies + // the same environment variable it will take precedence. + // +optional + Env map[string]string `json:"env,omitempty"` +} + +// Action defines an individual action to be run on a target machine. +type Action struct { + // Name is a name for the action. + Name string `json:"name"` + + // Image is an OCI image. + Image string `json:"image"` + + // Cmd defines the command to use when launching the image. + // +optional + Cmd *string `json:"cmd,omitempty"` + + // Args are a set of arguments to be passed to the container on launch. + // +optional + Args []string `json:"args,omitempty"` + + // Env defines environment variables used when launching the container. + //+optional + Env map[string]string `json:"env,omitempty"` + + // Volumes defines the volumes to mount into the container. + // +optional + Volumes []Volume `json:"volumes,omitempty"` + + // NetworkNamespace defines the network namespace to run the container in. This enables access + // to the host network namespace. + // See https://man7.org/linux/man-pages/man7/namespaces.7.html. + // +optional + NetworkNamespace *string `json:"networkNamespace,omitempty"` +} + +// Volume is a specification for mounting a volume in an action. Volumes take the form +// {SRC-VOLUME-NAME | SRC-HOST-DIR}:TGT-CONTAINER-DIR:OPTIONS. When specifying a VOLUME-NAME that +// does not exist it will be created for you. Examples: +// +// Read-only bind mount bound to /data +// +// /etc/data:/data:ro +// +// Writable volume name bound to /data +// +// shared_volume:/data +// +// See https://docs.docker.com/storage/volumes/ for additional details. +type Volume string + +// +kubebuilder:object:root=true +// +kubebuilder:resource:categories=tinkerbell,shortName=tpl +// +kubebuilder:unservedversion + +// Template defines a set of actions to be run on a target machine. The template is rendered +// prior to execution where it is exposed to Hardware and user defined data. Most fields within the +// TemplateSpec may contain templates values excluding .TemplateSpec.Actions[].Name. +// See https://pkg.go.dev/text/template for more details. +type Template struct { + metav1.TypeMeta `json:",inline"` + metav1.ObjectMeta `json:"metadata,omitempty"` + + Spec TemplateSpec `json:"spec,omitempty"` +} + +// +kubebuilder:object:root=true + +type TemplateList struct { + metav1.TypeMeta `json:",inline"` + metav1.ListMeta `json:"metadata,omitempty"` + Items []Template `json:"items"` +} + +func init() { + SchemeBuilder.Register(&Template{}, &TemplateList{}) +} diff --git a/api/v1alpha2/workflow.go b/api/v1alpha2/workflow.go new file mode 100644 index 000000000..577fa4090 --- /dev/null +++ b/api/v1alpha2/workflow.go @@ -0,0 +1,142 @@ +package v1alpha2 + +import ( + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" +) + +type WorkflowSpec struct { + // HardwareRef is a reference to a Hardware resource this workflow will execute on. + // If no namespace is specified the Workflow's namespace is assumed. + HardwareRef corev1.LocalObjectReference `json:"hardwareRef,omitempty"` + + // TemplateRef is a reference to a Template resource used to render workflow actions. + // If no namespace is specified the Workflow's namespace is assumed. + TemplateRef corev1.LocalObjectReference `json:"templateRef,omitempty"` + + // TemplateParams are a list of key-value pairs that are injected into templates at render + // time. TemplateParams are exposed to templates using a top level .Params key. + // + // For example, TemplateParams = {"foo": "bar"}, the foo key can be accessed via .Params.foo. + // +optional + TemplateParams map[string]string `json:"templateParams,omitempty"` + + // TimeoutSeconds defines the time the workflow has to complete. The timer begins when the first + // action is requested. When set to 0, no timeout is applied. + // +kubebuilder:default=0 + // +kubebuilder:validation:Minimum=0 + TimeoutSeconds int64 `json:"timeout,omitempty"` +} + +type WorkflowStatus struct { + // Actions is a list of action states. + Actions []ActionStatus `json:"actions"` + + // StartedAt is the time the first action was requested. Nil indicates the Workflow has not + // started. + StartedAt *metav1.Time `json:"startedAt,omitempty"` + + // LastTransition is the observed time when State transitioned last. + LastTransition *metav1.Time `json:"lastTransitioned,omitempty"` + + // State describes the current state of the workflow. For the workflow to enter the + // WorkflowStateSucceeded state all actions must be in ActionStateSucceeded. The Workflow will + // enter a WorkflowStateFailed if 1 or more Actions fails. + State WorkflowState `json:"state,omitempty"` + + // Conditions details a set of observations about the Workflow. + Conditions Conditions `json:"conditions"` +} + +// ActionStatus describes status information about an action. +type ActionStatus struct { + // Rendered is the rendered action. + Rendered Action `json:"rendered,omitempty"` + + // ID uniquely identifies the action status. + ID string `json:"id"` + + // StartedAt is the time the action was started as reported by the client. Nil indicates the + // Action has not started. + StartedAt *metav1.Time `json:"startedAt,omitempty"` + + // LastTransition is the observed time when State transitioned last. + LastTransition *metav1.Time `json:"lastTransitioned,omitempty"` + + // State describes the current state of the action. + State ActionState `json:"state,omitempty"` + + // FailureReason is a short CamelCase word or phrase describing why the Action entered + // ActionStateFailed. + FailureReason string `json:"failureReason,omitempty"` + + // FailureMessage is a free-form user friendly message describing why the Action entered the + // ActionStateFailed state. Typically, this is an elaboration on the Reason. + FailureMessage string `json:"failureMessage,omitempty"` +} + +// State describes the point in time state of a Workflow. +type WorkflowState string + +const ( + // WorkflowStatePending indicates the workflow is in a pending state. + WorkflowStatePending WorkflowState = "Pending" + + // WorkflowStateRunning indicates the first Action has been requested and the Workflow is in + // progress. + WorkflowStateRunning WorkflowState = "Running" + + // WorkflowStateSucceeded indicates all Workflow actions have successfully completed. + WorkflowStateSucceeded WorkflowState = "Succeeded" + + // WorkflowStateFailed indicates an Action entered a failure state. + WorkflowStateFailed WorkflowState = "Failed" +) + +// ActionState describes a point in time state of an Action. +type ActionState string + +const ( + // ActionStatePending indicates an Action is awaiting execution. + ActionStatePending ActionState = "Pending" + + // ActionStateRunning indicates an Action has begun execution. + ActionStateRunning ActionState = "Running" + + // ActionStateSucceeded indicates an Action completed execution successfully. + ActionStateSucceeded ActionState = "Succeeded" + + // ActionStatFailed indicates an Action failed to execute. Users may inspect the associated + // Workflow resource to gain deeper insights into why the action failed. + ActionStateFailed ActionState = "Failed" +) + +// +kubebuilder:object:root=true +// +kubebuilder:subresource:status +// +kubebuilder:resource:categories=tinkerbell,shortName=wf +// +kubebuilder:printcolumn:name="State",type="string",JSONPath=".status.state",description="State of the workflow such as Pending,Running etc" +// +kubebuilder:printcolumn:name="Hardware",type="string",JSONPath=".spec.hardwareRef",description="Hardware object that runs the workflow" +// +kubebuilder:printcolumn:name="Template",type="string",JSONPath=".spec.templateRef",description="Template to run on the associated Hardware" +// +kubebuilder:unservedversion + +// Workflow describes a set of actions to be run on a specific Hardware. Workflows execute +// once and should be considered ephemeral. +type Workflow struct { + metav1.TypeMeta `json:",inline"` + metav1.ObjectMeta `json:"metadata,omitempty"` + + Spec WorkflowSpec `json:"spec,omitempty"` + Status WorkflowStatus `json:"status,omitempty"` +} + +// +kubebuilder:object:root=true + +type WorkflowList struct { + metav1.TypeMeta `json:",inline"` + metav1.ListMeta `json:"metadata,omitempty"` + Items []Workflow `json:"items,omitempty"` +} + +func init() { + SchemeBuilder.Register(&Workflow{}, &WorkflowList{}) +} diff --git a/api/v1alpha2/zz_generated.deepcopy.go b/api/v1alpha2/zz_generated.deepcopy.go new file mode 100644 index 000000000..eae95c656 --- /dev/null +++ b/api/v1alpha2/zz_generated.deepcopy.go @@ -0,0 +1,666 @@ +//go:build !ignore_autogenerated +// +build !ignore_autogenerated + +/* +Copyright The Tinkerbell 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. +*/ + +// Code generated by controller-gen. DO NOT EDIT. + +package v1alpha2 + +import ( + "k8s.io/api/core/v1" + runtime "k8s.io/apimachinery/pkg/runtime" +) + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *Action) DeepCopyInto(out *Action) { + *out = *in + if in.Cmd != nil { + in, out := &in.Cmd, &out.Cmd + *out = new(string) + **out = **in + } + if in.Args != nil { + in, out := &in.Args, &out.Args + *out = make([]string, len(*in)) + copy(*out, *in) + } + if in.Env != nil { + in, out := &in.Env, &out.Env + *out = make(map[string]string, len(*in)) + for key, val := range *in { + (*out)[key] = val + } + } + if in.Volumes != nil { + in, out := &in.Volumes, &out.Volumes + *out = make([]Volume, len(*in)) + copy(*out, *in) + } + if in.NetworkNamespace != nil { + in, out := &in.NetworkNamespace, &out.NetworkNamespace + *out = new(string) + **out = **in + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Action. +func (in *Action) DeepCopy() *Action { + if in == nil { + return nil + } + out := new(Action) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *ActionStatus) DeepCopyInto(out *ActionStatus) { + *out = *in + in.Rendered.DeepCopyInto(&out.Rendered) + if in.StartedAt != nil { + in, out := &in.StartedAt, &out.StartedAt + *out = (*in).DeepCopy() + } + if in.LastTransition != nil { + in, out := &in.LastTransition, &out.LastTransition + *out = (*in).DeepCopy() + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ActionStatus. +func (in *ActionStatus) DeepCopy() *ActionStatus { + if in == nil { + return nil + } + out := new(ActionStatus) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *Condition) DeepCopyInto(out *Condition) { + *out = *in + if in.LastTransition != nil { + in, out := &in.LastTransition, &out.LastTransition + *out = (*in).DeepCopy() + } + if in.Reason != nil { + in, out := &in.Reason, &out.Reason + *out = new(string) + **out = **in + } + if in.Message != nil { + in, out := &in.Message, &out.Message + *out = new(string) + **out = **in + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Condition. +func (in *Condition) DeepCopy() *Condition { + if in == nil { + return nil + } + out := new(Condition) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in Conditions) DeepCopyInto(out *Conditions) { + { + in := &in + *out = make(Conditions, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Conditions. +func (in Conditions) DeepCopy() Conditions { + if in == nil { + return nil + } + out := new(Conditions) + in.DeepCopyInto(out) + return *out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *DHCP) DeepCopyInto(out *DHCP) { + *out = *in + if in.Gateway != nil { + in, out := &in.Gateway, &out.Gateway + *out = new(string) + **out = **in + } + if in.Hostname != nil { + in, out := &in.Hostname, &out.Hostname + *out = new(string) + **out = **in + } + if in.VLANID != nil { + in, out := &in.VLANID, &out.VLANID + *out = new(string) + **out = **in + } + if in.Nameservers != nil { + in, out := &in.Nameservers, &out.Nameservers + *out = make([]Nameserver, len(*in)) + copy(*out, *in) + } + if in.Timeservers != nil { + in, out := &in.Timeservers, &out.Timeservers + *out = make([]Timeserver, len(*in)) + copy(*out, *in) + } + if in.LeaseTimeSeconds != nil { + in, out := &in.LeaseTimeSeconds, &out.LeaseTimeSeconds + *out = new(int64) + **out = **in + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new DHCP. +func (in *DHCP) DeepCopy() *DHCP { + if in == nil { + return nil + } + out := new(DHCP) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *Hardware) DeepCopyInto(out *Hardware) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ObjectMeta.DeepCopyInto(&out.ObjectMeta) + in.HardwareSpec.DeepCopyInto(&out.HardwareSpec) +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Hardware. +func (in *Hardware) DeepCopy() *Hardware { + if in == nil { + return nil + } + out := new(Hardware) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *Hardware) 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 *HardwareList) DeepCopyInto(out *HardwareList) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ListMeta.DeepCopyInto(&out.ListMeta) + if in.Items != nil { + in, out := &in.Items, &out.Items + *out = make([]Hardware, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HardwareList. +func (in *HardwareList) DeepCopy() *HardwareList { + if in == nil { + return nil + } + out := new(HardwareList) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *HardwareList) 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 *HardwareSpec) DeepCopyInto(out *HardwareSpec) { + *out = *in + if in.NetworkInterfaces != nil { + in, out := &in.NetworkInterfaces, &out.NetworkInterfaces + *out = make(NetworkInterfaces, len(*in)) + for key, val := range *in { + (*out)[key] = *val.DeepCopy() + } + } + if in.IPXE != nil { + in, out := &in.IPXE, &out.IPXE + *out = new(IPXE) + (*in).DeepCopyInto(*out) + } + out.OSIE = in.OSIE + if in.KernelParams != nil { + in, out := &in.KernelParams, &out.KernelParams + *out = make([]string, len(*in)) + copy(*out, *in) + } + if in.Instance != nil { + in, out := &in.Instance, &out.Instance + *out = new(Instance) + (*in).DeepCopyInto(*out) + } + if in.StorageDevices != nil { + in, out := &in.StorageDevices, &out.StorageDevices + *out = make([]StorageDevice, len(*in)) + copy(*out, *in) + } + if in.BMCRef != nil { + in, out := &in.BMCRef, &out.BMCRef + *out = new(v1.LocalObjectReference) + **out = **in + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new HardwareSpec. +func (in *HardwareSpec) DeepCopy() *HardwareSpec { + if in == nil { + return nil + } + out := new(HardwareSpec) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *IPXE) DeepCopyInto(out *IPXE) { + *out = *in + if in.Content != nil { + in, out := &in.Content, &out.Content + *out = new(string) + **out = **in + } + if in.URL != nil { + in, out := &in.URL, &out.URL + *out = new(string) + **out = **in + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new IPXE. +func (in *IPXE) DeepCopy() *IPXE { + if in == nil { + return nil + } + out := new(IPXE) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *Instance) DeepCopyInto(out *Instance) { + *out = *in + if in.Userdata != nil { + in, out := &in.Userdata, &out.Userdata + *out = new(string) + **out = **in + } + if in.Vendordata != nil { + in, out := &in.Vendordata, &out.Vendordata + *out = new(string) + **out = **in + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Instance. +func (in *Instance) DeepCopy() *Instance { + if in == nil { + return nil + } + out := new(Instance) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *NetworkInterface) DeepCopyInto(out *NetworkInterface) { + *out = *in + in.DHCP.DeepCopyInto(&out.DHCP) +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new NetworkInterface. +func (in *NetworkInterface) DeepCopy() *NetworkInterface { + if in == nil { + return nil + } + out := new(NetworkInterface) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in NetworkInterfaces) DeepCopyInto(out *NetworkInterfaces) { + { + in := &in + *out = make(NetworkInterfaces, len(*in)) + for key, val := range *in { + (*out)[key] = *val.DeepCopy() + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new NetworkInterfaces. +func (in NetworkInterfaces) DeepCopy() NetworkInterfaces { + if in == nil { + return nil + } + out := new(NetworkInterfaces) + in.DeepCopyInto(out) + return *out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *OSIE) DeepCopyInto(out *OSIE) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ObjectMeta.DeepCopyInto(&out.ObjectMeta) + out.Spec = in.Spec +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new OSIE. +func (in *OSIE) DeepCopy() *OSIE { + if in == nil { + return nil + } + out := new(OSIE) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *OSIE) 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 *OSIEList) DeepCopyInto(out *OSIEList) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ListMeta.DeepCopyInto(&out.ListMeta) + if in.Items != nil { + in, out := &in.Items, &out.Items + *out = make([]OSIE, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new OSIEList. +func (in *OSIEList) DeepCopy() *OSIEList { + if in == nil { + return nil + } + out := new(OSIEList) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *OSIEList) 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 *OSIESpec) DeepCopyInto(out *OSIESpec) { + *out = *in +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new OSIESpec. +func (in *OSIESpec) DeepCopy() *OSIESpec { + if in == nil { + return nil + } + out := new(OSIESpec) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *Template) DeepCopyInto(out *Template) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ObjectMeta.DeepCopyInto(&out.ObjectMeta) + in.Spec.DeepCopyInto(&out.Spec) +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Template. +func (in *Template) DeepCopy() *Template { + if in == nil { + return nil + } + out := new(Template) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *Template) 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 *TemplateList) DeepCopyInto(out *TemplateList) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ListMeta.DeepCopyInto(&out.ListMeta) + if in.Items != nil { + in, out := &in.Items, &out.Items + *out = make([]Template, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new TemplateList. +func (in *TemplateList) DeepCopy() *TemplateList { + if in == nil { + return nil + } + out := new(TemplateList) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *TemplateList) 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 *TemplateSpec) DeepCopyInto(out *TemplateSpec) { + *out = *in + if in.Actions != nil { + in, out := &in.Actions, &out.Actions + *out = make([]Action, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } + if in.Volumes != nil { + in, out := &in.Volumes, &out.Volumes + *out = make([]Volume, len(*in)) + copy(*out, *in) + } + if in.Env != nil { + in, out := &in.Env, &out.Env + *out = make(map[string]string, len(*in)) + for key, val := range *in { + (*out)[key] = val + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new TemplateSpec. +func (in *TemplateSpec) DeepCopy() *TemplateSpec { + if in == nil { + return nil + } + out := new(TemplateSpec) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *Workflow) DeepCopyInto(out *Workflow) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ObjectMeta.DeepCopyInto(&out.ObjectMeta) + in.Spec.DeepCopyInto(&out.Spec) + in.Status.DeepCopyInto(&out.Status) +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Workflow. +func (in *Workflow) DeepCopy() *Workflow { + if in == nil { + return nil + } + out := new(Workflow) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *Workflow) 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 *WorkflowList) DeepCopyInto(out *WorkflowList) { + *out = *in + out.TypeMeta = in.TypeMeta + in.ListMeta.DeepCopyInto(&out.ListMeta) + if in.Items != nil { + in, out := &in.Items, &out.Items + *out = make([]Workflow, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new WorkflowList. +func (in *WorkflowList) DeepCopy() *WorkflowList { + if in == nil { + return nil + } + out := new(WorkflowList) + in.DeepCopyInto(out) + return out +} + +// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object. +func (in *WorkflowList) 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 *WorkflowSpec) DeepCopyInto(out *WorkflowSpec) { + *out = *in + out.HardwareRef = in.HardwareRef + out.TemplateRef = in.TemplateRef + if in.TemplateParams != nil { + in, out := &in.TemplateParams, &out.TemplateParams + *out = make(map[string]string, len(*in)) + for key, val := range *in { + (*out)[key] = val + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new WorkflowSpec. +func (in *WorkflowSpec) DeepCopy() *WorkflowSpec { + if in == nil { + return nil + } + out := new(WorkflowSpec) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *WorkflowStatus) DeepCopyInto(out *WorkflowStatus) { + *out = *in + if in.Actions != nil { + in, out := &in.Actions, &out.Actions + *out = make([]ActionStatus, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } + if in.StartedAt != nil { + in, out := &in.StartedAt, &out.StartedAt + *out = (*in).DeepCopy() + } + if in.LastTransition != nil { + in, out := &in.LastTransition, &out.LastTransition + *out = (*in).DeepCopy() + } + if in.Conditions != nil { + in, out := &in.Conditions, &out.Conditions + *out = make(Conditions, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new WorkflowStatus. +func (in *WorkflowStatus) DeepCopy() *WorkflowStatus { + if in == nil { + return nil + } + out := new(WorkflowStatus) + in.DeepCopyInto(out) + return out +} diff --git a/config/crd/bases/tinkerbell.org_hardware.yaml b/config/crd/bases/tinkerbell.org_hardware.yaml index c23b8edd2..21ae87c00 100644 --- a/config/crd/bases/tinkerbell.org_hardware.yaml +++ b/config/crd/bases/tinkerbell.org_hardware.yaml @@ -2,7 +2,7 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.10.0 + controller-gen.kubebuilder.io/version: v0.11.3 creationTimestamp: null name: hardware.tinkerbell.org spec: @@ -363,3 +363,134 @@ spec: storage: true subresources: status: {} + - additionalPrinterColumns: + - description: Baseboard management computer attached to the Hardware + jsonPath: .spec.bmcRef + name: BMC + type: string + name: v1alpha2 + schema: + openAPIV3Schema: + description: Hardware is a logical representation of a machine that can execute Workflows. + 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: + properties: + bmcRef: + description: BMCRef references a Rufio Machine object. + properties: + 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 + type: object + x-kubernetes-map-type: atomic + instance: + description: Instance describes instance specific data that is generally unused by Tinkerbell core. + properties: + userdata: + description: Userdata is data with a structure understood by the producer and consumer of the data. + type: string + vendordata: + description: Vendordata is data with a structure understood by the producer and consumer of the data. + type: string + type: object + ipxe: + description: IPXE provides iPXE script override fields. This is useful for debugging or netboot customization. + properties: + inline: + description: Content is an inline iPXE script. + type: string + url: + description: URL is a URL to a hosted iPXE script. + type: string + type: object + kernelParams: + description: KernelParams passed to the kernel when launching the OSIE. Parameters are joined with a space. + items: + type: string + type: array + networkInterfaces: + additionalProperties: + description: NetworkInterface is the desired configuration for a particular network interface. + properties: + dhcp: + description: DHCP is the basic network information for serving DHCP requests. + properties: + gateway: + description: Gateway is the default gateway address to serve. + pattern: (25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3} + type: string + hostname: + pattern: ^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9]"[A-Za-z0-9\-]*[A-Za-z0-9])$ + type: string + ip: + description: IP is an IPv4 address to serve. + pattern: (25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3} + type: string + leaseTimeSeconds: + default: 86400 + description: LeaseTimeSeconds to serve. 24h default. Maximum equates to max uint32 as defined by RFC 2132 § 9.2 (https://www.rfc-editor.org/rfc/rfc2132.html#section-9.2). + format: int64 + maximum: 4294967295 + minimum: 0 + type: integer + nameservers: + description: Nameservers to serve. + items: + description: Nameserver is an IP or hostname. + pattern: ^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])$|^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$ + type: string + type: array + netmask: + description: Netmask is an IPv4 netmask to serve. + type: string + timeservers: + description: Timeservers to serve. + items: + description: Timeserver is an IP or hostname. + pattern: ^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])$|^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$ + type: string + type: array + vlanId: + description: VLANID is a VLAN ID between 0 and 4096. + pattern: ^(([0-9][0-9]{0,2}|[1-3][0-9][0-9][0-9]|40([0-8][0-9]|9[0-6]))(,[1-9][0-9]{0,2}|[1-3][0-9][0-9][0-9]|40([0-8][0-9]|9[0-6]))*)$ + type: string + type: object + disableDhcp: + default: false + description: DisableDHCP disables DHCP for this interface. Implies DisableNetboot. + type: boolean + disableNetboot: + default: false + description: DisableNetboot disables netbooting for this interface. The interface will still receive network information speified on by DHCP. + type: boolean + type: object + description: NetworkInterfaces defines the desired DHCP and netboot configuration for a network interface. + type: object + osie: + description: OSIE describes the Operating System Installation Environment to be netbooted. + properties: + 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 + type: object + x-kubernetes-map-type: atomic + storageDevices: + description: StorageDevices is a list of storage devices that will be available in the OSIE. + items: + description: "StorageDevice describes a storage device path that will be present in the OSIE. StorageDevices must be valid Linux paths. They should not contain partitions. \n Good \n /dev/sda /dev/nvme0n1 \n Bad (contains partitions) \n /dev/sda1 /dev/nvme0n1p1 \n Bad (invalid Linux path) \n \\dev\\sda" + pattern: ^(/[^/ ]*)+/?$ + type: string + type: array + type: object + type: object + served: false + storage: false + subresources: {} diff --git a/config/crd/bases/tinkerbell.org_osies.yaml b/config/crd/bases/tinkerbell.org_osies.yaml new file mode 100644 index 000000000..dfd2ba236 --- /dev/null +++ b/config/crd/bases/tinkerbell.org_osies.yaml @@ -0,0 +1,57 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.3 + creationTimestamp: null + name: osies.tinkerbell.org +spec: + group: tinkerbell.org + names: + categories: + - tinkerbell + kind: OSIE + listKind: OSIEList + plural: osies + singular: osie + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: OSIE configuration. + properties: + baseURL: + type: string + initrd: + type: string + kernel: + type: string + type: object + served: true + storage: false + - name: v1alpha2 + schema: + openAPIV3Schema: + description: OSIE describes an Operating System Installation Environment. It is used by Tinkerbell to provision machines and should launch the Tink Worker component. + 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: + properties: + initrdUrl: + description: InitrdURL is a URL to an initrd image. + type: string + kernelUrl: + description: KernelURL is a URL to a kernel image. + type: string + type: object + type: object + served: true + storage: true diff --git a/config/crd/bases/tinkerbell.org_templates.yaml b/config/crd/bases/tinkerbell.org_templates.yaml index bff533719..24c0d5e62 100644 --- a/config/crd/bases/tinkerbell.org_templates.yaml +++ b/config/crd/bases/tinkerbell.org_templates.yaml @@ -2,7 +2,7 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.10.0 + controller-gen.kubebuilder.io/version: v0.11.3 creationTimestamp: null name: templates.tinkerbell.org spec: @@ -53,3 +53,72 @@ spec: storage: true subresources: status: {} + - name: v1alpha2 + schema: + openAPIV3Schema: + description: Template defines a set of actions to be run on a target machine. The template is rendered prior to execution where it is exposed to Hardware and user defined data. Most fields within the TemplateSpec may contain templates values excluding .TemplateSpec.Actions[].Name. See https://pkg.go.dev/text/template for more details. + 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: + properties: + actions: + description: Actions defines the set of actions to be run on a target machine. Actions are run sequentially in the order they are specified. At least 1 action must be specified. Names of actions must be unique within a Template. + items: + description: Action defines an individual action to be run on a target machine. + properties: + args: + description: Args are a set of arguments to be passed to the container on launch. + items: + type: string + type: array + cmd: + description: Cmd defines the command to use when launching the image. + type: string + env: + additionalProperties: + type: string + description: Env defines environment variables used when launching the container. + type: object + image: + description: Image is an OCI image. + type: string + name: + description: Name is a name for the action. + type: string + networkNamespace: + description: NetworkNamespace defines the network namespace to run the container in. This enables access to the host network namespace. See https://man7.org/linux/man-pages/man7/namespaces.7.html. + type: string + volumes: + description: Volumes defines the volumes to mount into the container. + items: + description: "Volume is a specification for mounting a volume in an action. Volumes take the form {SRC-VOLUME-NAME | SRC-HOST-DIR}:TGT-CONTAINER-DIR:OPTIONS. When specifying a VOLUME-NAME that does not exist it will be created for you. Examples: \n Read-only bind mount bound to /data \n /etc/data:/data:ro \n Writable volume name bound to /data \n shared_volume:/data \n See https://docs.docker.com/storage/volumes/ for additional details." + type: string + type: array + required: + - image + - name + type: object + minItems: 1 + type: array + env: + additionalProperties: + type: string + description: Env defines environment variables to be available in all actions. If an action specifies the same environment variable it will take precedence. + type: object + volumes: + description: Volumes to be mounted on all actions. If an action specifies the same volume it will take precedence. + items: + description: "Volume is a specification for mounting a volume in an action. Volumes take the form {SRC-VOLUME-NAME | SRC-HOST-DIR}:TGT-CONTAINER-DIR:OPTIONS. When specifying a VOLUME-NAME that does not exist it will be created for you. Examples: \n Read-only bind mount bound to /data \n /etc/data:/data:ro \n Writable volume name bound to /data \n shared_volume:/data \n See https://docs.docker.com/storage/volumes/ for additional details." + type: string + type: array + type: object + type: object + served: false + storage: false diff --git a/config/crd/bases/tinkerbell.org_workflows.yaml b/config/crd/bases/tinkerbell.org_workflows.yaml index 2307cfd2c..1a7f5d7e3 100644 --- a/config/crd/bases/tinkerbell.org_workflows.yaml +++ b/config/crd/bases/tinkerbell.org_workflows.yaml @@ -2,7 +2,7 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.10.0 + controller-gen.kubebuilder.io/version: v0.11.3 creationTimestamp: null name: workflows.tinkerbell.org spec: @@ -129,3 +129,172 @@ spec: storage: true subresources: status: {} + - additionalPrinterColumns: + - description: State of the workflow such as Pending,Running etc + jsonPath: .status.state + name: State + type: string + - description: Hardware object that runs the workflow + jsonPath: .spec.hardwareRef + name: Hardware + type: string + - description: Template to run on the associated Hardware + jsonPath: .spec.templateRef + name: Template + type: string + name: v1alpha2 + schema: + openAPIV3Schema: + description: Workflow describes a set of actions to be run on a specific Hardware. Workflows execute once and should be considered ephemeral. + 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: + properties: + hardwareRef: + description: HardwareRef is a reference to a Hardware resource this workflow will execute on. If no namespace is specified the Workflow's namespace is assumed. + properties: + 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 + type: object + x-kubernetes-map-type: atomic + templateParams: + additionalProperties: + type: string + description: "TemplateParams are a list of key-value pairs that are injected into templates at render time. TemplateParams are exposed to templates using a top level .Params key. \n For example, TemplateParams = {\"foo\": \"bar\"}, the foo key can be accessed via .Params.foo." + type: object + templateRef: + description: TemplateRef is a reference to a Template resource used to render workflow actions. If no namespace is specified the Workflow's namespace is assumed. + properties: + 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 + type: object + x-kubernetes-map-type: atomic + timeout: + default: 0 + description: TimeoutSeconds defines the time the workflow has to complete. The timer begins when the first action is requested. When set to 0, no timeout is applied. + format: int64 + minimum: 0 + type: integer + type: object + status: + properties: + actions: + description: Actions is a list of action states. + items: + description: ActionStatus describes status information about an action. + properties: + failureMessage: + description: FailureMessage is a free-form user friendly message describing why the Action entered the ActionStateFailed state. Typically, this is an elaboration on the Reason. + type: string + failureReason: + description: FailureReason is a short CamelCase word or phrase describing why the Action entered ActionStateFailed. + type: string + id: + description: ID uniquely identifies the action status. + type: string + lastTransitioned: + description: LastTransition is the observed time when State transitioned last. + format: date-time + type: string + rendered: + description: Rendered is the rendered action. + properties: + args: + description: Args are a set of arguments to be passed to the container on launch. + items: + type: string + type: array + cmd: + description: Cmd defines the command to use when launching the image. + type: string + env: + additionalProperties: + type: string + description: Env defines environment variables used when launching the container. + type: object + image: + description: Image is an OCI image. + type: string + name: + description: Name is a name for the action. + type: string + networkNamespace: + description: NetworkNamespace defines the network namespace to run the container in. This enables access to the host network namespace. See https://man7.org/linux/man-pages/man7/namespaces.7.html. + type: string + volumes: + description: Volumes defines the volumes to mount into the container. + items: + description: "Volume is a specification for mounting a volume in an action. Volumes take the form {SRC-VOLUME-NAME | SRC-HOST-DIR}:TGT-CONTAINER-DIR:OPTIONS. When specifying a VOLUME-NAME that does not exist it will be created for you. Examples: \n Read-only bind mount bound to /data \n /etc/data:/data:ro \n Writable volume name bound to /data \n shared_volume:/data \n See https://docs.docker.com/storage/volumes/ for additional details." + type: string + type: array + required: + - image + - name + type: object + startedAt: + description: StartedAt is the time the action was started as reported by the client. Nil indicates the Action has not started. + format: date-time + type: string + state: + description: State describes the current state of the action. + type: string + required: + - id + type: object + type: array + conditions: + description: Conditions details a set of observations about the Workflow. + items: + description: Condition defines an observation on a resource that is generally attainable by inspecting other status fields. + properties: + lastTransitionTime: + description: LastTransition is the last time the condition transitioned from one status to another. + format: date-time + type: string + message: + description: Message is a human readable message indicating details about the last transition. + type: string + reason: + description: Reason is a short CamelCase description for the conditions last transition. + type: string + status: + description: Status of the condition. + type: string + type: + description: Type of condition. + type: string + required: + - lastTransitionTime + - status + - type + type: object + type: array + lastTransitioned: + description: LastTransition is the observed time when State transitioned last. + format: date-time + type: string + startedAt: + description: StartedAt is the time the first action was requested. Nil indicates the Workflow has not started. + format: date-time + type: string + state: + description: State describes the current state of the workflow. For the workflow to enter the WorkflowStateSucceeded state all actions must be in ActionStateSucceeded. The Workflow will enter a WorkflowStateFailed if 1 or more Actions fails. + type: string + required: + - actions + - conditions + type: object + type: object + served: false + storage: false + subresources: + status: {}