Skip to content

Commit 47ace06

Browse files
authored
Merge pull request #3072 from jandubois/based-on
Assemble templates using the new `base` setting
2 parents 29dabb7 + 5602b96 commit 47ace06

File tree

21 files changed

+1788
-73
lines changed

21 files changed

+1788
-73
lines changed

.github/workflows/test.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -253,7 +253,8 @@ jobs:
253253
steps:
254254
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
255255
with:
256-
fetch-depth: 1
256+
# To avoid "failed to load YAML file \"templates/experimental/riscv64.yaml\": can't parse builtin Lima version \"3f3a6f6\": 3f3a6f6 is not in dotted-tri format"
257+
fetch-depth: 0
257258
- uses: actions/setup-go@f111f3307d8850f501ac008e886eec1fd1932a34 # v5.3.0
258259
with:
259260
go-version: 1.24.x

cmd/limactl/start.go

+3-1
Original file line numberDiff line numberDiff line change
@@ -205,7 +205,9 @@ func loadOrCreateInstance(cmd *cobra.Command, args []string, createOnly bool) (*
205205
return nil, err
206206
}
207207
}
208-
208+
if err := tmpl.Embed(cmd.Context(), true, true); err != nil {
209+
return nil, err
210+
}
209211
yqExprs, err := editflags.YQExpressions(flags, true)
210212
if err != nil {
211213
return nil, err

cmd/limactl/template.go

+72-5
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
package main
55

66
import (
7+
"errors"
78
"fmt"
89
"os"
910
"path/filepath"
@@ -26,6 +27,9 @@ func newTemplateCommand() *cobra.Command {
2627
// The template command is still hidden because the subcommands and options are still under development
2728
// and subject to change at any time.
2829
Hidden: true,
30+
PreRun: func(*cobra.Command, []string) {
31+
logrus.Warn("`limactl template` is experimental")
32+
},
2933
}
3034
templateCommand.AddCommand(
3135
newTemplateCopyCommand(),
@@ -46,30 +50,89 @@ var templateCopyExample = ` Template locators are local files, file://, https:/
4650
# Copy default template to STDOUT
4751
limactl template copy template://default -
4852
49-
# Copy template from web location to local file
50-
limactl template copy https://example.com/lima.yaml mighty-machine.yaml
53+
# Copy template from web location to local file and embed all external references
54+
# (this does not embed template:// references)
55+
limactl template copy --embed https://example.com/lima.yaml mighty-machine.yaml
5156
`
5257

5358
func newTemplateCopyCommand() *cobra.Command {
5459
templateCopyCommand := &cobra.Command{
55-
Use: "copy TEMPLATE DEST",
60+
Use: "copy [OPTIONS] TEMPLATE DEST",
5661
Short: "Copy template",
5762
Long: "Copy a template via locator to a local file",
5863
Example: templateCopyExample,
5964
Args: WrapArgsError(cobra.ExactArgs(2)),
6065
RunE: templateCopyAction,
6166
}
67+
templateCopyCommand.Flags().Bool("embed", false, "embed external dependencies into template")
68+
templateCopyCommand.Flags().Bool("embed-all", false, "embed all dependencies into template")
69+
templateCopyCommand.Flags().Bool("fill", false, "fill defaults")
70+
templateCopyCommand.Flags().Bool("verbatim", false, "don't make locators absolute")
6271
return templateCopyCommand
6372
}
6473

6574
func templateCopyAction(cmd *cobra.Command, args []string) error {
75+
embed, err := cmd.Flags().GetBool("embed")
76+
if err != nil {
77+
return err
78+
}
79+
embedAll, err := cmd.Flags().GetBool("embed-all")
80+
if err != nil {
81+
return err
82+
}
83+
fill, err := cmd.Flags().GetBool("fill")
84+
if err != nil {
85+
return err
86+
}
87+
verbatim, err := cmd.Flags().GetBool("verbatim")
88+
if err != nil {
89+
return err
90+
}
91+
if fill {
92+
embedAll = true
93+
}
94+
if embedAll {
95+
embed = true
96+
}
97+
if embed && verbatim {
98+
return errors.New("--verbatim cannot be used with any of --embed, --embed-all, or --fill")
99+
}
66100
tmpl, err := limatmpl.Read(cmd.Context(), "", args[0])
67101
if err != nil {
68102
return err
69103
}
70104
if len(tmpl.Bytes) == 0 {
71105
return fmt.Errorf("don't know how to interpret %q as a template locator", args[0])
72106
}
107+
if !verbatim {
108+
if embed {
109+
// Embed default base.yaml only when fill is true.
110+
if err := tmpl.Embed(cmd.Context(), embedAll, fill); err != nil {
111+
return err
112+
}
113+
} else {
114+
if err := tmpl.UseAbsLocators(); err != nil {
115+
return err
116+
}
117+
}
118+
}
119+
if fill {
120+
limaDir, err := dirnames.LimaDir()
121+
if err != nil {
122+
return err
123+
}
124+
// Load() will merge the template with override.yaml and default.yaml via FillDefaults().
125+
// FillDefaults() needs the potential instance directory to validate host templates using {{.Dir}}.
126+
filePath := filepath.Join(limaDir, tmpl.Name+".yaml")
127+
tmpl.Config, err = limayaml.Load(tmpl.Bytes, filePath)
128+
if err != nil {
129+
return err
130+
}
131+
tmpl.Bytes, err = limayaml.Marshal(tmpl.Config, false)
132+
if err != nil {
133+
return err
134+
}
135+
}
73136
writer := cmd.OutOrStdout()
74137
target := args[1]
75138
if target != "-" {
@@ -116,10 +179,14 @@ func templateValidateAction(cmd *cobra.Command, args []string) error {
116179
if tmpl.Name == "" {
117180
return fmt.Errorf("can't determine instance name from template locator %q", arg)
118181
}
182+
// Embed default base.yaml only when fill is true.
183+
if err := tmpl.Embed(cmd.Context(), true, fill); err != nil {
184+
return err
185+
}
119186
// Load() will merge the template with override.yaml and default.yaml via FillDefaults().
120187
// FillDefaults() needs the potential instance directory to validate host templates using {{.Dir}}.
121-
instDir := filepath.Join(limaDir, tmpl.Name)
122-
y, err := limayaml.Load(tmpl.Bytes, instDir)
188+
filePath := filepath.Join(limaDir, tmpl.Name+".yaml")
189+
y, err := limayaml.Load(tmpl.Bytes, filePath)
123190
if err != nil {
124191
return err
125192
}

hack/cache-common-inc.sh

+12-11
Original file line numberDiff line numberDiff line change
@@ -18,17 +18,18 @@ function error_exit() {
1818
# ```
1919
function download_template_if_needed() {
2020
local template="$1"
21-
case "${template}" in
22-
https://*)
23-
tmp_yaml=$(mktemp -d)/template.yaml
24-
curl -sSLf "${template}" >"${tmp_yaml}" || return
25-
echo "${tmp_yaml}"
26-
;;
27-
*)
28-
test -f "${template}" || return
29-
echo "${template}"
30-
;;
31-
esac
21+
tmp_yaml="$(mktemp -d)/template.yaml"
22+
# The upgrade test doesn't have limactl installed first. The old version wouldn't support `limactl tmpl` anyways.
23+
if command -v limactl >/dev/null; then
24+
limactl tmpl copy --embed-all "${template}" "${tmp_yaml}" || return
25+
else
26+
if [[ $template == https://* ]]; then
27+
curl -sSLf "${template}" >"${tmp_yaml}" || return
28+
else
29+
cp "${template}" "${tmp_yaml}"
30+
fi
31+
fi
32+
echo "${tmp_yaml}"
3233
}
3334

3435
# e.g.

hack/test-port-forwarding.pl

+5-5
Original file line numberDiff line numberDiff line change
@@ -212,11 +212,11 @@ sub JoinHostPort {
212212
# "ipv4" and "ipv6" will be replaced by the actual host ipv4 and ipv6 addresses.
213213
__DATA__
214214
portForwards:
215-
# We can't test that port 22 will be blocked because the guestagent has
216-
# been ignoring it since startup, so the log message is in the part of
217-
# the log we skipped.
218-
# skip: 127.0.0.1 22 → 127.0.0.1 2222
219-
# ignore: 127.0.0.1 sshLocalPort
215+
# We can't test that port 22 will be blocked because the guestagent has
216+
# been ignoring it since startup, so the log message is in the part of
217+
# the log we skipped.
218+
# skip: 127.0.0.1 22 → 127.0.0.1 2222
219+
# ignore: 127.0.0.1 sshLocalPort
220220
221221
- guestIP: 127.0.0.2
222222
guestPortRange: [3000, 3009]

hack/test-templates.sh

+6-2
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,8 @@ if [ "$#" -ne 1 ]; then
1717
exit 1
1818
fi
1919

20-
FILE="$1"
20+
# Resolve any ../ fragments in the filename because they are invalid in relative template locators
21+
FILE="$(cd "$(dirname "$1")" && pwd)/$(basename "$1")"
2122
NAME="$(basename -s .yaml "$FILE")"
2223
OS_HOST="$(uname -o)"
2324

@@ -135,6 +136,9 @@ if [[ -n ${CHECKS["port-forwards"]} ]]; then
135136
limactl validate "$FILE_HOST"
136137
fi
137138

139+
INFO "Make sure template embedding copies \"$FILE_HOST\" exactly"
140+
diff -u <(echo -n "base: $FILE_HOST" | limactl tmpl copy --embed - -) "$FILE_HOST"
141+
138142
function diagnose() {
139143
NAME="$1"
140144
set -x +e
@@ -207,7 +211,7 @@ fi
207211

208212
if [[ -n ${CHECKS["set-user"]} ]]; then
209213
INFO 'Testing that user settings can be provided by lima.yaml'
210-
limactl shell "$NAME" grep "^john:x:4711:4711:John Doe:/home/john-john" /etc/passwd
214+
limactl shell "$NAME" grep "^john:x:4711:4711:John Doe:/home/john-john:/usr/bin/bash" /etc/passwd
211215
fi
212216

213217
if [[ -n ${CHECKS["proxy-settings"]} ]]; then

hack/test-templates/test-misc.yaml

+3-17
Original file line numberDiff line numberDiff line change
@@ -3,21 +3,7 @@
33
# - snapshots
44
# - (More to come)
55
#
6-
# This template requires Lima v1.0.0-alpha.0 or later.
7-
images:
8-
# Try to use release-yyyyMMdd image if available. Note that release-yyyyMMdd will be removed after several months.
9-
- location: "https://cloud-images.ubuntu.com/releases/22.04/release-20220902/ubuntu-22.04-server-cloudimg-amd64.img"
10-
arch: "x86_64"
11-
digest: "sha256:c777670007cc5f132417b9e0bc01367ccfc2a989951ffa225bb1952917c3aa81"
12-
- location: "https://cloud-images.ubuntu.com/releases/22.04/release-20220902/ubuntu-22.04-server-cloudimg-arm64.img"
13-
arch: "aarch64"
14-
digest: "sha256:9620f479bd5a6cbf1e805654d41b27f4fc56ef20f916c8331558241734de81ae"
15-
# Fallback to the latest release image.
16-
# Hint: run `limactl prune` to invalidate the cache
17-
- location: "https://cloud-images.ubuntu.com/releases/24.04/release/ubuntu-24.04-server-cloudimg-amd64.img"
18-
arch: "x86_64"
19-
- location: "https://cloud-images.ubuntu.com/releases/24.04/release/ubuntu-24.04-server-cloudimg-arm64.img"
20-
arch: "aarch64"
6+
base: template://ubuntu-22.04
217

228
# 9p is not compatible with `limactl snapshot`
239
mountTypesUnsupported: ["9p"]
@@ -26,8 +12,6 @@ mounts:
2612
writable: true
2713
- location: "/tmp/lima test dir with spaces"
2814
writable: true
29-
- location: "/tmp/lima"
30-
writable: true
3115

3216
param:
3317
BOOT: boot
@@ -64,3 +48,5 @@ user:
6448
comment: John Doe
6549
home: "/home/{{.User}}-{{.User}}"
6650
uid: 4711
51+
# Ubuntu has identical /bin/bash and /usr/bin/bash
52+
shell: /usr/bin/bash

pkg/limatmpl/abs.go

+131
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
1+
// SPDX-FileCopyrightText: Copyright The Lima Authors
2+
// SPDX-License-Identifier: Apache-2.0
3+
4+
package limatmpl
5+
6+
import (
7+
"errors"
8+
"fmt"
9+
"net/url"
10+
"path"
11+
"path/filepath"
12+
"runtime"
13+
"strings"
14+
)
15+
16+
// UseAbsLocators will replace all relative template locators with absolute ones, so this template
17+
// can be stored anywhere and still reference the same base templates and files.
18+
func (tmpl *Template) UseAbsLocators() error {
19+
err := tmpl.useAbsLocators()
20+
return tmpl.ClearOnError(err)
21+
}
22+
23+
func (tmpl *Template) useAbsLocators() error {
24+
if err := tmpl.Unmarshal(); err != nil {
25+
return err
26+
}
27+
basePath, err := basePath(tmpl.Locator)
28+
if err != nil {
29+
return err
30+
}
31+
for i, baseLocator := range tmpl.Config.Base {
32+
absLocator, err := absPath(baseLocator.URL, basePath)
33+
if err != nil {
34+
return err
35+
}
36+
if i == 0 {
37+
// base can be either a single string (URL), or a single locator object, or a list whose first element can be either a string or an object
38+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.base | select(type == \"!!str\")) |= %q\n", absLocator))
39+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.base | select(type == \"!!map\") | .url) |= %q\n", absLocator))
40+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.base | select(type == \"!!seq\" and (.[0] | type) == \"!!str\") | .[0]) |= %q\n", absLocator))
41+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.base | select(type == \"!!seq\" and (.[0] | type) == \"!!map\") | .[0].url) |= %q\n", absLocator))
42+
} else {
43+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.base[%d] | select(type == \"!!str\")) |= %q\n", i, absLocator))
44+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.base[%d] | select(type == \"!!map\") | .url) |= %q\n", i, absLocator))
45+
}
46+
}
47+
for i, p := range tmpl.Config.Probes {
48+
if p.File != nil {
49+
absLocator, err := absPath(p.File.URL, basePath)
50+
if err != nil {
51+
return err
52+
}
53+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.probes[%d].file | select(type == \"!!str\")) = %q\n", i, absLocator))
54+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.probes[%d].file | select(type == \"!!map\") | .url) = %q\n", i, absLocator))
55+
}
56+
}
57+
for i, p := range tmpl.Config.Provision {
58+
if p.File != nil {
59+
absLocator, err := absPath(p.File.URL, basePath)
60+
if err != nil {
61+
return err
62+
}
63+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.provision[%d].file | select(type == \"!!str\")) = %q\n", i, absLocator))
64+
tmpl.expr.WriteString(fmt.Sprintf("| ($a.provision[%d].file | select(type == \"!!map\") | .url) = %q\n", i, absLocator))
65+
}
66+
}
67+
return tmpl.evalExpr()
68+
}
69+
70+
// withVolume adds the volume name of the current working directory to a path without volume name.
71+
// On Windows filepath.Abs() only returns a "rooted" name, but does not add the volume name.
72+
// withVolume also normalizes all path separators to the platform native one.
73+
func withVolume(path string) (string, error) {
74+
if runtime.GOOS == "windows" && filepath.VolumeName(path) == "" {
75+
root, err := filepath.Abs("/")
76+
if err != nil {
77+
return "", err
78+
}
79+
path = filepath.VolumeName(root) + path
80+
}
81+
return filepath.Clean(path), nil
82+
}
83+
84+
// basePath returns the locator in absolute format, but without the filename part.
85+
func basePath(locator string) (string, error) {
86+
u, err := url.Parse(locator)
87+
// Single-letter schemes will be drive names on Windows, e.g. "c:/foo"
88+
if err == nil && len(u.Scheme) > 1 {
89+
// path.Dir("") returns ".", which must be removed for url.JoinPath() to do the right thing later
90+
return u.Scheme + "://" + strings.TrimSuffix(path.Dir(path.Join(u.Host, u.Path)), "."), nil
91+
}
92+
base, err := filepath.Abs(filepath.Dir(locator))
93+
if err != nil {
94+
return "", err
95+
}
96+
return withVolume(base)
97+
}
98+
99+
// absPath either returns the locator directly, or combines it with the basePath if the locator is a relative path.
100+
func absPath(locator, basePath string) (string, error) {
101+
if locator == "" {
102+
return "", errors.New("locator is empty")
103+
}
104+
u, err := url.Parse(locator)
105+
if err == nil && len(u.Scheme) > 1 {
106+
return locator, nil
107+
}
108+
// Check for rooted locator; filepath.IsAbs() returns false on Windows when the volume name is missing
109+
volumeLen := len(filepath.VolumeName(locator))
110+
if locator[volumeLen] != '/' && locator[volumeLen] != filepath.Separator {
111+
switch {
112+
case basePath == "":
113+
return "", errors.New("basePath is empty")
114+
case basePath == "-":
115+
return "", errors.New("can't use relative paths when reading template from STDIN")
116+
case strings.Contains(locator, "../"):
117+
return "", fmt.Errorf("relative locator path %q must not contain '../' segments", locator)
118+
case volumeLen != 0:
119+
return "", fmt.Errorf("relative locator path %q must not include a volume name", locator)
120+
}
121+
u, err = url.Parse(basePath)
122+
if err != nil {
123+
return "", err
124+
}
125+
if len(u.Scheme) > 1 {
126+
return u.JoinPath(locator).String(), nil
127+
}
128+
locator = filepath.Join(basePath, locator)
129+
}
130+
return withVolume(locator)
131+
}

0 commit comments

Comments
 (0)