Skip to content

Commit 044275e

Browse files
committed
vm commit: reset machine ID before committing image
1 parent 64299d3 commit 044275e

File tree

5 files changed

+66
-34
lines changed

5 files changed

+66
-34
lines changed

cmd/image_build.go

+6-3
Original file line numberDiff line numberDiff line change
@@ -212,11 +212,14 @@ func imageBuildCommand() *cobra.Command {
212212
}
213213

214214
buildConfig := virter.ImageBuildConfig{
215-
ImageName: newImageName,
216215
ContainerName: containerName,
217-
ShutdownTimeout: shutdownTimeout,
218216
ProvisionConfig: provisionConfig,
219-
ResetMachineID: resetMachineID,
217+
CommitConfig: virter.CommitConfig{
218+
ImageName: newImageName,
219+
Shutdown: true,
220+
ShutdownTimeout: shutdownTimeout,
221+
ResetMachineID: resetMachineID,
222+
},
220223
}
221224

222225
p = mpb.NewWithContext(ctx, DefaultContainerOpt())

cmd/vm_commit.go

+11-2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import (
1212
)
1313

1414
func vmCommitCommand() *cobra.Command {
15+
var resetMachineID bool
1516
var shutdown bool
1617

1718
var commitCmd = &cobra.Command{
@@ -39,7 +40,14 @@ the virtual machine name if no other value is given.`,
3940

4041
p := mpb.NewWithContext(ctx, DefaultContainerOpt())
4142

42-
err = v.VMCommit(ctx, actualtime.ActualTime{}, vmName, imageName, shutdown, shutdownTimeout, viper.GetBool("libvirt.static_dhcp"), virter.WithProgress(DefaultProgressFormat(p)))
43+
commitConfig := virter.CommitConfig{
44+
ImageName: imageName,
45+
Shutdown: shutdown,
46+
ShutdownTimeout: shutdownTimeout,
47+
ResetMachineID: resetMachineID,
48+
}
49+
50+
err = v.VMCommit(ctx, actualtime.ActualTime{}, vmName, commitConfig, viper.GetBool("libvirt.static_dhcp"), virter.WithProgress(DefaultProgressFormat(p)))
4351
if err != nil {
4452
log.Fatal(err)
4553
}
@@ -57,7 +65,8 @@ the virtual machine name if no other value is given.`,
5765
},
5866
}
5967

60-
commitCmd.Flags().BoolVarP(&shutdown, "shutdown", "s", false, "whether to shut the VM down and wait until it stops (default false)")
68+
commitCmd.Flags().BoolVar(&resetMachineID, "reset-machine-id", true, "Whether or not to clear the /etc/machine-id file after provisioning")
69+
commitCmd.Flags().BoolVarP(&shutdown, "shutdown", "s", true, "Whether to shut the VM down and wait until it stops")
6170

6271
return commitCmd
6372
}

internal/virter/image.go

+2-16
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,6 @@ import (
1515
"net/http"
1616
"strings"
1717
"sync"
18-
"time"
1918

2019
"github.com/LINBIT/containerapi"
2120
regv1 "github.com/google/go-containerregistry/pkg/v1"
@@ -531,11 +530,9 @@ type ImageBuildTools struct {
531530

532531
// ImageBuildConfig contains the configuration for building an image
533532
type ImageBuildConfig struct {
534-
ImageName string
535533
ContainerName string
536-
ShutdownTimeout time.Duration
537534
ProvisionConfig ProvisionConfig
538-
ResetMachineID bool
535+
CommitConfig CommitConfig
539536
}
540537

541538
func (v *Virter) imageBuildProvisionCommit(ctx context.Context, tools ImageBuildTools, vmConfig VMConfig, readyConfig VmReadyConfig, buildConfig ImageBuildConfig, opts ...LayerOperationOption) error {
@@ -547,17 +544,6 @@ func (v *Virter) imageBuildProvisionCommit(ctx context.Context, tools ImageBuild
547544
return err
548545
}
549546

550-
if buildConfig.ResetMachineID {
551-
// starting the VM creates a machine-id
552-
// we want these IDs to be unique, so reset to empty
553-
resetMachineID := ProvisionStep{
554-
Shell: &ProvisionShellStep{
555-
Script: "truncate -c -s 0 /etc/machine-id",
556-
},
557-
}
558-
buildConfig.ProvisionConfig.Steps = append(buildConfig.ProvisionConfig.Steps, resetMachineID)
559-
}
560-
561547
for _, s := range buildConfig.ProvisionConfig.Steps {
562548
if s.Container != nil {
563549
containerCfg := containerapi.NewContainerConfig(
@@ -580,7 +566,7 @@ func (v *Virter) imageBuildProvisionCommit(ctx context.Context, tools ImageBuild
580566
}
581567
}
582568

583-
err = v.VMCommit(ctx, tools.AfterNotifier, vmConfig.Name, buildConfig.ImageName, true, buildConfig.ShutdownTimeout, vmConfig.StaticDHCP, opts...)
569+
err = v.VMCommit(ctx, tools.AfterNotifier, vmConfig.Name, buildConfig.CommitConfig, vmConfig.StaticDHCP, opts...)
584570
if err != nil {
585571
return err
586572
}

internal/virter/vm.go

+38-11
Original file line numberDiff line numberDiff line change
@@ -380,28 +380,55 @@ func (v *Virter) rmSnapshots(domain libvirt.Domain) error {
380380
return nil
381381
}
382382

383+
// CommitConfig contains the configuration for committing a VM
384+
type CommitConfig struct {
385+
ImageName string
386+
Shutdown bool
387+
ShutdownTimeout time.Duration
388+
ResetMachineID bool
389+
}
390+
383391
// VMCommit commits a VM to an image. If shutdown is true, the VM is shut down
384392
// before committing. If shutdown is false, the caller is responsible for
385393
// ensuring that the VM is not running.
386-
func (v *Virter) VMCommit(ctx context.Context, afterNotifier AfterNotifier, vmName, imageName string, shutdown bool, shutdownTimeout time.Duration, staticDHCP bool, opts ...LayerOperationOption) error {
394+
func (v *Virter) VMCommit(ctx context.Context, afterNotifier AfterNotifier, vmName string, commitConfig CommitConfig, staticDHCP bool, opts ...LayerOperationOption) error {
387395
domain, err := v.libvirt.DomainLookupByName(vmName)
388396
if err != nil {
389397
return fmt.Errorf("could not get domain: %w", err)
390398
}
391399

392-
if shutdown {
393-
err = v.vmShutdown(ctx, afterNotifier, shutdownTimeout, domain)
394-
if err != nil {
395-
return err
400+
active, err := v.libvirt.DomainIsActive(domain)
401+
if err != nil {
402+
return fmt.Errorf("could not check if domain is active: %w", err)
403+
}
404+
405+
running := active != 0
406+
407+
// Check if shutdown is allowed before resetting machine ID.
408+
if running && !commitConfig.Shutdown {
409+
return fmt.Errorf("must allow shutdown to commit a running VM")
410+
}
411+
412+
if commitConfig.ResetMachineID {
413+
if !running {
414+
return fmt.Errorf("cannot reset machine ID of VM '%s' that is not running", vmName)
396415
}
397-
} else {
398-
active, err := v.libvirt.DomainIsActive(domain)
416+
417+
// Starting the VM creates a machine ID.
418+
// We want these IDs to be unique, so reset to empty.
419+
err = v.VMExecShell(
420+
ctx, []string{vmName},
421+
&ProvisionShellStep{Script: "truncate -c -s 0 /etc/machine-id"})
422+
399423
if err != nil {
400-
return fmt.Errorf("could not check if domain is active: %w", err)
424+
return err
401425
}
426+
}
402427

403-
if active != 0 {
404-
return fmt.Errorf("cannot commit a running VM")
428+
if running && commitConfig.Shutdown {
429+
err = v.vmShutdown(ctx, afterNotifier, commitConfig.ShutdownTimeout, domain)
430+
if err != nil {
431+
return err
405432
}
406433
}
407434

@@ -424,7 +451,7 @@ func (v *Virter) VMCommit(ctx context.Context, afterNotifier AfterNotifier, vmNa
424451
return err
425452
}
426453

427-
_, err = v.MakeImage(imageName, volumeLayer, opts...)
454+
_, err = v.MakeImage(commitConfig.ImageName, volumeLayer, opts...)
428455
if err != nil {
429456
return err
430457
}

internal/virter/vm_test.go

+9-2
Original file line numberDiff line numberDiff line change
@@ -267,7 +267,7 @@ func TestVMCommit(t *testing.T) {
267267

268268
an := new(mocks.AfterNotifier)
269269

270-
if r[commitShutdown] {
270+
if r[commitShutdown] && r[commitDomainActive] {
271271
if r[commitShutdownTimeout] {
272272
timeout := make(chan time.Time, 1)
273273
timeout <- time.Unix(0, 0)
@@ -281,7 +281,14 @@ func TestVMCommit(t *testing.T) {
281281
pool, err := l.StoragePoolLookupByName(poolName)
282282
assert.NoError(t, err)
283283

284-
err = v.VMCommit(context.Background(), an, vmName, vmName, r[commitShutdown], shutdownTimeout, false)
284+
commitConfig := virter.CommitConfig{
285+
ImageName: vmName,
286+
Shutdown: r[commitShutdown],
287+
ShutdownTimeout: shutdownTimeout,
288+
ResetMachineID: false,
289+
}
290+
291+
err = v.VMCommit(context.Background(), an, vmName, commitConfig, false)
285292
if expectOk {
286293
assert.NoError(t, err)
287294

0 commit comments

Comments
 (0)