k3s/contrib/podex/podex.go

264 lines
7.2 KiB
Go
Raw Normal View History

/*
Copyright 2014 Google Inc. 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.
*/
// podex is a command line tool to bootstrap kubernetes container
2014-10-03 00:26:34 +00:00
// manifest from docker image metadata.
2014-10-02 08:29:54 +00:00
//
// Manifests can then be edited by a human to match deployment needs.
//
// Example usage:
//
// $ docker pull google/nodejs-hello
// $ podex -yaml google/nodejs-hello > google/nodejs-hello/pod.yaml
// $ podex -json google/nodejs-hello > google/nodejs-hello/pod.json
2014-10-02 08:16:07 +00:00
package main
import (
"bytes"
2014-10-02 08:16:07 +00:00
"encoding/json"
"flag"
"fmt"
"io"
2014-10-02 08:16:07 +00:00
"log"
"net/http"
2014-10-02 08:16:07 +00:00
"os"
"strconv"
"strings"
2014-12-01 04:12:37 +00:00
"github.com/ghodss/yaml"
goyaml "gopkg.in/yaml.v2"
2014-10-02 08:16:07 +00:00
)
const usage = "podex [-format=yaml|json] [-type=pod|container] [-id NAME] IMAGES..."
2014-10-02 08:16:07 +00:00
var manifestFormat = flag.String("format", "yaml", "manifest format to output, `yaml` or `json`")
var manifestType = flag.String("type", "pod", "manifest type to output, `pod` or `container`")
var manifestName = flag.String("name", "", "manifest name, default to image base name")
func init() {
flag.Usage = func() {
fmt.Fprintf(os.Stderr, "Usage: %s\n", usage)
flag.PrintDefaults()
}
}
2014-10-02 08:16:07 +00:00
type image struct {
Host string
Namespace string
Image string
Tag string
}
2014-10-02 08:16:07 +00:00
func main() {
flag.Parse()
if flag.NArg() < 1 {
flag.Usage()
log.Fatal("pod: missing image argument")
2014-10-02 08:16:07 +00:00
}
if *manifestName == "" {
2014-10-20 13:06:36 +00:00
if flag.NArg() > 1 {
flag.Usage()
log.Fatal("podex: -id arg is required when passing more than one image")
2014-10-20 13:06:36 +00:00
}
_, _, *manifestName, _ = splitDockerImageName(flag.Arg(0))
2014-10-02 08:16:07 +00:00
}
podContainers := []goyaml.MapSlice{}
2014-10-20 13:06:36 +00:00
for _, imageName := range flag.Args() {
host, namespace, repo, tag := splitDockerImageName(imageName)
container := goyaml.MapSlice{
{Key: "name", Value: repo},
{Key: "image", Value: imageName},
2014-10-20 13:06:36 +00:00
}
img, err := getImageMetadata(host, namespace, repo, tag)
2014-10-20 13:06:36 +00:00
if err != nil {
log.Fatalf("failed to get image metadata %q: %v", imageName, err)
2014-10-20 13:06:36 +00:00
}
portSlice := []goyaml.MapSlice{}
for p := range img.ContainerConfig.ExposedPorts {
2014-10-20 13:06:36 +00:00
port, err := strconv.Atoi(p.Port())
if err != nil {
log.Fatalf("failed to parse port %q: %v", p.Port(), err)
2014-10-20 13:06:36 +00:00
}
portEntry := goyaml.MapSlice{{
Key: "name",
Value: strings.Join([]string{repo, p.Proto(), p.Port()}, "-"),
}, {
Key: "containerPort",
Value: port,
}}
portSlice = append(portSlice, portEntry)
if p.Proto() != "tcp" {
portEntry = append(portEntry, goyaml.MapItem{Key: "protocol", Value: strings.ToUpper(p.Proto())})
}
}
if len(img.ContainerConfig.ExposedPorts) > 0 {
container = append(container, goyaml.MapItem{Key: "ports", Value: portSlice})
2014-10-20 13:06:36 +00:00
}
podContainers = append(podContainers, container)
2014-10-02 08:16:07 +00:00
}
2014-10-20 13:06:36 +00:00
2014-10-02 17:48:38 +00:00
// TODO(proppy): add flag to handle multiple version
containerManifest := goyaml.MapSlice{
{Key: "version", Value: "v1beta2"},
{Key: "containers", Value: podContainers},
2014-10-02 08:16:07 +00:00
}
2014-10-20 13:06:36 +00:00
var data interface{}
switch *manifestType {
case "container":
containerManifest = append(goyaml.MapSlice{
{Key: "id", Value: *manifestName},
}, containerManifest...)
data = containerManifest
case "pod":
data = goyaml.MapSlice{
{Key: "id", Value: *manifestName},
{Key: "kind", Value: "Pod"},
{Key: "apiVersion", Value: "v1beta1"},
{Key: "desiredState", Value: goyaml.MapSlice{
{Key: "manifest", Value: containerManifest},
}},
2014-10-02 08:16:07 +00:00
}
default:
flag.Usage()
log.Fatalf("unsupported manifest type %q", *manifestType)
2014-10-02 08:16:07 +00:00
}
yamlBytes, err := goyaml.Marshal(data)
if err != nil {
log.Fatalf("failed to marshal container manifest: %v", err)
}
switch *manifestFormat {
case "yaml":
os.Stdout.Write(yamlBytes)
case "json":
jsonBytes, err := yaml.YAMLToJSON(yamlBytes)
2014-10-02 08:16:07 +00:00
if err != nil {
log.Fatalf("failed to marshal container manifest into JSON: %v", err)
2014-10-02 08:16:07 +00:00
}
var jsonPretty bytes.Buffer
if err := json.Indent(&jsonPretty, jsonBytes, "", " "); err != nil {
log.Fatalf("failed to indent json %q: %v", string(jsonBytes), err)
}
io.Copy(os.Stdout, &jsonPretty)
default:
flag.Usage()
log.Fatalf("unsupported manifest format %q", *manifestFormat)
2014-10-02 08:16:07 +00:00
}
}
2014-10-20 13:06:36 +00:00
// splitDockerImageName split a docker image name of the form [HOST/][NAMESPACE/]REPOSITORY[:TAG]
func splitDockerImageName(imageName string) (host, namespace, repo, tag string) {
hostNamespaceImage := strings.Split(imageName, "/")
last := len(hostNamespaceImage) - 1
repoTag := strings.Split(hostNamespaceImage[last], ":")
repo = repoTag[0]
if len(repoTag) > 1 {
tag = repoTag[1]
}
switch len(hostNamespaceImage) {
case 2:
host = ""
namespace = hostNamespaceImage[0]
case 3:
host = hostNamespaceImage[0]
namespace = hostNamespaceImage[1]
}
2014-10-20 13:06:36 +00:00
return
}
type Port string
func (p Port) Port() string {
parts := strings.Split(string(p), "/")
return parts[0]
}
func (p Port) Proto() string {
parts := strings.Split(string(p), "/")
if len(parts) == 1 {
return "tcp"
}
return parts[1]
}
type imageMetadata struct {
ID string `json:"id"`
ContainerConfig struct {
ExposedPorts map[Port]struct{}
} `json:"container_config"`
}
func getImageMetadata(host, namespace, repo, tag string) (*imageMetadata, error) {
if host == "" {
host = "index.docker.io"
}
if namespace == "" {
namespace = "library"
}
if tag == "" {
tag = "latest"
}
req, err := http.NewRequest("GET", fmt.Sprintf("https://%s/v1/repositories/%s/%s/images", host, namespace, repo), nil)
if err != nil {
return nil, fmt.Errorf("error creating request: %v", err)
}
req.Header.Add("X-Docker-Token", "true")
resp, err := http.DefaultClient.Do(req)
if err != nil {
return nil, fmt.Errorf("error getting X-Docker-Token from index.docker.io: %v", err)
}
endpoints := resp.Header.Get("X-Docker-Endpoints")
token := resp.Header.Get("X-Docker-Token")
req, err = http.NewRequest("GET", fmt.Sprintf("https://%s/v1/repositories/%s/%s/tags/%s", endpoints, namespace, repo, tag), nil)
if err != nil {
return nil, fmt.Errorf("error creating request: %v", err)
}
req.Header.Add("Authorization", "Token "+token)
resp, err = http.DefaultClient.Do(req)
if err != nil {
return nil, fmt.Errorf("error getting image id for %s/%s:%s %v", namespace, repo, tag, err)
}
var imageID string
if err = json.NewDecoder(resp.Body).Decode(&imageID); err != nil {
return nil, fmt.Errorf("error decoding image id: %v", err)
}
req, err = http.NewRequest("GET", fmt.Sprintf("https://%s/v1/images/%s/json", endpoints, imageID), nil)
if err != nil {
return nil, fmt.Errorf("error creating request: %v", err)
}
req.Header.Add("Authorization", "Token "+token)
resp, err = http.DefaultClient.Do(req)
if err != nil {
return nil, fmt.Errorf("error getting json for image %q: %v", imageID, err)
}
var image imageMetadata
2015-01-09 12:13:25 +00:00
if err := json.NewDecoder(resp.Body).Decode(&image); err != nil {
return nil, fmt.Errorf("error decoding image %q metadata: %v", imageID, err)
}
return &image, nil
}