1
0
mirror of https://github.com/ko-build/ko.git synced 2025-02-01 19:14:40 +02:00

Use working directory and build config dir (#427)

* Use working directory and build config `dir`

Use the working directory from `BuildOptions` to load `.ko.yaml`.

Also, use the `dir` build config field to load package information,
instead of assuming that `go.mod` is in the current working directory.

This removes the `init()` function from `./pkg/commands/config.go`.

And avoids the global viper instance, which caused some Heisenbugs (and
associated hair loss).

Fixes: #422, #424

* Return error instead of log.Fatal

`log.Fatal` is no longer needed in `loadConfig()`, since it's no longer
an `init()` function.

Also removed `log.Fatal` from `createBuildConfigMap()`.
This commit is contained in:
Halvard Skogsrud 2021-08-28 02:55:39 +10:00 committed by GitHub
parent 29fccb0832
commit 1fcfd75417
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 115 additions and 37 deletions

View File

@ -41,6 +41,11 @@ import (
"golang.org/x/tools/go/packages"
)
const (
// configDefaultBaseImage is the default base image if not specified in .ko.yaml.
configDefaultBaseImage = "gcr.io/distroless/static:nonroot"
)
var (
defaultBaseImage string
baseImageOverrides map[string]string
@ -168,8 +173,8 @@ func createCancellableContext() context.Context {
return ctx
}
func createBuildConfigs(baseDir string, configs []build.Config) map[string]build.Config {
buildConfigs = make(map[string]build.Config)
func createBuildConfigMap(workingDirectory string, configs []build.Config) (map[string]build.Config, error) {
buildConfigsByImportPath := make(map[string]build.Config)
for i, config := range configs {
// Make sure to behave like GoReleaser by defaulting to the current
// directory in case the build or main field is not set, check
@ -181,74 +186,80 @@ func createBuildConfigs(baseDir string, configs []build.Config) map[string]build
config.Main = "."
}
// To behave like GoReleaser, check whether the configured path points to a
// source file, and if so, just use the directory it is in
var path string
if fi, err := os.Stat(filepath.Join(baseDir, config.Dir, config.Main)); err == nil && fi.Mode().IsRegular() {
path = filepath.Dir(filepath.Join(config.Dir, config.Main))
// baseDir is the directory where `go list` will be run to look for package information
baseDir := filepath.Join(workingDirectory, config.Dir)
} else {
path = filepath.Join(config.Dir, config.Main)
// To behave like GoReleaser, check whether the configured `main` config value points to a
// source file, and if so, just use the directory it is in
path := config.Main
if fi, err := os.Stat(filepath.Join(baseDir, config.Main)); err == nil && fi.Mode().IsRegular() {
path = filepath.Dir(config.Main)
}
// By default, paths configured in the builds section are considered
// local import paths, therefore add a "./" equivalent as a prefix to
// the constructured import path
importPath := fmt.Sprint(".", string(filepath.Separator), path)
localImportPath := fmt.Sprint(".", string(filepath.Separator), path)
pkgs, err := packages.Load(&packages.Config{Mode: packages.NeedName, Dir: baseDir}, importPath)
pkgs, err := packages.Load(&packages.Config{Mode: packages.NeedName, Dir: baseDir}, localImportPath)
if err != nil {
log.Fatalf("'builds': entry #%d does not contain a usuable path (%s): %v", i, importPath, err)
return nil, fmt.Errorf("'builds': entry #%d does not contain a valid local import path (%s) for directory (%s): %v", i, localImportPath, baseDir, err)
}
if len(pkgs) != 1 {
log.Fatalf("'builds': entry #%d results in %d local packages, only 1 is expected", i, len(pkgs))
return nil, fmt.Errorf("'builds': entry #%d results in %d local packages, only 1 is expected", i, len(pkgs))
}
importPath := pkgs[0].PkgPath
buildConfigsByImportPath[importPath] = config
}
importPath = pkgs[0].PkgPath
buildConfigs[importPath] = config
return buildConfigsByImportPath, nil
}
return buildConfigs
// loadConfig reads build configuration from defaults, environment variables, and the `.ko.yaml` config file.
func loadConfig(workingDirectory string) error {
v := viper.New()
if workingDirectory == "" {
workingDirectory = "."
}
func init() {
// If omitted, use this base image.
viper.SetDefault("defaultBaseImage", "gcr.io/distroless/static:nonroot")
viper.SetConfigName(".ko") // .yaml is implicit
viper.SetEnvPrefix("KO")
viper.AutomaticEnv()
v.SetDefault("defaultBaseImage", configDefaultBaseImage)
v.SetConfigName(".ko") // .yaml is implicit
v.SetEnvPrefix("KO")
v.AutomaticEnv()
if override := os.Getenv("KO_CONFIG_PATH"); override != "" {
viper.AddConfigPath(override)
v.AddConfigPath(override)
}
viper.AddConfigPath("./")
v.AddConfigPath(workingDirectory)
if err := viper.ReadInConfig(); err != nil {
if err := v.ReadInConfig(); err != nil {
if _, ok := err.(viper.ConfigFileNotFoundError); !ok {
log.Fatalf("error reading config file: %v", err)
return fmt.Errorf("error reading config file: %v", err)
}
}
ref := viper.GetString("defaultBaseImage")
ref := v.GetString("defaultBaseImage")
if _, err := name.ParseReference(ref); err != nil {
log.Fatalf("'defaultBaseImage': error parsing %q as image reference: %v", ref, err)
return fmt.Errorf("'defaultBaseImage': error parsing %q as image reference: %v", ref, err)
}
defaultBaseImage = ref
baseImageOverrides = make(map[string]string)
overrides := viper.GetStringMapString("baseImageOverrides")
for k, v := range overrides {
if _, err := name.ParseReference(v); err != nil {
log.Fatalf("'baseImageOverrides': error parsing %q as image reference: %v", v, err)
overrides := v.GetStringMapString("baseImageOverrides")
for key, value := range overrides {
if _, err := name.ParseReference(value); err != nil {
return fmt.Errorf("'baseImageOverrides': error parsing %q as image reference: %v", value, err)
}
baseImageOverrides[k] = v
baseImageOverrides[key] = value
}
var builds []build.Config
if err := viper.UnmarshalKey("builds", &builds); err != nil {
log.Fatalf("configuration section 'builds' cannot be parsed")
if err := v.UnmarshalKey("builds", &builds); err != nil {
return fmt.Errorf("configuration section 'builds' cannot be parsed")
}
buildConfigs = createBuildConfigs(".", builds)
var err error
buildConfigs, err = createBuildConfigMap(workingDirectory, builds)
return err
}

View File

@ -59,6 +59,37 @@ func TestOverrideDefaultBaseImageUsingBuildOption(t *testing.T) {
}
}
// TestDefaultBaseImage is a canary-type test for ensuring that config has been read when creating a builder.
func TestDefaultBaseImage(t *testing.T) {
_, err := NewBuilder(context.Background(), &options.BuildOptions{
WorkingDirectory: "testdata/config",
})
if err != nil {
t.Fatal(err)
}
wantDefaultBaseImage := "gcr.io/distroless/base:nonroot" // matches value in ./testdata/.ko.yaml
if defaultBaseImage != wantDefaultBaseImage {
t.Fatalf("wanted defaultBaseImage %s, got %s", wantDefaultBaseImage, defaultBaseImage)
}
}
func TestBuildConfigWithWorkingDirectoryAndDirAndMain(t *testing.T) {
_, err := NewBuilder(context.Background(), &options.BuildOptions{
WorkingDirectory: "testdata/paths",
})
if err != nil {
t.Fatalf("NewBuilder(): %+v", err)
}
if len(buildConfigs) != 1 {
t.Fatalf("expected 1 build config, got %d", len(buildConfigs))
}
expectedImportPath := "example.com/testapp/cmd/foo" // module from app/go.mod + `main` from .ko.yaml
if _, exists := buildConfigs[expectedImportPath]; !exists {
t.Fatalf("expected build config for import path [%s], got %+v", expectedImportPath, buildConfigs)
}
}
func TestCreateBuildConfigs(t *testing.T) {
compare := func(expected string, actual string) {
if expected != actual {
@ -75,7 +106,11 @@ func TestCreateBuildConfigs(t *testing.T) {
}
for _, b := range buildConfigs {
for importPath, buildCfg := range createBuildConfigs("../..", []build.Config{b}) {
buildConfigMap, err := createBuildConfigMap("../..", []build.Config{b})
if err != nil {
t.Fatal(err)
}
for importPath, buildCfg := range buildConfigMap {
switch buildCfg.ID {
case "defaults":
compare("github.com/google/ko", importPath)

View File

@ -122,6 +122,9 @@ func NewBuilder(ctx context.Context, bo *options.BuildOptions) (build.Interface,
}
func makeBuilder(ctx context.Context, bo *options.BuildOptions) (*build.Caching, error) {
if err := loadConfig(bo.WorkingDirectory); err != nil {
return nil, err
}
opt, err := gobuildOptions(bo)
if err != nil {
return nil, fmt.Errorf("error setting up builder options: %v", err)

1
pkg/commands/testdata/config/.ko.yaml vendored Normal file
View File

@ -0,0 +1 @@
defaultBaseImage: gcr.io/distroless/base:nonroot

4
pkg/commands/testdata/paths/.ko.yaml vendored Normal file
View File

@ -0,0 +1,4 @@
builds:
- id: app-with-main-package-in-different-directory-to-go-mod-and-ko-yaml
dir: ./app
main: ./cmd/foo

View File

@ -0,0 +1,21 @@
// Copyright 2021 Google LLC All Rights Reserved.
//
// 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 main
import "fmt"
func main() {
fmt.Println("cmd/foo")
}

View File

@ -0,0 +1,3 @@
module example.com/testapp
go 1.15