Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement VM plugin subsystem #3384

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
11 changes: 11 additions & 0 deletions pkg/plugins/constants.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
package plugins

import "time"

const (
// DefaultTimeout is the default timeout for gRPC operations
DefaultTimeout = 30 * time.Second

// DefaultPluginDir is the default directory where plugins are stored
DefaultPluginDir = "/usr/local/lib/lima/plugins"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should be probably like PREFIX/libexec/lima/plugins/vm.

The PREFIX must be detected via the os.Executable path

// self: /usr/local/bin/limactl
selfDir := filepath.Dir(self)
selfDirDir := filepath.Dir(selfDir)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Alternatively we can look for lima-plugin-vm-<PLUGIN> in PATH.
This model is similar to containerd runtime plugins. (containerd-shim-runc-v2)

)
176 changes: 176 additions & 0 deletions pkg/plugins/driver.proto
Original file line number Diff line number Diff line change
@@ -0,0 +1,176 @@
syntax = "proto3";

package plugins;

option go_package = "github.com/lima-vm/lima/pkg/plugins";

service VMDriver {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Eventually this service should also have a method "Fulfill the empty fields such as mountType"

rpc GetMetadata(GetMetadataRequest) returns (GetMetadataResponse);
rpc StartVM(StartVMRequest) returns (StartVMResponse);
rpc StopVM(StopVMRequest) returns (StopVMResponse);
rpc Initialize(InitializeRequest) returns (InitializeResponse);
rpc CreateDisk(CreateDiskRequest) returns (CreateDiskResponse);
rpc Validate(ValidateRequest) returns (ValidateResponse);
rpc Register(RegisterRequest) returns (RegisterResponse);
rpc Unregister(UnregisterRequest) returns (UnregisterResponse);
rpc ChangeDisplayPassword(ChangeDisplayPasswordRequest) returns (ChangeDisplayPasswordResponse);
rpc GetDisplayConnection(GetDisplayConnectionRequest) returns (GetDisplayConnectionResponse);
rpc CreateSnapshot(CreateSnapshotRequest) returns (CreateSnapshotResponse);
rpc ApplySnapshot(ApplySnapshotRequest) returns (ApplySnapshotResponse);
rpc DeleteSnapshot(DeleteSnapshotRequest) returns (DeleteSnapshotResponse);
rpc ListSnapshots(ListSnapshotsRequest) returns (ListSnapshotsResponse);
rpc GetGuestAgentConnection(GetGuestAgentConnectionRequest) returns (GetGuestAgentConnectionResponse);
}

// Response is a common response type used by helper functions
message Response {
bool success = 1;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Probably Response should not be returned on an error?

gRPC has its own error reporting system
https://github.com/grpc/grpc-go/blob/6819ed796fcd0232a46dab21c1b7826aa7f1d561/examples/features/error_details/client/main.go#L55-L65

string message = 2;
}

message GetMetadataRequest {}

message GetMetadataResponse {
string name = 1;
string version = 2;
string description = 3;
repeated string supported_vm_types = 4;
Copy link
Member

@AkihiroSuda AkihiroSuda Mar 27, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is an interesting topic.

  • Is there any use case to support multiple VM types in a single plugin? (e.g., "qemu" plugin may support both "qemu" and "qemu-utm" VM types?)
  • Also, is there any use case to support the same VM type in multiple plugins?

}

message StartVMRequest {
string config = 1; // Serialized config
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What is the serialization format?

}

message StartVMResponse {
bool success = 1;
string message = 2;
bool can_run_gui = 3;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't this available in Metadata?

}

message StopVMRequest {
string instance_id = 1;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is "id" different from "name"?

}

message StopVMResponse {
bool success = 1;
string message = 2;
}

message InitializeRequest {
string instance_id = 1;
string config = 2;
}

message InitializeResponse {
bool success = 1;
string message = 2;
}

message CreateDiskRequest {
string instance_id = 1;
string config = 2;
}

message CreateDiskResponse {
bool success = 1;
string message = 2;
}

message ValidateRequest {
string config = 1;
}

message ValidateResponse {
bool success = 1;
string message = 2;
}

message RegisterRequest {
string instance_id = 1;
string config = 2;
}

message RegisterResponse {
bool success = 1;
string message = 2;
}

message UnregisterRequest {
string instance_id = 1;
}

message UnregisterResponse {
bool success = 1;
string message = 2;
}

message ChangeDisplayPasswordRequest {
string instance_id = 1;
string password = 2;
}

message ChangeDisplayPasswordResponse {
bool success = 1;
string message = 2;
}

message GetDisplayConnectionRequest {
string instance_id = 1;
}

message GetDisplayConnectionResponse {
bool success = 1;
string message = 2;
string connection = 3;
}

message CreateSnapshotRequest {
string instance_id = 1;
string tag = 2;
}

message CreateSnapshotResponse {
bool success = 1;
string message = 2;
}

message ApplySnapshotRequest {
string instance_id = 1;
string tag = 2;
}

message ApplySnapshotResponse {
bool success = 1;
string message = 2;
}

message DeleteSnapshotRequest {
string instance_id = 1;
string tag = 2;
}

message DeleteSnapshotResponse {
bool success = 1;
string message = 2;
}

message ListSnapshotsRequest {
string instance_id = 1;
}

message ListSnapshotsResponse {
bool success = 1;
string message = 2;
string snapshots = 3;
}

message GetGuestAgentConnectionRequest {
string instance_id = 1;
}

message GetGuestAgentConnectionResponse {
bool success = 1;
string message = 2;
bool forward_guest_agent = 3;
string connection_address = 4;
}
Loading
Loading