1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
|
/*
Copyright 2019 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package nodeimage
import (
"io"
"sigs.k8s.io/kind/pkg/errors"
"sigs.k8s.io/kind/pkg/exec"
)
type containerdImporter struct {
containerCmder exec.Cmder
}
func newContainerdImporter(containerCmder exec.Cmder) *containerdImporter {
return &containerdImporter{
containerCmder: containerCmder,
}
}
func (c *containerdImporter) Prepare() error {
if err := c.containerCmder.Command(
"bash", "-c", "nohup containerd > /dev/null 2>&1 &",
).Run(); err != nil {
return err
}
return nil
}
func (c *containerdImporter) WaitForReady() error {
// ctr doesn't respect timeouts when the socket doesn't exist
// so we'll look for the socket to exist ourselves, THEN attempt ctr info
// TODO: we are assuming the socket path, and this is kind of hacky
if err := c.containerCmder.Command(
"bash", "-c", `set -e
# wait for socket to exist
for i in {0..3}; do
if [ -S /run/containerd/containerd.sock ]; then
break
fi
sleep "$i"
done
# check healthy
ctr info
`,
).Run(); err != nil {
return errors.Wrap(err, "failed to wait for containerd to become ready")
}
return nil
}
func (c *containerdImporter) End() error {
return c.containerCmder.Command("pkill", "containerd").Run()
}
func (c *containerdImporter) Pull(image, platform string) error {
return c.containerCmder.Command(
"ctr", "--namespace=k8s.io", "content", "fetch", "--platform="+platform, image,
).SetStdout(io.Discard).SetStderr(io.Discard).Run()
}
func (c *containerdImporter) LoadCommand() exec.Cmd {
return c.containerCmder.Command(
// TODO: ideally we do not need this in the future. we have fixed at least one image
"ctr", "--namespace=k8s.io", "images", "import", "--label=io.cri-containerd.pinned=pinned", "--all-platforms", "--no-unpack", "--digests", "-",
)
}
func (c *containerdImporter) Tag(src, target string) error {
return c.containerCmder.Command(
"ctr", "--namespace=k8s.io", "images", "tag", "--force", src, target,
).Run()
}
func (c *containerdImporter) ListImported() ([]string, error) {
return exec.OutputLines(c.containerCmder.Command("ctr", "--namespace=k8s.io", "images", "list", "-q"))
}
|