Editing APIs
Tips, tutorials, examples, best practices.
Tips
Some helpful tips to make life easier while writing conversion routines.
- Do not delete fields. It's easier to write loss-less conversion routines if the field is renamed to something that indicates it's no longer used. This becomes more important in the spoke-hub-spoke scenario, because the hub will not have an annotation preserving the original spoke's values.
Example 1: Add a field to a resource
In the DWS repository, add a new field to the Status
section of the SystemConfiguration
hub resource. In this case, the hub version is v1alpha2
.
Step 1: Add a new ResourceError field
diff --git a/api/v1alpha2/systemconfiguration_types.go b/api/v1alpha2/systemconfiguration_types.go
index 3e4d29fb..2c65e3a0 100644
--- a/api/v1alpha2/systemconfiguration_types.go
+++ b/api/v1alpha2/systemconfiguration_types.go
@@ -82,6 +82,9 @@ type SystemConfigurationSpec struct {
type SystemConfigurationStatus struct {
// Ready indicates when the SystemConfiguration has been reconciled
Ready bool `json:"ready"`
+
+ // Error information
+ ResourceError `json:",inline"`
}
//+kubebuilder:object:root=true
```
(An aside: That `ResourceError` type used for this new field is inlined, bringing a field named `Error`, so that's what we'll see next in the conversion routines.)
### Step 2: Run the conversion code generator
Run the generator to update the generated conversion routines for this type.
```console
make generate-go-conversions
This will emit some scary-looking output:
/Users/droehrich/work/rabsw/dws-7.git/bin/conversion-gen \
--output-file=zz_generated.conversion.go \
--go-header-file=./hack/boilerplate.go.txt \
./api/v1alpha1
E0913 11:13:26.993464 23295 conversion.go:741] Warning: could not find nor generate a final Conversion function for github.com/DataWorkflowServices/dws/api/v1alpha2.SystemConfigurationStatus -> github.com/DataWorkflowServices/dws/api/v1alpha1.SystemConfigurationStatus
E0913 11:13:26.993609 23295 conversion.go:742] the following fields need manual conversion:
E0913 11:13:26.993613 23295 conversion.go:744] - ResourceError
Let's look around for a missing routine:
$ git status
On branch api-change-demo
Changes not staged for commit:
(use "git add <file>..." to update what will be committed)
(use "git restore <file>..." to discard changes in working directory)
modified: api/v1alpha1/zz_generated.conversion.go
modified: api/v1alpha2/systemconfiguration_types.go
Look at the generated code for a clue. There are two here. First, a comment about our new field requiring manual conversion. Second, the generated entrypoint for the SystemConfigurationStatus
conversion routine has been taken away:
$ git diff -U1 api/v1alpha1/zz_generated.conversion.go
diff --git a/api/v1alpha1/zz_generated.conversion.go b/api/v1alpha1/zz_generated.conversion.go
index 1f1b3cb6..d7c915aa 100644
--- a/api/v1alpha1/zz_generated.conversion.go
+++ b/api/v1alpha1/zz_generated.conversion.go
@@ -2082,2 +2082,3 @@ func autoConvert_v1alpha2_SystemConfigurationStatus_To_v1alpha1_SystemConfigurat
out.Ready = in.Ready
+ // WARNING: in.ResourceError requires manual conversion: does not exist in peer-type
return nil
@@ -2085,7 +2086,2 @@ func autoConvert_v1alpha2_SystemConfigurationStatus_To_v1alpha1_SystemConfigurat
-// Convert_v1alpha2_SystemConfigurationStatus_To_v1alpha1_SystemConfigurationStatus is an autogenerated conversion function.
-func Convert_v1alpha2_SystemConfigurationStatus_To_v1alpha1_SystemConfigurationStatus(in *v1alpha2.SystemConfigurationStatus, out *SystemConfigurationStatus, s conversion.Scope) error {
- return autoConvert_v1alpha2_SystemConfigurationStatus_To_v1alpha1_SystemConfigurationStatus(in, out, s)
-}
-
func autoConvert_v1alpha1_SystemConfigurationStorageNode_To_v1alpha2_SystemConfigurationStorageNode(in *SystemConfigurationStorageNode, out *v1alpha2.SystemConfigurationStorageNode, s conversion.Scope) error {
The generator is forcing us to acknowledge that we have conversion covered for that new field. We have to manually replace that entrypoint to hook up the conversion again.
Step 3: Restore the generated conversion entrypoint
At the end of api/v1alpha1/conversion.go
we will find other entrypoint routines like this one, for earlier changes to the SystemConfiguration
type. Cut-n-paste that entrypoint routine from the diff above into an appropriate place near the end of conversion.go
, adjusting it just enough so make vet
succeeds.
$ git diff -U1 api/v1alpha1/conversion.go
diff --git a/api/v1alpha1/conversion.go b/api/v1alpha1/conversion.go
index f409f9b2..e11a41eb 100644
--- a/api/v1alpha1/conversion.go
+++ b/api/v1alpha1/conversion.go
@@ -581,2 +581,6 @@ func Convert_v1alpha2_SystemConfigurationSpec_To_v1alpha1_SystemConfigurationSpe
+func Convert_v1alpha2_SystemConfigurationStatus_To_v1alpha1_SystemConfigurationStatus(in *dwsv1alpha2.SystemConfigurationStatus, out *SystemConfigurationStatus, s apiconversion.Scope) error {
+ return autoConvert_v1alpha2_SystemConfigurationStatus_To_v1alpha1_SystemConfigurationStatus(in, out, s)
+}
+
func Convert_v1alpha2_StorageSpec_To_v1alpha1_StorageSpec(in *dwsv1alpha2.StorageSpec, out *StorageSpec, s apiconversion.Scope) error {
Confirm that it compiles:
Step 4: Add the new field to the conversion routine
Add this field to the ConvertTo()
(convert from spoke to hub) conversion routine for the SystemConfiguration
type.
$ git diff -U1 api/v1alpha1/conversion.go
diff --git a/api/v1alpha1/conversion.go b/api/v1alpha1/conversion.go
index f409f9b2..706f16bb 100644
--- a/api/v1alpha1/conversion.go
+++ b/api/v1alpha1/conversion.go
@@ -369,2 +369,3 @@ func (src *SystemConfiguration) ConvertTo(dstRaw conversion.Hub) error {
dst.Spec.PortsCooldownInSeconds = restored.Spec.PortsCooldownInSeconds
+ dst.Status.Error = restored.Status.Error
In this case, we won't add this field to the ConvertFrom()
(convert from hub to spoke) routine because we don't need to represent it in any way in the spoke. This will not always be the case, so do what makes sense for your particular change.
Try the test suite, but make it easy on yourself and try make vet
first: