Skip to content

Commit 5bb9901

Browse files
committedDec 10, 2024·
api: A generic copy function for vim25/types
This patch provides a helper function, `Copy[T any](src T) T` for creating exact copies of Go objects, specifically those from vim25/types. The copy function encodes the Go object to JSON using the vim JSON encoder. Then the copy function decodes the data to a new instance of the same type. The function is implemented using Go generics, so it does not depend directly on reflection (though Go's JSON encoder does). Signed-off-by: akutz <akutz@vmware.com>
1 parent 8b2a311 commit 5bb9901

File tree

3 files changed

+242
-0
lines changed

3 files changed

+242
-0
lines changed
 

‎vim25/types/deep_copy.go

+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
/*
2+
Copyright (c) 2024-2024 VMware, Inc. All Rights Reserved.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
*/
16+
17+
package types
18+
19+
import (
20+
"bytes"
21+
)
22+
23+
// DeepCopyInto creates a deep-copy of src by encoding it to JSON and then
24+
// decoding that into dst.
25+
// Please note, empty slices or maps in src that are set to omitempty will be
26+
// nil in the copied object.
27+
func DeepCopyInto[T AnyType](dst *T, src T) error {
28+
var w bytes.Buffer
29+
e := NewJSONEncoder(&w)
30+
if err := e.Encode(src); err != nil {
31+
return err
32+
}
33+
d := NewJSONDecoder(&w)
34+
if err := d.Decode(dst); err != nil {
35+
return err
36+
}
37+
return nil
38+
}
39+
40+
// MustDeepCopyInto panics if DeepCopyInto returns an error.
41+
func MustDeepCopyInto[T AnyType](dst *T, src T) error {
42+
if err := DeepCopyInto(dst, src); err != nil {
43+
panic(err)
44+
}
45+
return nil
46+
}
47+
48+
// DeepCopy creates a deep-copy of src by encoding it to JSON and then decoding
49+
// that into a new instance of T.
50+
// Please note, empty slices or maps in src that are set to omitempty will be
51+
// nil in the copied object.
52+
func DeepCopy[T AnyType](src T) (T, error) {
53+
var dst T
54+
err := DeepCopyInto(&dst, src)
55+
return dst, err
56+
}
57+
58+
// MustDeepCopy panics if DeepCopy returns an error.
59+
func MustDeepCopy[T AnyType](src T) T {
60+
dst, err := DeepCopy(src)
61+
if err != nil {
62+
panic(err)
63+
}
64+
return dst
65+
}

‎vim25/types/deep_copy_test.go

+173
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,173 @@
1+
/*
2+
Copyright (c) 2024-2024 VMware, Inc. All Rights Reserved.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
*/
16+
17+
package types
18+
19+
import (
20+
"testing"
21+
22+
"github.com/google/go-cmp/cmp"
23+
"github.com/stretchr/testify/assert"
24+
)
25+
26+
func TestMustDeepCopy(t *testing.T) {
27+
newConfigSpec := func() VirtualMachineConfigSpec {
28+
return VirtualMachineConfigSpec{
29+
Name: "vm-001",
30+
GuestId: "otherGuest",
31+
Files: &VirtualMachineFileInfo{VmPathName: "[datastore1]"},
32+
NumCPUs: 1,
33+
MemoryMB: 128,
34+
DeviceChange: []BaseVirtualDeviceConfigSpec{
35+
&VirtualDeviceConfigSpec{
36+
Operation: VirtualDeviceConfigSpecOperationAdd,
37+
Device: &VirtualLsiLogicController{VirtualSCSIController{
38+
SharedBus: VirtualSCSISharingNoSharing,
39+
VirtualController: VirtualController{
40+
BusNumber: 0,
41+
VirtualDevice: VirtualDevice{
42+
Key: 1000,
43+
},
44+
},
45+
}},
46+
},
47+
&VirtualDeviceConfigSpec{
48+
Operation: VirtualDeviceConfigSpecOperationAdd,
49+
FileOperation: VirtualDeviceConfigSpecFileOperationCreate,
50+
Device: &VirtualDisk{
51+
VirtualDevice: VirtualDevice{
52+
Key: 0,
53+
ControllerKey: 1000,
54+
UnitNumber: NewInt32(10),
55+
Backing: &VirtualDiskFlatVer2BackingInfo{
56+
DiskMode: string(VirtualDiskModePersistent),
57+
ThinProvisioned: NewBool(true),
58+
VirtualDeviceFileBackingInfo: VirtualDeviceFileBackingInfo{
59+
FileName: "[datastore1]",
60+
},
61+
},
62+
},
63+
CapacityInKB: 4000000,
64+
},
65+
},
66+
&VirtualDeviceConfigSpec{
67+
Operation: VirtualDeviceConfigSpecOperationAdd,
68+
Device: &VirtualE1000{VirtualEthernetCard{
69+
VirtualDevice: VirtualDevice{
70+
Key: 0,
71+
DeviceInfo: &Description{
72+
Label: "Network Adapter 1",
73+
Summary: "VM Network",
74+
},
75+
Backing: &VirtualEthernetCardNetworkBackingInfo{
76+
VirtualDeviceDeviceBackingInfo: VirtualDeviceDeviceBackingInfo{
77+
DeviceName: "VM Network",
78+
},
79+
},
80+
},
81+
AddressType: string(VirtualEthernetCardMacTypeGenerated),
82+
}},
83+
},
84+
},
85+
ExtraConfig: []BaseOptionValue{
86+
&OptionValue{Key: "bios.bootOrder", Value: "ethernet0"},
87+
},
88+
}
89+
}
90+
91+
t.Run("a string", func(t *testing.T) {
92+
t.Parallel()
93+
var dst AnyType
94+
assert.NotPanics(t, func() {
95+
dst = MustDeepCopy("hello")
96+
})
97+
assert.Equal(t, "hello", dst)
98+
})
99+
100+
t.Run("a *uint8", func(t *testing.T) {
101+
t.Parallel()
102+
var dst AnyType
103+
assert.NotPanics(t, func() {
104+
dst = MustDeepCopy(New(uint8(42)))
105+
})
106+
assert.Equal(t, &[]uint8{42}[0], dst)
107+
})
108+
109+
t.Run("a VirtualMachineConfigSpec", func(t *testing.T) {
110+
t.Parallel()
111+
var dst AnyType
112+
assert.NotPanics(t, func() {
113+
dst = MustDeepCopy(newConfigSpec())
114+
})
115+
assert.Equal(t, newConfigSpec(), dst)
116+
})
117+
118+
t.Run("a *VirtualMachineConfigSpec", func(t *testing.T) {
119+
t.Parallel()
120+
var dst AnyType
121+
assert.NotPanics(t, func() {
122+
dst = MustDeepCopy(New(newConfigSpec()))
123+
})
124+
assert.Equal(t, New(newConfigSpec()), dst)
125+
})
126+
127+
t.Run("a **VirtualMachineConfigSpec", func(t *testing.T) {
128+
t.Parallel()
129+
130+
var dst AnyType
131+
exp := newConfigSpec()
132+
ptrExp := &exp
133+
ptrPtrExp := &ptrExp
134+
135+
assert.NotPanics(t, func() {
136+
dst = MustDeepCopy(New(New(newConfigSpec())))
137+
})
138+
139+
assert.Equal(t, ptrPtrExp, dst)
140+
141+
exp.Name += "-not-equal"
142+
assert.NotEqual(t, ptrPtrExp, dst)
143+
})
144+
145+
t.Run("a VirtualMachineConfigSpec with nil DeviceChange vs empty", func(t *testing.T) {
146+
t.Parallel()
147+
148+
t.Run("src is nil, exp is nil", func(t *testing.T) {
149+
t.Parallel()
150+
var dst AnyType
151+
exp, src := newConfigSpec(), newConfigSpec()
152+
exp.DeviceChange = nil
153+
src.DeviceChange = nil
154+
assert.NotPanics(t, func() {
155+
dst = MustDeepCopy(src)
156+
})
157+
assert.Equal(t, exp, dst, cmp.Diff(exp, dst))
158+
})
159+
160+
t.Run("src is empty, exp is nil", func(t *testing.T) {
161+
t.Parallel()
162+
var dst AnyType
163+
exp, src := newConfigSpec(), newConfigSpec()
164+
exp.DeviceChange = nil
165+
src.DeviceChange = []BaseVirtualDeviceConfigSpec{}
166+
assert.NotPanics(t, func() {
167+
dst = MustDeepCopy(src)
168+
})
169+
assert.Equal(t, exp, dst, cmp.Diff(exp, dst))
170+
})
171+
172+
})
173+
}

‎vim25/types/helpers.go

+4
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,10 @@ func EnumValuesAsStrings[T ~string](enumValues []T) []string {
3434
return stringValues
3535
}
3636

37+
func New[T any](t T) *T {
38+
return &t
39+
}
40+
3741
func NewBool(v bool) *bool {
3842
return &v
3943
}

0 commit comments

Comments
 (0)
Please sign in to comment.