Skip to content

Commit 260c8a2

Browse files
committed
hostagent: Add GET /v1/driver/config, PATCH /v1/driver/config APIs to ha.sock
- pkg/driver: Add `Driver.RuntimeConfig` - pkg/hostagent/api/client: Add `HostAgentClient.DriverConfig` - pkg/hostagent/api/server: Add `Backend.DriverConfig` - pkg/hostagent: Add `HostAgent.DriverRuntimeConfig` - pkg/vz: Add `LimaVzDriver.RuntimeConfig` - pkg/instance: Add `saveOnStop` parameter to `StopGracefully` Signed-off-by: Norio Nomura <[email protected]>
1 parent 815592f commit 260c8a2

File tree

7 files changed

+135
-4
lines changed

7 files changed

+135
-4
lines changed

cmd/limactl/stop.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ func stopAction(cmd *cobra.Command, args []string) error {
3939
if force {
4040
instance.StopForcibly(inst)
4141
} else {
42-
err = instance.StopGracefully(inst)
42+
err = instance.StopGracefully(inst, false)
4343
}
4444
// TODO: should we also reconcile networks if graceful stop returned an error?
4545
if err == nil {

pkg/driver/driver.go

+7
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,9 @@ type Driver interface {
6868

6969
// GuestAgentConn returns the guest agent connection, or nil (if forwarded by ssh).
7070
GuestAgentConn(_ context.Context) (net.Conn, error)
71+
72+
// RuntimeConfig accepts config containing changes to the runtime configuration, and returns the updated runtime configuration.
73+
RuntimeConfig(_ context.Context, config interface{}) (interface{}, error)
7174
}
7275

7376
type BaseDriver struct {
@@ -149,3 +152,7 @@ func (d *BaseDriver) GuestAgentConn(_ context.Context) (net.Conn, error) {
149152
// use the unix socket forwarded by host agent
150153
return nil, nil
151154
}
155+
156+
func (d *BaseDriver) RuntimeConfig(_ context.Context, _ interface{}) (interface{}, error) {
157+
return nil, fmt.Errorf("unimplemented")
158+
}

pkg/hostagent/api/client/client.go

+35
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ package client
44
// Apache License 2.0
55

66
import (
7+
"bytes"
78
"context"
89
"encoding/json"
910
"fmt"
11+
"io"
1012
"net/http"
1113

1214
"github.com/lima-vm/lima/pkg/hostagent/api"
@@ -16,6 +18,7 @@ import (
1618
type HostAgentClient interface {
1719
HTTPClient() *http.Client
1820
Info(context.Context) (*api.Info, error)
21+
DriverConfig(context.Context, interface{}) (interface{}, error)
1922
}
2023

2124
// NewHostAgentClient creates a client.
@@ -62,3 +65,35 @@ func (c *client) Info(ctx context.Context) (*api.Info, error) {
6265
}
6366
return &info, nil
6467
}
68+
69+
func (c *client) DriverConfig(ctx context.Context, config interface{}) (interface{}, error) {
70+
u := fmt.Sprintf("http://%s/%s/driver/config", c.dummyHost, c.version)
71+
method := "GET"
72+
var body io.Reader
73+
if config != nil {
74+
method = "PATCH"
75+
b, err := json.Marshal(config)
76+
if err != nil {
77+
return nil, err
78+
}
79+
body = bytes.NewBuffer(b)
80+
}
81+
req, err := http.NewRequestWithContext(ctx, method, u, body)
82+
if err != nil {
83+
return nil, err
84+
}
85+
req.Header.Set("Content-Type", "application/json")
86+
resp, err := c.Do(req)
87+
if err != nil {
88+
return nil, err
89+
}
90+
defer resp.Body.Close()
91+
if err := httpclientutil.Successful(resp); err != nil {
92+
return nil, err
93+
}
94+
dec := json.NewDecoder(resp.Body)
95+
if err := dec.Decode(&config); err != nil {
96+
return nil, err
97+
}
98+
return &config, nil
99+
}

pkg/hostagent/api/server/server.go

+30
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,36 @@ func (b *Backend) GetInfo(w http.ResponseWriter, r *http.Request) {
5050
_, _ = w.Write(m)
5151
}
5252

53+
// DriverConfig is the handler for GET /v1/driver/config and PATCH /v1/driver/config.
54+
func (b *Backend) DriverConfig(w http.ResponseWriter, r *http.Request) {
55+
ctx := r.Context()
56+
ctx, cancel := context.WithCancel(ctx)
57+
defer cancel()
58+
var config interface{}
59+
if r.Method == http.MethodPatch {
60+
defer r.Body.Close()
61+
if err := json.NewDecoder(r.Body).Decode(&config); err != nil {
62+
b.onError(w, err, http.StatusBadRequest)
63+
return
64+
}
65+
}
66+
config, err := b.Agent.DriverRuntimeConfig(ctx, config)
67+
if err != nil {
68+
b.onError(w, err, http.StatusInternalServerError)
69+
return
70+
}
71+
m, err := json.Marshal(config)
72+
if err != nil {
73+
b.onError(w, err, http.StatusInternalServerError)
74+
return
75+
}
76+
w.Header().Set("Content-Type", "application/json")
77+
w.WriteHeader(http.StatusOK)
78+
_, _ = w.Write(m)
79+
}
80+
5381
func AddRoutes(r *http.ServeMux, b *Backend) {
5482
r.Handle("/v1/info", http.HandlerFunc(b.GetInfo))
83+
r.Handle("GET /v1/driver/config", http.HandlerFunc(b.DriverConfig))
84+
r.Handle("PATCH /v1/driver/config", http.HandlerFunc(b.DriverConfig))
5585
}

pkg/hostagent/hostagent.go

+4
Original file line numberDiff line numberDiff line change
@@ -423,6 +423,10 @@ func (a *HostAgent) Info(_ context.Context) (*hostagentapi.Info, error) {
423423
return info, nil
424424
}
425425

426+
func (a *HostAgent) DriverRuntimeConfig(ctx context.Context, config interface{}) (interface{}, error) {
427+
return a.driver.RuntimeConfig(ctx, config)
428+
}
429+
426430
func (a *HostAgent) startHostAgentRoutines(ctx context.Context) error {
427431
if *a.instConfig.Plain {
428432
logrus.Info("Running in plain mode. Mounts, port forwarding, containerd, etc. will be ignored. Guest agent will not be running.")

pkg/instance/stop.go

+27-3
Original file line numberDiff line numberDiff line change
@@ -9,19 +9,43 @@ import (
99
"strings"
1010
"time"
1111

12+
hostagentclient "github.com/lima-vm/lima/pkg/hostagent/api/client"
1213
hostagentevents "github.com/lima-vm/lima/pkg/hostagent/events"
14+
"github.com/lima-vm/lima/pkg/limayaml"
1315
"github.com/lima-vm/lima/pkg/osutil"
1416
"github.com/lima-vm/lima/pkg/store"
1517
"github.com/lima-vm/lima/pkg/store/filenames"
1618
"github.com/sirupsen/logrus"
1719
)
1820

19-
func StopGracefully(inst *store.Instance) error {
20-
// TODO: support store.StatusSuspended
21-
if inst.Status != store.StatusRunning {
21+
func StopGracefully(inst *store.Instance, saveOnStop bool) error {
22+
if inst.Status == store.StatusSaved {
23+
if saveOnStop {
24+
return fmt.Errorf("Instance %q is already saved", inst.Name)
25+
}
26+
return fmt.Errorf("Instance %q is saved. To stop, run `limactl start %s && limactl stop %s`", inst.Name, inst.Name, inst.Name)
27+
} else if inst.Status != store.StatusRunning {
2228
return fmt.Errorf("expected status %q, got %q (maybe use `limactl stop -f`?)", store.StatusRunning, inst.Status)
2329
}
2430

31+
if inst.VMType == limayaml.VZ {
32+
haSock := filepath.Join(inst.Dir, filenames.HostAgentSock)
33+
haClient, err := hostagentclient.NewHostAgentClient(haSock)
34+
if err != nil {
35+
logrus.WithError(err).Error("Failed to create a host agent client")
36+
}
37+
ctx, cancel := context.WithTimeout(context.TODO(), 3*time.Second)
38+
defer cancel()
39+
disableSaveOnStopConfig := struct {
40+
SaveOnStop bool `json:"saveOnStop"`
41+
}{SaveOnStop: saveOnStop}
42+
_, err = haClient.DriverConfig(ctx, disableSaveOnStopConfig)
43+
if err != nil {
44+
return fmt.Errorf("failed to disable saveOnStop: %w", err)
45+
}
46+
} else if saveOnStop {
47+
return fmt.Errorf("save is not supported for %q", inst.VMType)
48+
}
2549
begin := time.Now() // used for logrus propagation
2650
logrus.Infof("Sending SIGINT to hostagent process %d", inst.HostAgentPID)
2751
if err := osutil.SysKill(inst.HostAgentPID, osutil.SigInt); err != nil {

pkg/vz/vz_driver_darwin.go

+31
Original file line numberDiff line numberDiff line change
@@ -8,14 +8,17 @@ import (
88
"fmt"
99
"net"
1010
"path/filepath"
11+
"runtime"
1112
"time"
1213

1314
"github.com/Code-Hex/vz/v3"
15+
"github.com/mitchellh/mapstructure"
1416

1517
"github.com/sirupsen/logrus"
1618

1719
"github.com/lima-vm/lima/pkg/driver"
1820
"github.com/lima-vm/lima/pkg/limayaml"
21+
"github.com/lima-vm/lima/pkg/osutil"
1922
"github.com/lima-vm/lima/pkg/reflectutil"
2023
"github.com/lima-vm/lima/pkg/store/filenames"
2124
)
@@ -204,6 +207,34 @@ func (l *LimaVzDriver) RunGUI() error {
204207
return fmt.Errorf("RunGUI is not supported for the given driver '%s' and display '%s'", "vz", *l.Instance.Config.Video.Display)
205208
}
206209

210+
func (l *LimaVzDriver) RuntimeConfig(_ context.Context, config interface{}) (interface{}, error) {
211+
if config == nil {
212+
return l.config, nil
213+
}
214+
var newConfig LimaVzDriverRuntimeConfig
215+
err := mapstructure.Decode(config, &newConfig)
216+
if err != nil {
217+
return nil, err
218+
}
219+
if newConfig.SaveOnStop {
220+
if runtime.GOARCH != "arm64" {
221+
return nil, fmt.Errorf("saveOnStop is not supported on %s", runtime.GOARCH)
222+
} else if runtime.GOOS != "darwin" {
223+
return nil, fmt.Errorf("saveOnStop is not supported on %s", runtime.GOOS)
224+
} else if macOSProductVersion, err := osutil.ProductVersion(); err != nil {
225+
return nil, fmt.Errorf("failed to get macOS product version: %w", err)
226+
} else if macOSProductVersion.Major < 14 {
227+
return nil, fmt.Errorf("saveOnStop is not supported on macOS %d", macOSProductVersion.Major)
228+
}
229+
logrus.Info("VZ RuntimeConfiguration changed: SaveOnStop is enabled")
230+
l.config.SaveOnStop = true
231+
} else {
232+
logrus.Info("VZ RuntimeConfiguration changed: SaveOnStop is disabled")
233+
l.config.SaveOnStop = false
234+
}
235+
return l.config, nil
236+
}
237+
207238
func (l *LimaVzDriver) Stop(_ context.Context) error {
208239
if l.config.SaveOnStop {
209240
machineStatePath := filepath.Join(l.Instance.Dir, filenames.VzMachineState)

0 commit comments

Comments
 (0)