2019-04-26 16:28:22 -07:00
// Copyright 2018 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 commands
import (
2020-02-11 10:44:25 -08:00
"fmt"
2019-04-26 16:28:22 -07:00
"log"
"os"
"os/exec"
2019-04-30 13:08:54 -05:00
"github.com/google/ko/pkg/commands/options"
2019-04-26 16:28:22 -07:00
"github.com/spf13/cobra"
"github.com/spf13/pflag"
2020-02-11 10:44:25 -08:00
"golang.org/x/sync/errgroup"
2020-01-15 13:16:11 -08:00
"k8s.io/cli-runtime/pkg/genericclioptions"
2019-04-26 16:28:22 -07:00
)
// addCreate augments our CLI surface with apply.
func addCreate ( topLevel * cobra . Command ) {
koCreateFlags := [ ] string { }
2020-02-19 09:30:01 -08:00
po := & options . PublishOptions { }
2019-04-26 16:28:22 -07:00
fo := & options . FilenameOptions { }
2019-06-27 16:55:47 -04:00
so := & options . SelectorOptions { }
2019-07-10 01:11:20 -04:00
sto := & options . StrictOptions { }
2019-09-11 10:07:02 -07:00
bo := & options . BuildOptions { }
2019-04-26 16:28:22 -07:00
create := & cobra . Command {
Use : "create -f FILENAME" ,
Short : "Create the input files with image references resolved to built/pushed image digests." ,
Long : ` This sub-command finds import path references within the provided files, builds them into Go binaries, containerizes them, publishes them, and then feeds the resulting yaml into "kubectl create". ` ,
Example : `
# Build and publish import path references to a Docker
# Registry as :
# $ { KO_DOCKER_REPO } / < package name > - < hash of import path >
# Then , feed the resulting yaml into "kubectl create" .
# When KO_DOCKER_REPO is ko . local , it is the same as if
# -- local was passed .
ko create - f config /
# Build and publish import path references to a Docker
# Registry preserving import path names as :
# $ { KO_DOCKER_REPO } / < import path >
# Then , feed the resulting yaml into "kubectl create" .
ko create -- preserve - import - paths - f config /
# Build and publish import path references to a Docker
# daemon as :
# ko . local / < import path >
# Then , feed the resulting yaml into "kubectl create" .
ko create -- local - f config /
# Create from stdin :
cat config . yaml | ko create - f - ` ,
Args : cobra . NoArgs ,
Run : func ( cmd * cobra . Command , args [ ] string ) {
2020-01-16 13:27:43 -08:00
if ! isKubectlAvailable ( ) {
log . Print ( "error: kubectl is not available. kubectl must be installed to use ko create." )
return
}
2019-09-11 10:07:02 -07:00
builder , err := makeBuilder ( bo )
2019-04-30 13:08:54 -05:00
if err != nil {
log . Fatalf ( "error creating builder: %v" , err )
}
2020-02-19 09:30:01 -08:00
publisher , err := makePublisher ( po )
2019-04-30 13:08:54 -05:00
if err != nil {
log . Fatalf ( "error creating publisher: %v" , err )
}
2020-02-19 09:30:01 -08:00
defer publisher . Close ( )
2019-04-26 16:28:22 -07:00
// Create a set of ko-specific flags to ignore when passing through
// kubectl global flags.
ignoreSet := make ( map [ string ] struct { } )
for _ , s := range koCreateFlags {
ignoreSet [ s ] = struct { } { }
}
// Filter out ko flags from what we will pass through to kubectl.
kubectlFlags := [ ] string { }
cmd . Flags ( ) . Visit ( func ( flag * pflag . Flag ) {
if _ , ok := ignoreSet [ flag . Name ] ; ! ok {
kubectlFlags = append ( kubectlFlags , "--" + flag . Name , flag . Value . String ( ) )
}
} )
// Issue a "kubectl create" command reading from stdin,
// to which we will pipe the resolved files.
argv := [ ] string { "create" , "-f" , "-" }
argv = append ( argv , kubectlFlags ... )
kubectlCmd := exec . Command ( "kubectl" , argv ... )
// Pass through our environment
kubectlCmd . Env = os . Environ ( )
// Pass through our std{out,err} and make our resolved buffer stdin.
kubectlCmd . Stderr = os . Stderr
kubectlCmd . Stdout = os . Stdout
// Wire up kubectl stdin to resolveFilesToWriter.
stdin , err := kubectlCmd . StdinPipe ( )
if err != nil {
log . Fatalf ( "error piping to 'kubectl create': %v" , err )
}
2020-02-11 10:44:25 -08:00
// Cancel on signals.
ctx := createCancellableContext ( )
// Make sure builds are cancelled if kubectl create fails.
g , ctx := errgroup . WithContext ( ctx )
g . Go ( func ( ) error {
2019-04-26 16:28:22 -07:00
// kubectl buffers data before starting to create it, which
// can lead to resources being created more slowly than desired.
// In the case of --watch, it can lead to resources not being
// applied at all until enough iteration has occurred. To work
// around this, we prime the stream with a bunch of empty objects
// which kubectl will discard.
// See https://github.com/google/go-containerregistry/pull/348
for i := 0 ; i < 1000 ; i ++ {
stdin . Write ( [ ] byte ( "---\n" ) )
}
// Once primed kick things off.
2020-02-11 10:44:25 -08:00
return resolveFilesToWriter ( ctx , builder , publisher , fo , so , sto , stdin )
} )
g . Go ( func ( ) error {
// Run it.
if err := kubectlCmd . Run ( ) ; err != nil {
return fmt . Errorf ( "error executing 'kubectl create': %v" , err )
}
return nil
} )
2019-04-26 16:28:22 -07:00
2020-02-11 10:44:25 -08:00
if err := g . Wait ( ) ; err != nil {
log . Fatal ( err )
2019-04-26 16:28:22 -07:00
}
} ,
}
2020-02-19 09:30:01 -08:00
options . AddPublishArg ( create , po )
2019-04-26 16:28:22 -07:00
options . AddFileArg ( create , fo )
2019-06-27 16:55:47 -04:00
options . AddSelectorArg ( create , so )
2019-07-10 01:11:20 -04:00
options . AddStrictArg ( create , sto )
2019-09-11 10:07:02 -07:00
options . AddBuildOptions ( create , bo )
2019-04-26 16:28:22 -07:00
// Collect the ko-specific apply flags before registering the kubectl global
// flags so that we can ignore them when passing kubectl global flags through
// to kubectl.
create . Flags ( ) . VisitAll ( func ( flag * pflag . Flag ) {
koCreateFlags = append ( koCreateFlags , flag . Name )
} )
// Register the kubectl global flags.
2020-01-15 13:16:11 -08:00
kubeConfigFlags := genericclioptions . NewConfigFlags ( false )
2019-04-26 16:28:22 -07:00
kubeConfigFlags . AddFlags ( create . Flags ( ) )
topLevel . AddCommand ( create )
}