package instance
import (
"context"
"errors"
"fmt"
"os"
"path/filepath"
"github.com/lima-vm/lima/v2/pkg/cidata"
"github.com/lima-vm/lima/v2/pkg/driverutil"
"github.com/lima-vm/lima/v2/pkg/limatype"
"github.com/lima-vm/lima/v2/pkg/limatype/dirnames"
"github.com/lima-vm/lima/v2/pkg/limatype/filenames"
"github.com/lima-vm/lima/v2/pkg/limayaml"
"github.com/lima-vm/lima/v2/pkg/osutil"
"github.com/lima-vm/lima/v2/pkg/store"
"github.com/lima-vm/lima/v2/pkg/version"
)
func Create(ctx context.Context, instName string, instConfig []byte, saveBrokenYAML bool) (*limatype.Instance, error) {
if instName == "" {
return nil, errors.New("got empty instName")
}
if len(instConfig) == 0 {
return nil, errors.New("got empty instConfig")
}
instDir, err := dirnames.InstanceDir(instName)
if err != nil {
return nil, err
}
maxSockName := filepath.Join(instDir, filenames.LongestSock)
if len(maxSockName) >= osutil.UnixPathMax {
return nil, fmt.Errorf("instance name %q too long: %q must be less than UNIX_PATH_MAX=%d characters, but is %d",
instName, maxSockName, osutil.UnixPathMax, len(maxSockName))
}
if _, err := os.Stat(instDir); !errors.Is(err, os.ErrNotExist) {
return nil, fmt.Errorf("instance %q already exists (%q)", instName, instDir)
}
filePath := filepath.Join(instDir, filenames.LimaYAML)
loadedInstConfig, err := limayaml.LoadWithWarnings(ctx, instConfig, filePath)
if err != nil {
return nil, err
}
if err := driverutil.ResolveVMType(ctx, loadedInstConfig, filePath); err != nil {
return nil, fmt.Errorf("failed to resolve vm for %q: %w", filePath, err)
}
if err := limayaml.Validate(loadedInstConfig, true); err != nil {
if !saveBrokenYAML {
return nil, err
}
rejectedYAML := "lima.REJECTED.yaml"
if writeErr := os.WriteFile(rejectedYAML, instConfig, 0o644); writeErr != nil {
return nil, fmt.Errorf("the YAML is invalid, attempted to save the buffer as %q but failed: %w: %w", rejectedYAML, writeErr, err)
}
return nil, fmt.Errorf("the YAML is invalid, saved the buffer as %q: %w", rejectedYAML, err)
}
if err := os.MkdirAll(instDir, 0o700); err != nil {
return nil, err
}
if err := os.WriteFile(filePath, instConfig, 0o644); err != nil {
return nil, err
}
if err := cidata.GenerateCloudConfig(ctx, instDir, instName, loadedInstConfig); err != nil {
return nil, err
}
if err := os.WriteFile(filepath.Join(instDir, filenames.LimaVersion), []byte(version.Version), 0o444); err != nil {
return nil, err
}
inst, err := store.Inspect(ctx, instName)
if err != nil {
return nil, err
}
limaDriver, err := driverutil.CreateConfiguredDriver(inst, 0)
if err != nil {
return nil, fmt.Errorf("failed to create driver instance: %w", err)
}
if err := limaDriver.Create(ctx); err != nil {
return nil, err
}
return inst, nil
}