distrobuilder/sources/ubuntu-http.go
Stéphane Graber a7f1a578f8
Update for LXD Unpack change
Signed-off-by: Stéphane Graber <stgraber@ubuntu.com>
2019-02-03 09:10:53 +01:00

142 lines
3.3 KiB
Go

package sources
import (
"crypto/sha256"
"errors"
"fmt"
"io/ioutil"
"net/http"
"net/url"
"os"
"path"
"path/filepath"
"regexp"
"strings"
lxd "github.com/lxc/lxd/shared"
"github.com/lxc/distrobuilder/shared"
)
// UbuntuHTTP represents the Ubuntu HTTP downloader.
type UbuntuHTTP struct {
fname string
}
// NewUbuntuHTTP creates a new UbuntuHTTP instance.
func NewUbuntuHTTP() *UbuntuHTTP {
return &UbuntuHTTP{}
}
// Run downloads the tarball and unpacks it.
func (s *UbuntuHTTP) Run(definition shared.Definition, rootfsDir string) error {
baseURL := fmt.Sprintf("%s/releases/%s/release/", definition.Source.URL,
definition.Image.Release)
if strings.ContainsAny(definition.Image.Release, "0123456789") {
s.fname = fmt.Sprintf("ubuntu-base-%s-base-%s.tar.gz",
definition.Image.Release, definition.Image.ArchitectureMapped)
} else {
// if release is non-numerical, find the latest release
s.fname = getLatestRelease(definition.Source.URL,
definition.Image.Release, definition.Image.ArchitectureMapped)
if s.fname == "" {
return fmt.Errorf("Couldn't find latest release")
}
}
url, err := url.Parse(baseURL)
if err != nil {
return err
}
checksumFile := ""
// Force gpg checks when using http
if !definition.Source.SkipVerification && url.Scheme != "https" {
if len(definition.Source.Keys) == 0 {
return errors.New("GPG keys are required if downloading from HTTP")
}
checksumFile = baseURL + "SHA256SUMS"
shared.DownloadHash(baseURL+"SHA256SUMS.gpg", "", nil)
shared.DownloadHash(checksumFile, "", nil)
valid, err := shared.VerifyFile(
filepath.Join(os.TempDir(), "SHA256SUMS"),
filepath.Join(os.TempDir(), "SHA256SUMS.gpg"),
definition.Source.Keys,
definition.Source.Keyserver)
if err != nil {
return err
}
if !valid {
return fmt.Errorf("Failed to validate tarball")
}
}
err = shared.DownloadHash(baseURL+s.fname, checksumFile, sha256.New())
if err != nil {
return fmt.Errorf("Error downloading Ubuntu image: %s", err)
}
err = s.unpack(filepath.Join(os.TempDir(), s.fname), rootfsDir)
if err != nil {
return err
}
if definition.Source.AptSources != "" {
// Run the template
out, err := shared.RenderTemplate(definition.Source.AptSources, definition)
if err != nil {
return err
}
// Append final new line if missing
if !strings.HasSuffix(out, "\n") {
out += "\n"
}
// Replace content of sources.list with the templated content.
file, err := os.Create(filepath.Join(rootfsDir, "etc", "apt", "sources.list"))
if err != nil {
return err
}
defer file.Close()
file.WriteString(out)
}
return nil
}
func (s UbuntuHTTP) unpack(filePath, rootDir string) error {
os.RemoveAll(rootDir)
os.MkdirAll(rootDir, 0755)
err := lxd.Unpack(filePath, rootDir, false, false, nil)
if err != nil {
return fmt.Errorf("Failed to unpack tarball: %s", err)
}
return nil
}
func getLatestRelease(URL, release, arch string) string {
resp, err := http.Get(URL + path.Join("/", "releases", release, "release"))
if err != nil {
fmt.Fprintln(os.Stderr, err)
return ""
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
regex := regexp.MustCompile(fmt.Sprintf("ubuntu-base-\\d{2}\\.\\d{2}(\\.\\d+)?-base-%s.tar.gz", arch))
releases := regex.FindAllString(string(body), -1)
if len(releases) > 1 {
return string(releases[len(releases)-1])
}
return ""
}